|  | <!doctype html> | 
|  | <title>Range mutation tests</title> | 
|  | <link rel="author" title="Aryeh Gregor" href=ayg@aryeh.name> | 
|  | <meta name=timeout content=long> | 
|  |  | 
|  | <div id=log></div> | 
|  | <script src=/resources/testharness.js></script> | 
|  | <script src=/resources/testharnessreport.js></script> | 
|  | <script src=../common.js></script> | 
|  | <script> | 
|  | "use strict"; | 
|  |  | 
|  | // These tests probably use too much abstraction and too little copy-paste. | 
|  | // Reader beware. | 
|  | // | 
|  | // TODO: | 
|  | // | 
|  | // * Lots and lots and lots more different types of ranges | 
|  | // * insertBefore() with DocumentFragments | 
|  | // * Fill out other insert/remove tests | 
|  | // * normalize() (https://www.w3.org/Bugs/Public/show_bug.cgi?id=13843) | 
|  |  | 
|  | // Give a textual description of the range we're testing, for the test names. | 
|  | function describeRange(startContainer, startOffset, endContainer, endOffset) { | 
|  | if (startContainer == endContainer && startOffset == endOffset) { | 
|  | return "range collapsed at (" + startContainer + ", " + startOffset + ")"; | 
|  | } else if (startContainer == endContainer) { | 
|  | return "range on " + startContainer + " from " + startOffset + " to " + endOffset; | 
|  | } else { | 
|  | return "range from (" + startContainer + ", " + startOffset + ") to (" + endContainer + ", " + endOffset + ")"; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Lists of the various types of nodes we'll want to use.  We use strings that | 
|  | // we can later eval(), so that we can produce legible test names. | 
|  | var textNodes = [ | 
|  | "paras[0].firstChild", | 
|  | "paras[1].firstChild", | 
|  | "foreignTextNode", | 
|  | "xmlTextNode", | 
|  | "detachedTextNode", | 
|  | "detachedForeignTextNode", | 
|  | "detachedXmlTextNode", | 
|  | ]; | 
|  | var commentNodes = [ | 
|  | "comment", | 
|  | "foreignComment", | 
|  | "xmlComment", | 
|  | "detachedComment", | 
|  | "detachedForeignComment", | 
|  | "detachedXmlComment", | 
|  | ]; | 
|  | var characterDataNodes = textNodes.concat(commentNodes); | 
|  |  | 
|  | // This function is slightly scary, but it works well enough, so . . . | 
|  | // sourceTests is an array of test data that will be altered in mysterious ways | 
|  | // before being passed off to doTest, descFn is something that takes an element | 
|  | // of sourceTests and produces the first part of a human-readable description | 
|  | // of the test, testFn is the function that doTest will call to do the actual | 
|  | // work and tell it what results to expect. | 
|  | function doTests(sourceTests, descFn, testFn) { | 
|  | var tests = []; | 
|  | for (var i = 0; i < sourceTests.length; i++) { | 
|  | var params = sourceTests[i]; | 
|  | var len = params.length; | 
|  | tests.push([ | 
|  | descFn(params) + ", with unselected " + describeRange(params[len - 4], params[len - 3], params[len - 2], params[len - 1]), | 
|  | // The closure here ensures that the params that testFn get are the | 
|  | // current version of params, not the version from the last | 
|  | // iteration of this loop.  We test that none of the parameters | 
|  | // evaluate to undefined to catch bugs in our eval'ing, like | 
|  | // mistyping a property name. | 
|  | function(params) { return function() { | 
|  | var evaledParams = params.map(eval); | 
|  | for (var i = 0; i < evaledParams.length; i++) { | 
|  | assert_true(typeof evaledParams[i] != "undefined", | 
|  | "Test bug: " + params[i] + " is undefined"); | 
|  | } | 
|  | return testFn.apply(null, evaledParams); | 
|  | } }(params), | 
|  | false, | 
|  | params[len - 4], | 
|  | params[len - 3], | 
|  | params[len - 2], | 
|  | params[len - 1] | 
|  | ]); | 
|  | tests.push([ | 
|  | descFn(params) + ", with selected " + describeRange(params[len - 4], params[len - 3], params[len - 2], params[len - 1]), | 
|  | function(params) { return function() { | 
|  | var evaledParams = params.map(eval); | 
|  | for (var i = 0; i < evaledParams.length; i++) { | 
|  | assert_true(typeof evaledParams[i] != "undefined", | 
|  | "Test bug: " + params[i] + " is undefined"); | 
|  | } | 
|  | return testFn.apply(null, evaledParams); | 
|  | } }(params), | 
|  | true, | 
|  | params[len - 4], | 
|  | params[len - 3], | 
|  | params[len - 2], | 
|  | params[len - 1] | 
|  | ]); | 
|  | } | 
|  | generate_tests(doTest, tests); | 
|  | } | 
|  |  | 
|  | // Set up the range, call the callback function to do the DOM modification and | 
|  | // tell us what to expect.  The callback function needs to return a | 
|  | // four-element array with the expected start/end containers/offsets, and | 
|  | // receives no arguments.  useSelection tells us whether the Range should be | 
|  | // added to a Selection and the Selection tested to ensure that the mutation | 
|  | // affects user selections as well as other ranges; every test is run with this | 
|  | // both false and true, because when it's set to true WebKit and Opera fail all | 
|  | // tests' sanity checks, which is unhelpful.  The last four parameters just | 
|  | // tell us what range to build. | 
|  | function doTest(callback, useSelection, startContainer, startOffset, endContainer, endOffset) { | 
|  | // Recreate all the test nodes in case they were altered by the last test | 
|  | // run. | 
|  | setupRangeTests(); | 
|  | startContainer = eval(startContainer); | 
|  | startOffset = eval(startOffset); | 
|  | endContainer = eval(endContainer); | 
|  | endOffset = eval(endOffset); | 
|  |  | 
|  | var ownerDoc = startContainer.nodeType == Node.DOCUMENT_NODE | 
|  | ? startContainer | 
|  | : startContainer.ownerDocument; | 
|  | var range = ownerDoc.createRange(); | 
|  | range.setStart(startContainer, startOffset); | 
|  | range.setEnd(endContainer, endOffset); | 
|  |  | 
|  | if (useSelection) { | 
|  | getSelection().removeAllRanges(); | 
|  | getSelection().addRange(range); | 
|  |  | 
|  | assert_equals(getSelection().rangeCount, 1, | 
|  | "Sanity check: selection must have exactly one range after adding the range"); | 
|  | assert_equals(getSelection().getRangeAt(0), range, | 
|  | "Sanity check: selection's range must initially be the same as the range we added"); | 
|  | assert_equals(range.startContainer, startContainer, | 
|  | "Sanity check: range's startContainer must initially be the one we set"); | 
|  | assert_equals(range.endContainer, endContainer, | 
|  | "Sanity check: range's endContainer must initially be the one we set"); | 
|  | assert_equals(range.startOffset, startOffset, | 
|  | "Sanity check: range's startOffset must initially be the one we set"); | 
|  | assert_equals(range.endOffset, endOffset, | 
|  | "Sanity check: range's endOffset must initially be the one we set"); | 
|  | } | 
|  |  | 
|  | var expected = callback(); | 
|  |  | 
|  | if (useSelection) { | 
|  | assert_equals(getSelection().getRangeAt(0), range, | 
|  | "The range we added must not be removed from the selection"); | 
|  | } | 
|  | assert_equals(range.startContainer, expected[0], | 
|  | "Wrong start container"); | 
|  | assert_equals(range.startOffset, expected[1], | 
|  | "Wrong start offset"); | 
|  | assert_equals(range.endContainer, expected[2], | 
|  | "Wrong end container"); | 
|  | assert_equals(range.endOffset, expected[3], | 
|  | "Wrong end offset"); | 
|  | } | 
|  |  | 
|  |  | 
|  | // Now we get to the specific tests. | 
|  |  | 
|  | function testSplitText(oldNode, offset, startContainer, startOffset, endContainer, endOffset) { | 
|  | // Save these for later | 
|  | var originalStartOffset = startOffset; | 
|  | var originalEndOffset = endOffset; | 
|  | var originalLength = oldNode.length; | 
|  |  | 
|  | var newNode; | 
|  | try { | 
|  | newNode = oldNode.splitText(offset); | 
|  | } catch (e) { | 
|  | // Should only happen if offset is negative | 
|  | return [startContainer, startOffset, endContainer, endOffset]; | 
|  | } | 
|  |  | 
|  | // First we adjust for replacing data: | 
|  | // | 
|  | // "Replace data with offset offset, count count, and data the empty | 
|  | // string." | 
|  | // | 
|  | // That translates to offset = offset, count = originalLength - offset, | 
|  | // data = "".  node is oldNode. | 
|  | // | 
|  | // "For every boundary point whose node is node, and whose offset is | 
|  | // greater than offset but less than or equal to offset plus count, set its | 
|  | // offset to offset." | 
|  | if (startContainer == oldNode | 
|  | && startOffset > offset | 
|  | && startOffset <= originalLength) { | 
|  | startOffset = offset; | 
|  | } | 
|  |  | 
|  | if (endContainer == oldNode | 
|  | && endOffset > offset | 
|  | && endOffset <= originalLength) { | 
|  | endOffset = offset; | 
|  | } | 
|  |  | 
|  | // "For every boundary point whose node is node, and whose offset is | 
|  | // greater than offset plus count, add the length of data to its offset, | 
|  | // then subtract count from it." | 
|  | // | 
|  | // Can't happen: offset plus count is originalLength. | 
|  |  | 
|  | // Now we insert a node, if oldNode's parent isn't null: "For each boundary | 
|  | // point whose node is the new parent of the affected node and whose offset | 
|  | // is greater than the new index of the affected node, add one to the | 
|  | // boundary point's offset." | 
|  | if (startContainer == oldNode.parentNode | 
|  | && startOffset > 1 + indexOf(oldNode)) { | 
|  | startOffset++; | 
|  | } | 
|  |  | 
|  | if (endContainer == oldNode.parentNode | 
|  | && endOffset > 1 + indexOf(oldNode)) { | 
|  | endOffset++; | 
|  | } | 
|  |  | 
|  | // Finally, the splitText stuff itself: | 
|  | // | 
|  | // "If parent is not null, run these substeps: | 
|  | // | 
|  | //   * "For each range whose start node is node and start offset is greater | 
|  | //   than offset, set its start node to new node and decrease its start | 
|  | //   offset by offset. | 
|  | // | 
|  | //   * "For each range whose end node is node and end offset is greater | 
|  | //   than offset, set its end node to new node and decrease its end offset | 
|  | //   by offset. | 
|  | // | 
|  | //   * "For each range whose start node is parent and start offset is equal | 
|  | //   to the index of node + 1, increase its start offset by one. | 
|  | // | 
|  | //   * "For each range whose end node is parent and end offset is equal to | 
|  | //   the index of node + 1, increase its end offset by one." | 
|  | if (oldNode.parentNode) { | 
|  | if (startContainer == oldNode && originalStartOffset > offset) { | 
|  | startContainer = newNode; | 
|  | startOffset = originalStartOffset - offset; | 
|  | } | 
|  |  | 
|  | if (endContainer == oldNode && originalEndOffset > offset) { | 
|  | endContainer = newNode; | 
|  | endOffset = originalEndOffset - offset; | 
|  | } | 
|  |  | 
|  | if (startContainer == oldNode.parentNode | 
|  | && startOffset == 1 + indexOf(oldNode)) { | 
|  | startOffset++; | 
|  | } | 
|  |  | 
|  | if (endContainer == oldNode.parentNode | 
|  | && endOffset == 1 + indexOf(oldNode)) { | 
|  | endOffset++; | 
|  | } | 
|  | } | 
|  |  | 
|  | return [startContainer, startOffset, endContainer, endOffset]; | 
|  | } | 
|  |  | 
|  | // The offset argument is unsigned, so per WebIDL -1 should wrap to 4294967295, | 
|  | // which is probably longer than the length, so it should throw an exception. | 
|  | // This is no different from the other cases where the offset is longer than | 
|  | // the length, and the wrapping complicates my testing slightly, so I won't | 
|  | // bother testing negative values here or in other cases. | 
|  | var splitTextTests = []; | 
|  | for (var i = 0; i < textNodes.length; i++) { | 
|  | var node = textNodes[i]; | 
|  | splitTextTests.push([node, 376, node, 0, node, 1]); | 
|  | splitTextTests.push([node, 0, node, 0, node, 0]); | 
|  | splitTextTests.push([node, 1, node, 1, node, 1]); | 
|  | splitTextTests.push([node, node + ".length", node, node + ".length", node, node + ".length"]); | 
|  | splitTextTests.push([node, 1, node, 1, node, 3]); | 
|  | splitTextTests.push([node, 2, node, 1, node, 3]); | 
|  | splitTextTests.push([node, 3, node, 1, node, 3]); | 
|  | } | 
|  |  | 
|  | splitTextTests.push( | 
|  | ["paras[0].firstChild", 1, "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", 1, "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 2, "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 3, "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 2, "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 3, "paras[0]", 0, "paras[0].firstChild", 3] | 
|  | ); | 
|  |  | 
|  |  | 
|  | function testReplaceDataAlgorithm(node, offset, count, data, callback, startContainer, startOffset, endContainer, endOffset) { | 
|  | // Mutation works the same any time DOM Core's "replace data" algorithm is | 
|  | // invoked.  node, offset, count, data are as in that algorithm.  The | 
|  | // callback is what does the actual setting.  Not to be confused with | 
|  | // testReplaceData, which tests the replaceData() method. | 
|  |  | 
|  | // Barring any provision to the contrary, the containers and offsets must | 
|  | // not change. | 
|  | var expectedStartContainer = startContainer; | 
|  | var expectedStartOffset = startOffset; | 
|  | var expectedEndContainer = endContainer; | 
|  | var expectedEndOffset = endOffset; | 
|  |  | 
|  | var originalParent = node.parentNode; | 
|  | var originalData = node.data; | 
|  |  | 
|  | var exceptionThrown = false; | 
|  | try { | 
|  | callback(); | 
|  | } catch (e) { | 
|  | // Should only happen if offset is greater than length | 
|  | exceptionThrown = true; | 
|  | } | 
|  |  | 
|  | assert_equals(node.parentNode, originalParent, | 
|  | "Sanity check failed: changing data changed the parent"); | 
|  |  | 
|  | // "User agents must run the following steps whenever they replace data of | 
|  | // a CharacterData node, as though they were written in the specification | 
|  | // for that algorithm after all other steps. In particular, the steps must | 
|  | // not be executed if the algorithm threw an exception." | 
|  | if (exceptionThrown) { | 
|  | assert_equals(node.data, originalData, | 
|  | "Sanity check failed: exception thrown but data changed"); | 
|  | } else { | 
|  | assert_equals(node.data, | 
|  | originalData.substr(0, offset) + data + originalData.substr(offset + count), | 
|  | "Sanity check failed: data not changed as expected"); | 
|  | } | 
|  |  | 
|  | // "For every boundary point whose node is node, and whose offset is | 
|  | // greater than offset but less than or equal to offset plus count, set | 
|  | // its offset to offset." | 
|  | if (!exceptionThrown | 
|  | && startContainer == node | 
|  | && startOffset > offset | 
|  | && startOffset <= offset + count) { | 
|  | expectedStartOffset = offset; | 
|  | } | 
|  |  | 
|  | if (!exceptionThrown | 
|  | && endContainer == node | 
|  | && endOffset > offset | 
|  | && endOffset <= offset + count) { | 
|  | expectedEndOffset = offset; | 
|  | } | 
|  |  | 
|  | // "For every boundary point whose node is node, and whose offset is | 
|  | // greater than offset plus count, add the length of data to its offset, | 
|  | // then subtract count from it." | 
|  | if (!exceptionThrown | 
|  | && startContainer == node | 
|  | && startOffset > offset + count) { | 
|  | expectedStartOffset += data.length - count; | 
|  | } | 
|  |  | 
|  | if (!exceptionThrown | 
|  | && endContainer == node | 
|  | && endOffset > offset + count) { | 
|  | expectedEndOffset += data.length - count; | 
|  | } | 
|  |  | 
|  | return [expectedStartContainer, expectedStartOffset, expectedEndContainer, expectedEndOffset]; | 
|  | } | 
|  |  | 
|  | function testInsertData(node, offset, data, startContainer, startOffset, endContainer, endOffset) { | 
|  | return testReplaceDataAlgorithm(node, offset, 0, data, | 
|  | function() { node.insertData(offset, data) }, | 
|  | startContainer, startOffset, endContainer, endOffset); | 
|  | } | 
|  |  | 
|  | var insertDataTests = []; | 
|  | for (var i = 0; i < characterDataNodes.length; i++) { | 
|  | var node = characterDataNodes[i]; | 
|  | insertDataTests.push([node, 376, '"foo"', node, 0, node, 1]); | 
|  | insertDataTests.push([node, 0, '"foo"', node, 0, node, 0]); | 
|  | insertDataTests.push([node, 1, '"foo"', node, 1, node, 1]); | 
|  | insertDataTests.push([node, node + ".length", '"foo"', node, node + ".length", node, node + ".length"]); | 
|  | insertDataTests.push([node, 1, '"foo"', node, 1, node, 3]); | 
|  | insertDataTests.push([node, 2, '"foo"', node, 1, node, 3]); | 
|  | insertDataTests.push([node, 3, '"foo"', node, 1, node, 3]); | 
|  |  | 
|  | insertDataTests.push([node, 376, '""', node, 0, node, 1]); | 
|  | insertDataTests.push([node, 0, '""', node, 0, node, 0]); | 
|  | insertDataTests.push([node, 1, '""', node, 1, node, 1]); | 
|  | insertDataTests.push([node, node + ".length", '""', node, node + ".length", node, node + ".length"]); | 
|  | insertDataTests.push([node, 1, '""', node, 1, node, 3]); | 
|  | insertDataTests.push([node, 2, '""', node, 1, node, 3]); | 
|  | insertDataTests.push([node, 3, '""', node, 1, node, 3]); | 
|  | } | 
|  |  | 
|  | insertDataTests.push( | 
|  | ["paras[0].firstChild", 1, '"foo"', "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", 1, '"foo"', "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, '"foo"', "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 2, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 3, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 2, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 3, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3] | 
|  | ); | 
|  |  | 
|  |  | 
|  | function testAppendData(node, data, startContainer, startOffset, endContainer, endOffset) { | 
|  | return testReplaceDataAlgorithm(node, node.length, 0, data, | 
|  | function() { node.appendData(data) }, | 
|  | startContainer, startOffset, endContainer, endOffset); | 
|  | } | 
|  |  | 
|  | var appendDataTests = []; | 
|  | for (var i = 0; i < characterDataNodes.length; i++) { | 
|  | var node = characterDataNodes[i]; | 
|  | appendDataTests.push([node, '"foo"', node, 0, node, 1]); | 
|  | appendDataTests.push([node, '"foo"', node, 0, node, 0]); | 
|  | appendDataTests.push([node, '"foo"', node, 1, node, 1]); | 
|  | appendDataTests.push([node, '"foo"', node, 0, node, node + ".length"]); | 
|  | appendDataTests.push([node, '"foo"', node, 1, node, node + ".length"]); | 
|  | appendDataTests.push([node, '"foo"', node, node + ".length", node, node + ".length"]); | 
|  | appendDataTests.push([node, '"foo"', node, 1, node, 3]); | 
|  |  | 
|  | appendDataTests.push([node, '""', node, 0, node, 1]); | 
|  | appendDataTests.push([node, '""', node, 0, node, 0]); | 
|  | appendDataTests.push([node, '""', node, 1, node, 1]); | 
|  | appendDataTests.push([node, '""', node, 0, node, node + ".length"]); | 
|  | appendDataTests.push([node, '""', node, 1, node, node + ".length"]); | 
|  | appendDataTests.push([node, '""', node, node + ".length", node, node + ".length"]); | 
|  | appendDataTests.push([node, '""', node, 1, node, 3]); | 
|  | } | 
|  |  | 
|  | appendDataTests.push( | 
|  | ["paras[0].firstChild", '""', "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", '""', "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", '""', "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", '""', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", '""', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  |  | 
|  | ["paras[0].firstChild", '"foo"', "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", '"foo"', "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", '"foo"', "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", '"foo"', "paras[0]", 0, "paras[0].firstChild", 3] | 
|  | ); | 
|  |  | 
|  |  | 
|  | function testDeleteData(node, offset, count, startContainer, startOffset, endContainer, endOffset) { | 
|  | return testReplaceDataAlgorithm(node, offset, count, "", | 
|  | function() { node.deleteData(offset, count) }, | 
|  | startContainer, startOffset, endContainer, endOffset); | 
|  | } | 
|  |  | 
|  | var deleteDataTests = []; | 
|  | for (var i = 0; i < characterDataNodes.length; i++) { | 
|  | var node = characterDataNodes[i]; | 
|  | deleteDataTests.push([node, 376, 2, node, 0, node, 1]); | 
|  | deleteDataTests.push([node, 0, 2, node, 0, node, 0]); | 
|  | deleteDataTests.push([node, 1, 2, node, 1, node, 1]); | 
|  | deleteDataTests.push([node, node + ".length", 2, node, node + ".length", node, node + ".length"]); | 
|  | deleteDataTests.push([node, 1, 2, node, 1, node, 3]); | 
|  | deleteDataTests.push([node, 2, 2, node, 1, node, 3]); | 
|  | deleteDataTests.push([node, 3, 2, node, 1, node, 3]); | 
|  |  | 
|  | deleteDataTests.push([node, 376, 0, node, 0, node, 1]); | 
|  | deleteDataTests.push([node, 0, 0, node, 0, node, 0]); | 
|  | deleteDataTests.push([node, 1, 0, node, 1, node, 1]); | 
|  | deleteDataTests.push([node, node + ".length", 0, node, node + ".length", node, node + ".length"]); | 
|  | deleteDataTests.push([node, 1, 0, node, 1, node, 3]); | 
|  | deleteDataTests.push([node, 2, 0, node, 1, node, 3]); | 
|  | deleteDataTests.push([node, 3, 0, node, 1, node, 3]); | 
|  |  | 
|  | deleteDataTests.push([node, 376, 631, node, 0, node, 1]); | 
|  | deleteDataTests.push([node, 0, 631, node, 0, node, 0]); | 
|  | deleteDataTests.push([node, 1, 631, node, 1, node, 1]); | 
|  | deleteDataTests.push([node, node + ".length", 631, node, node + ".length", node, node + ".length"]); | 
|  | deleteDataTests.push([node, 1, 631, node, 1, node, 3]); | 
|  | deleteDataTests.push([node, 2, 631, node, 1, node, 3]); | 
|  | deleteDataTests.push([node, 3, 631, node, 1, node, 3]); | 
|  | } | 
|  |  | 
|  | deleteDataTests.push( | 
|  | ["paras[0].firstChild", 1, 2, "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", 1, 2, "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 2, "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 2, "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 2, 2, "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 3, 2, "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 2, "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 2, 2, "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 3, 2, "paras[0]", 0, "paras[0].firstChild", 3] | 
|  | ); | 
|  |  | 
|  |  | 
|  | function testReplaceData(node, offset, count, data, startContainer, startOffset, endContainer, endOffset) { | 
|  | return testReplaceDataAlgorithm(node, offset, count, data, | 
|  | function() { node.replaceData(offset, count, data) }, | 
|  | startContainer, startOffset, endContainer, endOffset); | 
|  | } | 
|  |  | 
|  | var replaceDataTests = []; | 
|  | for (var i = 0; i < characterDataNodes.length; i++) { | 
|  | var node = characterDataNodes[i]; | 
|  | replaceDataTests.push([node, 376, 0, '"foo"', node, 0, node, 1]); | 
|  | replaceDataTests.push([node, 0, 0, '"foo"', node, 0, node, 0]); | 
|  | replaceDataTests.push([node, 1, 0, '"foo"', node, 1, node, 1]); | 
|  | replaceDataTests.push([node, node + ".length", 0, '"foo"', node, node + ".length", node, node + ".length"]); | 
|  | replaceDataTests.push([node, 1, 0, '"foo"', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 2, 0, '"foo"', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 3, 0, '"foo"', node, 1, node, 3]); | 
|  |  | 
|  | replaceDataTests.push([node, 376, 0, '""', node, 0, node, 1]); | 
|  | replaceDataTests.push([node, 0, 0, '""', node, 0, node, 0]); | 
|  | replaceDataTests.push([node, 1, 0, '""', node, 1, node, 1]); | 
|  | replaceDataTests.push([node, node + ".length", 0, '""', node, node + ".length", node, node + ".length"]); | 
|  | replaceDataTests.push([node, 1, 0, '""', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 2, 0, '""', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 3, 0, '""', node, 1, node, 3]); | 
|  |  | 
|  | replaceDataTests.push([node, 376, 1, '"foo"', node, 0, node, 1]); | 
|  | replaceDataTests.push([node, 0, 1, '"foo"', node, 0, node, 0]); | 
|  | replaceDataTests.push([node, 1, 1, '"foo"', node, 1, node, 1]); | 
|  | replaceDataTests.push([node, node + ".length", 1, '"foo"', node, node + ".length", node, node + ".length"]); | 
|  | replaceDataTests.push([node, 1, 1, '"foo"', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 2, 1, '"foo"', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 3, 1, '"foo"', node, 1, node, 3]); | 
|  |  | 
|  | replaceDataTests.push([node, 376, 1, '""', node, 0, node, 1]); | 
|  | replaceDataTests.push([node, 0, 1, '""', node, 0, node, 0]); | 
|  | replaceDataTests.push([node, 1, 1, '""', node, 1, node, 1]); | 
|  | replaceDataTests.push([node, node + ".length", 1, '""', node, node + ".length", node, node + ".length"]); | 
|  | replaceDataTests.push([node, 1, 1, '""', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 2, 1, '""', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 3, 1, '""', node, 1, node, 3]); | 
|  |  | 
|  | replaceDataTests.push([node, 376, 47, '"foo"', node, 0, node, 1]); | 
|  | replaceDataTests.push([node, 0, 47, '"foo"', node, 0, node, 0]); | 
|  | replaceDataTests.push([node, 1, 47, '"foo"', node, 1, node, 1]); | 
|  | replaceDataTests.push([node, node + ".length", 47, '"foo"', node, node + ".length", node, node + ".length"]); | 
|  | replaceDataTests.push([node, 1, 47, '"foo"', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 2, 47, '"foo"', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 3, 47, '"foo"', node, 1, node, 3]); | 
|  |  | 
|  | replaceDataTests.push([node, 376, 47, '""', node, 0, node, 1]); | 
|  | replaceDataTests.push([node, 0, 47, '""', node, 0, node, 0]); | 
|  | replaceDataTests.push([node, 1, 47, '""', node, 1, node, 1]); | 
|  | replaceDataTests.push([node, node + ".length", 47, '""', node, node + ".length", node, node + ".length"]); | 
|  | replaceDataTests.push([node, 1, 47, '""', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 2, 47, '""', node, 1, node, 3]); | 
|  | replaceDataTests.push([node, 3, 47, '""', node, 1, node, 3]); | 
|  | } | 
|  |  | 
|  | replaceDataTests.push( | 
|  | ["paras[0].firstChild", 1, 0, '"foo"', "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", 1, 0, '"foo"', "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 0, '"foo"', "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 0, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 2, 0, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 3, 0, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 0, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 2, 0, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 3, 0, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  |  | 
|  | ["paras[0].firstChild", 1, 1, '"foo"', "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", 1, 1, '"foo"', "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 1, '"foo"', "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 1, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 2, 1, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 3, 1, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 1, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 2, 1, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 3, 1, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  |  | 
|  | ["paras[0].firstChild", 1, 47, '"foo"', "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0].firstChild", 1, 47, '"foo"', "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 47, '"foo"', "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 47, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 2, 47, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 3, 47, '"foo"', "paras[0].firstChild", 1, "paras[0]", 1], | 
|  | ["paras[0].firstChild", 1, 47, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 2, 47, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3], | 
|  | ["paras[0].firstChild", 3, 47, '"foo"', "paras[0]", 0, "paras[0].firstChild", 3] | 
|  | ); | 
|  |  | 
|  |  | 
|  | // There are lots of ways to set data, so we pass a callback that does the | 
|  | // actual setting. | 
|  | function testDataChange(node, attr, op, rval, startContainer, startOffset, endContainer, endOffset) { | 
|  | return testReplaceDataAlgorithm(node, 0, node.length, op == "=" ? rval : node[attr] + rval, | 
|  | function() { | 
|  | if (op == "=") { | 
|  | node[attr] = rval; | 
|  | } else if (op == "+=") { | 
|  | node[attr] += rval; | 
|  | } else { | 
|  | throw "Unknown op " + op; | 
|  | } | 
|  | }, | 
|  | startContainer, startOffset, endContainer, endOffset); | 
|  | } | 
|  |  | 
|  | var dataChangeTests = []; | 
|  | var dataChangeTestAttrs = ["data", "textContent", "nodeValue"]; | 
|  | for (var i = 0; i < characterDataNodes.length; i++) { | 
|  | var node = characterDataNodes[i]; | 
|  | var dataChangeTestRanges = [ | 
|  | [node, 0, node, 0], | 
|  | [node, 0, node, 1], | 
|  | [node, 1, node, 1], | 
|  | [node, 0, node, node + ".length"], | 
|  | [node, 1, node, node + ".length"], | 
|  | [node, node + ".length", node, node + ".length"], | 
|  | ]; | 
|  |  | 
|  | for (var j = 0; j < dataChangeTestRanges.length; j++) { | 
|  | for (var k = 0; k < dataChangeTestAttrs.length; k++) { | 
|  | dataChangeTests.push([ | 
|  | node, | 
|  | '"' + dataChangeTestAttrs[k] + '"', | 
|  | '"="', | 
|  | '""', | 
|  | ].concat(dataChangeTestRanges[j])); | 
|  |  | 
|  | dataChangeTests.push([ | 
|  | node, | 
|  | '"' + dataChangeTestAttrs[k] + '"', | 
|  | '"="', | 
|  | '"foo"', | 
|  | ].concat(dataChangeTestRanges[j])); | 
|  |  | 
|  | dataChangeTests.push([ | 
|  | node, | 
|  | '"' + dataChangeTestAttrs[k] + '"', | 
|  | '"="', | 
|  | node + "." + dataChangeTestAttrs[k], | 
|  | ].concat(dataChangeTestRanges[j])); | 
|  |  | 
|  | dataChangeTests.push([ | 
|  | node, | 
|  | '"' + dataChangeTestAttrs[k] + '"', | 
|  | '"+="', | 
|  | '""', | 
|  | ].concat(dataChangeTestRanges[j])); | 
|  |  | 
|  | dataChangeTests.push([ | 
|  | node, | 
|  | '"' + dataChangeTestAttrs[k] + '"', | 
|  | '"+="', | 
|  | '"foo"', | 
|  | ].concat(dataChangeTestRanges[j])); | 
|  |  | 
|  | dataChangeTests.push([ | 
|  | node, | 
|  | '"' + dataChangeTestAttrs[k] + '"', | 
|  | '"+="', | 
|  | node + "." + dataChangeTestAttrs[k] | 
|  | ].concat(dataChangeTestRanges[j])); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | // Now we test node insertions and deletions, as opposed to just data changes. | 
|  | // To avoid loads of repetition, we define modifyForRemove() and | 
|  | // modifyForInsert(). | 
|  |  | 
|  | // If we were to remove removedNode from its parent, what would the boundary | 
|  | // point [node, offset] become?  Returns [new node, new offset].  Must be | 
|  | // called BEFORE the node is actually removed, so its parent is not null.  (If | 
|  | // the parent is null, it will do nothing.) | 
|  | function modifyForRemove(removedNode, point) { | 
|  | var oldParent = removedNode.parentNode; | 
|  | var oldIndex = indexOf(removedNode); | 
|  | if (!oldParent) { | 
|  | return point; | 
|  | } | 
|  |  | 
|  | // "For each boundary point whose node is removed node or a descendant of | 
|  | // it, set the boundary point to (old parent, old index)." | 
|  | if (point[0] == removedNode || isDescendant(point[0], removedNode)) { | 
|  | return [oldParent, oldIndex]; | 
|  | } | 
|  |  | 
|  | // "For each boundary point whose node is old parent and whose offset is | 
|  | // greater than old index, subtract one from its offset." | 
|  | if (point[0] == oldParent && point[1] > oldIndex) { | 
|  | return [point[0], point[1] - 1]; | 
|  | } | 
|  |  | 
|  | return point; | 
|  | } | 
|  |  | 
|  | // Update the given boundary point [node, offset] to account for the fact that | 
|  | // insertedNode was just inserted into its current position.  This must be | 
|  | // called AFTER insertedNode was already inserted. | 
|  | function modifyForInsert(insertedNode, point) { | 
|  | // "For each boundary point whose node is the new parent of the affected | 
|  | // node and whose offset is greater than the new index of the affected | 
|  | // node, add one to the boundary point's offset." | 
|  | if (point[0] == insertedNode.parentNode && point[1] > indexOf(insertedNode)) { | 
|  | return [point[0], point[1] + 1]; | 
|  | } | 
|  |  | 
|  | return point; | 
|  | } | 
|  |  | 
|  |  | 
|  | function testInsertBefore(newParent, affectedNode, refNode, startContainer, startOffset, endContainer, endOffset) { | 
|  | var expectedStart = [startContainer, startOffset]; | 
|  | var expectedEnd = [endContainer, endOffset]; | 
|  |  | 
|  | expectedStart = modifyForRemove(affectedNode, expectedStart); | 
|  | expectedEnd = modifyForRemove(affectedNode, expectedEnd); | 
|  |  | 
|  | try { | 
|  | newParent.insertBefore(affectedNode, refNode); | 
|  | } catch (e) { | 
|  | // For our purposes, assume that DOM Core is true -- i.e., ignore | 
|  | // mutation events and similar. | 
|  | return [startContainer, startOffset, endContainer, endOffset]; | 
|  | } | 
|  |  | 
|  | expectedStart = modifyForInsert(affectedNode, expectedStart); | 
|  | expectedEnd = modifyForInsert(affectedNode, expectedEnd); | 
|  |  | 
|  | return expectedStart.concat(expectedEnd); | 
|  | } | 
|  |  | 
|  | var insertBeforeTests = [ | 
|  | // Moving a node to its current position | 
|  | ["testDiv", "paras[0]", "paras[1]", "paras[0]", 0, "paras[0]", 0], | 
|  | ["testDiv", "paras[0]", "paras[1]", "paras[0]", 0, "paras[0]", 1], | 
|  | ["testDiv", "paras[0]", "paras[1]", "paras[0]", 1, "paras[0]", 1], | 
|  | ["testDiv", "paras[0]", "paras[1]", "testDiv", 0, "testDiv", 2], | 
|  | ["testDiv", "paras[0]", "paras[1]", "testDiv", 1, "testDiv", 1], | 
|  | ["testDiv", "paras[0]", "paras[1]", "testDiv", 1, "testDiv", 2], | 
|  | ["testDiv", "paras[0]", "paras[1]", "testDiv", 2, "testDiv", 2], | 
|  |  | 
|  | // Stuff that actually moves something.  Note that paras[0] and paras[1] | 
|  | // are both children of testDiv. | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 0, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 0, "testDiv", 2], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 1, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 1, "testDiv", 2], | 
|  | ["paras[0]", "paras[1]", "null", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "paras[1]", "null", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "null", "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "null", "testDiv", 0, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "null", "testDiv", 0, "testDiv", 2], | 
|  | ["paras[0]", "paras[1]", "null", "testDiv", 1, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "null", "testDiv", 1, "testDiv", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", 0], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 1, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 0, "foreignDoc", 0], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 0, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 0, "foreignDoc", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 1, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "null", "foreignDoc", 0, "foreignDoc", 1], | 
|  | ["paras[0]", "xmlTextNode", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "xmlTextNode", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "xmlTextNode", "paras[0].firstChild", "paras[0]", 1, "paras[0]", 1], | 
|  |  | 
|  | // Stuff that throws exceptions | 
|  | ["paras[0]", "paras[0]", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "testDiv", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "document", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "foreignDoc", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "document.doctype", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ]; | 
|  |  | 
|  |  | 
|  | function testReplaceChild(newParent, newChild, oldChild, startContainer, startOffset, endContainer, endOffset) { | 
|  | var expectedStart = [startContainer, startOffset]; | 
|  | var expectedEnd = [endContainer, endOffset]; | 
|  |  | 
|  | expectedStart = modifyForRemove(oldChild, expectedStart); | 
|  | expectedEnd = modifyForRemove(oldChild, expectedEnd); | 
|  |  | 
|  | if (newChild != oldChild) { | 
|  | // Don't do this twice, if they're the same! | 
|  | expectedStart = modifyForRemove(newChild, expectedStart); | 
|  | expectedEnd = modifyForRemove(newChild, expectedEnd); | 
|  | } | 
|  |  | 
|  | try { | 
|  | newParent.replaceChild(newChild, oldChild); | 
|  | } catch (e) { | 
|  | return [startContainer, startOffset, endContainer, endOffset]; | 
|  | } | 
|  |  | 
|  | expectedStart = modifyForInsert(newChild, expectedStart); | 
|  | expectedEnd = modifyForInsert(newChild, expectedEnd); | 
|  |  | 
|  | return expectedStart.concat(expectedEnd); | 
|  | } | 
|  |  | 
|  | var replaceChildTests = [ | 
|  | // Moving a node to its current position.  Doesn't match most browsers' | 
|  | // behavior, but we probably want to keep the spec the same anyway: | 
|  | // https://bugzilla.mozilla.org/show_bug.cgi?id=647603 | 
|  | ["testDiv", "paras[0]", "paras[0]", "paras[0]", 0, "paras[0]", 0], | 
|  | ["testDiv", "paras[0]", "paras[0]", "paras[0]", 0, "paras[0]", 1], | 
|  | ["testDiv", "paras[0]", "paras[0]", "paras[0]", 1, "paras[0]", 1], | 
|  | ["testDiv", "paras[0]", "paras[0]", "testDiv", 0, "testDiv", 2], | 
|  | ["testDiv", "paras[0]", "paras[0]", "testDiv", 1, "testDiv", 1], | 
|  | ["testDiv", "paras[0]", "paras[0]", "testDiv", 1, "testDiv", 2], | 
|  | ["testDiv", "paras[0]", "paras[0]", "testDiv", 2, "testDiv", 2], | 
|  |  | 
|  | // Stuff that actually moves something. | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 0, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 0, "testDiv", 2], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 1, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0].firstChild", "testDiv", 1, "testDiv", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", 0], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.documentElement", "foreignDoc", 1, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 0, "foreignDoc", 0], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 0, "foreignDoc", 1], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 0, "foreignDoc", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc.doctype", "foreignDoc", 1, "foreignDoc", 1], | 
|  | ["paras[0]", "xmlTextNode", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "xmlTextNode", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "xmlTextNode", "paras[0].firstChild", "paras[0]", 1, "paras[0]", 1], | 
|  |  | 
|  | // Stuff that throws exceptions | 
|  | ["paras[0]", "paras[0]", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "testDiv", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "document", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "foreignDoc", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "document.doctype", "paras[0].firstChild", "paras[0]", 0, "paras[0]", 1], | 
|  | ]; | 
|  |  | 
|  |  | 
|  | function testAppendChild(newParent, affectedNode, startContainer, startOffset, endContainer, endOffset) { | 
|  | var expectedStart = [startContainer, startOffset]; | 
|  | var expectedEnd = [endContainer, endOffset]; | 
|  |  | 
|  | expectedStart = modifyForRemove(affectedNode, expectedStart); | 
|  | expectedEnd = modifyForRemove(affectedNode, expectedEnd); | 
|  |  | 
|  | try { | 
|  | newParent.appendChild(affectedNode); | 
|  | } catch (e) { | 
|  | return [startContainer, startOffset, endContainer, endOffset]; | 
|  | } | 
|  |  | 
|  | // These two lines will actually never do anything, if you think about it, | 
|  | // but let's leave them in so correctness is more obvious. | 
|  | expectedStart = modifyForInsert(affectedNode, expectedStart); | 
|  | expectedEnd = modifyForInsert(affectedNode, expectedEnd); | 
|  |  | 
|  | return expectedStart.concat(expectedEnd); | 
|  | } | 
|  |  | 
|  | var appendChildTests = [ | 
|  | // Moving a node to its current position | 
|  | ["testDiv", "testDiv.lastChild", "testDiv.lastChild", 0, "testDiv.lastChild", 0], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv.lastChild", 0, "testDiv.lastChild", 1], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv.lastChild", 1, "testDiv.lastChild", 1], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv", "testDiv.childNodes.length - 2", "testDiv", "testDiv.childNodes.length"], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv", "testDiv.childNodes.length - 2", "testDiv", "testDiv.childNodes.length - 1"], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv", "testDiv.childNodes.length - 1", "testDiv", "testDiv.childNodes.length"], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv", "testDiv.childNodes.length - 1", "testDiv", "testDiv.childNodes.length - 1"], | 
|  | ["testDiv", "testDiv.lastChild", "testDiv", "testDiv.childNodes.length", "testDiv", "testDiv.childNodes.length"], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv.lastChild", 0, "detachedDiv.lastChild", 0], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv.lastChild", 0, "detachedDiv.lastChild", 1], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv.lastChild", 1, "detachedDiv.lastChild", 1], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv", "detachedDiv.childNodes.length - 2", "detachedDiv", "detachedDiv.childNodes.length"], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv", "detachedDiv.childNodes.length - 2", "detachedDiv", "detachedDiv.childNodes.length - 1"], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv", "detachedDiv.childNodes.length - 1", "detachedDiv", "detachedDiv.childNodes.length"], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv", "detachedDiv.childNodes.length - 1", "detachedDiv", "detachedDiv.childNodes.length - 1"], | 
|  | ["detachedDiv", "detachedDiv.lastChild", "detachedDiv", "detachedDiv.childNodes.length", "detachedDiv", "detachedDiv.childNodes.length"], | 
|  |  | 
|  | // Stuff that actually moves something | 
|  | ["paras[0]", "paras[1]", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "paras[1]", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0]", "paras[1]", "testDiv", 0, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "testDiv", 0, "testDiv", 2], | 
|  | ["paras[0]", "paras[1]", "testDiv", 1, "testDiv", 1], | 
|  | ["paras[0]", "paras[1]", "testDiv", 1, "testDiv", 2], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc", "foreignDoc.childNodes.length - 1", "foreignDoc", "foreignDoc.childNodes.length"], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc", "foreignDoc.childNodes.length - 1", "foreignDoc", "foreignDoc.childNodes.length - 1"], | 
|  | ["foreignDoc", "detachedComment", "foreignDoc", "foreignDoc.childNodes.length", "foreignDoc", "foreignDoc.childNodes.length"], | 
|  | ["foreignDoc", "detachedComment", "detachedComment", 0, "detachedComment", 5], | 
|  | ["paras[0]", "xmlTextNode", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "xmlTextNode", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "xmlTextNode", "paras[0]", 1, "paras[0]", 1], | 
|  |  | 
|  | // Stuff that throws exceptions | 
|  | ["paras[0]", "paras[0]", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "testDiv", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "document", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "foreignDoc", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "document.doctype", "paras[0]", 0, "paras[0]", 1], | 
|  | ]; | 
|  |  | 
|  |  | 
|  | function testRemoveChild(affectedNode, startContainer, startOffset, endContainer, endOffset) { | 
|  | var expectedStart = [startContainer, startOffset]; | 
|  | var expectedEnd = [endContainer, endOffset]; | 
|  |  | 
|  | expectedStart = modifyForRemove(affectedNode, expectedStart); | 
|  | expectedEnd = modifyForRemove(affectedNode, expectedEnd); | 
|  |  | 
|  | // We don't test cases where the parent is wrong, so this should never | 
|  | // throw an exception. | 
|  | affectedNode.parentNode.removeChild(affectedNode); | 
|  |  | 
|  | return expectedStart.concat(expectedEnd); | 
|  | } | 
|  |  | 
|  | var removeChildTests = [ | 
|  | ["paras[0]", "paras[0]", 0, "paras[0]", 0], | 
|  | ["paras[0]", "paras[0]", 0, "paras[0]", 1], | 
|  | ["paras[0]", "paras[0]", 1, "paras[0]", 1], | 
|  | ["paras[0]", "testDiv", 0, "testDiv", 0], | 
|  | ["paras[0]", "testDiv", 0, "testDiv", 1], | 
|  | ["paras[0]", "testDiv", 1, "testDiv", 1], | 
|  | ["paras[0]", "testDiv", 0, "testDiv", 2], | 
|  | ["paras[0]", "testDiv", 1, "testDiv", 2], | 
|  | ["paras[0]", "testDiv", 2, "testDiv", 2], | 
|  |  | 
|  | ["foreignDoc.documentElement", "foreignDoc", 0, "foreignDoc", "foreignDoc.childNodes.length"], | 
|  | ]; | 
|  |  | 
|  |  | 
|  | // Finally run everything.  All grouped together at the end so that I can | 
|  | // easily comment out some of them, so I don't have to wait for all test types | 
|  | // to debug only some of them. | 
|  | doTests(splitTextTests, function(params) { return params[0] + ".splitText(" + params[1] + ")" }, testSplitText); | 
|  | doTests(insertDataTests, function(params) { return params[0] + ".insertData(" + params[1] + ", " + params[2] + ")" }, testInsertData); | 
|  | doTests(appendDataTests, function(params) { return params[0] + ".appendData(" + params[1] + ")" }, testAppendData); | 
|  | doTests(deleteDataTests, function(params) { return params[0] + ".deleteData(" + params[1] + ", " + params[2] + ")" }, testDeleteData); | 
|  | doTests(replaceDataTests, function(params) { return params[0] + ".replaceData(" + params[1] + ", " + params[2] + ", " + params[3] + ")" }, testReplaceData); | 
|  | doTests(dataChangeTests, function(params) { return params[0] + "." + eval(params[1]) + " " + eval(params[2]) + ' ' + params[3] }, testDataChange); | 
|  | doTests(insertBeforeTests, function(params) { return params[0] + ".insertBefore(" + params[1] + ", " + params[2] + ")" }, testInsertBefore); | 
|  | doTests(replaceChildTests, function(params) { return params[0] + ".replaceChild(" + params[1] + ", " + params[2] + ")" }, testReplaceChild); | 
|  | doTests(appendChildTests, function(params) { return params[0] + ".appendChild(" + params[1] + ")" }, testAppendChild); | 
|  | doTests(removeChildTests, function(params) { return params[0] + ".parentNode.removeChild(" + params[0] + ")" }, testRemoveChild); | 
|  |  | 
|  |  | 
|  | testDiv.style.display = "none"; | 
|  | </script> |