%PDF- %PDF-
| Direktori : /home/tjamichg/intranet.tjamich.gob.mx/intranet/common/vendors/jstree/test/libs/ |
| Current File : /home/tjamichg/intranet.tjamich.gob.mx/intranet/common/vendors/jstree/test/libs/qunit.js |
/**
* QUnit v1.12.0 - A JavaScript Unit Testing Framework
*
* http://qunitjs.com
*
* Copyright 2013 jQuery Foundation and other contributors
* Released under the MIT license.
* https://jquery.org/license/
*/
(function (window) {
var QUnit,
assert,
config,
onErrorFnPrev,
testId = 0,
fileName = (sourceFromStacktrace(0) || "" ).replace(/(:\d+)+\)?/, "").replace(/.+\//, ""),
toString = Object.prototype.toString,
hasOwn = Object.prototype.hasOwnProperty,
// Keep a local reference to Date (GH-283)
Date = window.Date,
setTimeout = window.setTimeout,
defined = {
setTimeout: typeof window.setTimeout !== "undefined",
sessionStorage: (function () {
var x = "qunit-test-string";
try {
sessionStorage.setItem(x, x);
sessionStorage.removeItem(x);
return true;
} catch (e) {
return false;
}
}())
},
/**
* Provides a normalized error string, correcting an issue
* with IE 7 (and prior) where Error.prototype.toString is
* not properly implemented
*
* Based on http://es5.github.com/#x15.11.4.4
*
* @param {String|Error} error
* @return {String} error message
*/
errorString = function (error) {
var name, message,
errorString = error.toString();
if (errorString.substring(0, 7) === "[object") {
name = error.name ? error.name.toString() : "Error";
message = error.message ? error.message.toString() : "";
if (name && message) {
return name + ": " + message;
} else if (name) {
return name;
} else if (message) {
return message;
} else {
return "Error";
}
} else {
return errorString;
}
},
/**
* Makes a clone of an object using only Array or Object as base,
* and copies over the own enumerable properties.
*
* @param {Object} obj
* @return {Object} New object with only the own properties (recursively).
*/
objectValues = function (obj) {
// Grunt 0.3.x uses an older version of jshint that still has jshint/jshint#392.
/*jshint newcap: false */
var key, val,
vals = QUnit.is("array", obj) ? [] : {};
for (key in obj) {
if (hasOwn.call(obj, key)) {
val = obj[key];
vals[key] = val === Object(val) ? objectValues(val) : val;
}
}
return vals;
};
function Test(settings) {
extend(this, settings);
this.assertions = [];
this.testNumber = ++Test.count;
}
Test.count = 0;
Test.prototype = {
init: function () {
var a, b, li,
tests = id("qunit-tests");
if (tests) {
b = document.createElement("strong");
b.innerHTML = this.nameHtml;
// `a` initialized at top of scope
a = document.createElement("a");
a.innerHTML = "Rerun";
a.href = QUnit.url({ testNumber: this.testNumber });
li = document.createElement("li");
li.appendChild(b);
li.appendChild(a);
li.className = "running";
li.id = this.id = "qunit-test-output" + testId++;
tests.appendChild(li);
}
},
setup: function () {
if (
// Emit moduleStart when we're switching from one module to another
this.module !== config.previousModule ||
// They could be equal (both undefined) but if the previousModule property doesn't
// yet exist it means this is the first test in a suite that isn't wrapped in a
// module, in which case we'll just emit a moduleStart event for 'undefined'.
// Without this, reporters can get testStart before moduleStart which is a problem.
!hasOwn.call(config, "previousModule")
) {
if (hasOwn.call(config, "previousModule")) {
runLoggingCallbacks("moduleDone", QUnit, {
name: config.previousModule,
failed: config.moduleStats.bad,
passed: config.moduleStats.all - config.moduleStats.bad,
total: config.moduleStats.all
});
}
config.previousModule = this.module;
config.moduleStats = { all: 0, bad: 0 };
runLoggingCallbacks("moduleStart", QUnit, {
name: this.module
});
}
config.current = this;
this.testEnvironment = extend({
setup: function () {
},
teardown: function () {
}
}, this.moduleTestEnvironment);
this.started = +new Date();
runLoggingCallbacks("testStart", QUnit, {
name: this.testName,
module: this.module
});
/*jshint camelcase:false */
/**
* Expose the current test environment.
*
* @deprecated since 1.12.0: Use QUnit.config.current.testEnvironment instead.
*/
QUnit.current_testEnvironment = this.testEnvironment;
/*jshint camelcase:true */
if (!config.pollution) {
saveGlobal();
}
if (config.notrycatch) {
this.testEnvironment.setup.call(this.testEnvironment, QUnit.assert);
return;
}
try {
this.testEnvironment.setup.call(this.testEnvironment, QUnit.assert);
} catch (e) {
QUnit.pushFailure("Setup failed on " + this.testName + ": " + ( e.message || e ), extractStacktrace(e, 1));
}
},
run: function () {
config.current = this;
var running = id("qunit-testresult");
if (running) {
running.innerHTML = "Running: <br/>" + this.nameHtml;
}
if (this.async) {
QUnit.stop();
}
this.callbackStarted = +new Date();
if (config.notrycatch) {
this.callback.call(this.testEnvironment, QUnit.assert);
this.callbackRuntime = +new Date() - this.callbackStarted;
return;
}
try {
this.callback.call(this.testEnvironment, QUnit.assert);
this.callbackRuntime = +new Date() - this.callbackStarted;
} catch (e) {
this.callbackRuntime = +new Date() - this.callbackStarted;
QUnit.pushFailure("Died on test #" + (this.assertions.length + 1) + " " + this.stack + ": " + ( e.message || e ), extractStacktrace(e, 0));
// else next test will carry the responsibility
saveGlobal();
// Restart the tests if they're blocking
if (config.blocking) {
QUnit.start();
}
}
},
teardown: function () {
config.current = this;
if (config.notrycatch) {
if (typeof this.callbackRuntime === "undefined") {
this.callbackRuntime = +new Date() - this.callbackStarted;
}
this.testEnvironment.teardown.call(this.testEnvironment, QUnit.assert);
return;
} else {
try {
this.testEnvironment.teardown.call(this.testEnvironment, QUnit.assert);
} catch (e) {
QUnit.pushFailure("Teardown failed on " + this.testName + ": " + ( e.message || e ), extractStacktrace(e, 1));
}
}
checkPollution();
},
finish: function () {
config.current = this;
if (config.requireExpects && this.expected === null) {
QUnit.pushFailure("Expected number of assertions to be defined, but expect() was not called.", this.stack);
} else if (this.expected !== null && this.expected !== this.assertions.length) {
QUnit.pushFailure("Expected " + this.expected + " assertions, but " + this.assertions.length + " were run", this.stack);
} else if (this.expected === null && !this.assertions.length) {
QUnit.pushFailure("Expected at least one assertion, but none were run - call expect(0) to accept zero assertions.", this.stack);
}
var i, assertion, a, b, time, li, ol,
test = this,
good = 0,
bad = 0,
tests = id("qunit-tests");
this.runtime = +new Date() - this.started;
config.stats.all += this.assertions.length;
config.moduleStats.all += this.assertions.length;
if (tests) {
ol = document.createElement("ol");
ol.className = "qunit-assert-list";
for (i = 0; i < this.assertions.length; i++) {
assertion = this.assertions[i];
li = document.createElement("li");
li.className = assertion.result ? "pass" : "fail";
li.innerHTML = assertion.message || ( assertion.result ? "okay" : "failed" );
ol.appendChild(li);
if (assertion.result) {
good++;
} else {
bad++;
config.stats.bad++;
config.moduleStats.bad++;
}
}
// store result when possible
if (QUnit.config.reorder && defined.sessionStorage) {
if (bad) {
sessionStorage.setItem("qunit-test-" + this.module + "-" + this.testName, bad);
} else {
sessionStorage.removeItem("qunit-test-" + this.module + "-" + this.testName);
}
}
if (bad === 0) {
addClass(ol, "qunit-collapsed");
}
// `b` initialized at top of scope
b = document.createElement("strong");
b.innerHTML = this.nameHtml + " <b class='counts'>(<b class='failed'>" + bad + "</b>, <b class='passed'>" + good + "</b>, " + this.assertions.length + ")</b>";
addEvent(b, "click", function () {
var next = b.parentNode.lastChild,
collapsed = hasClass(next, "qunit-collapsed");
( collapsed ? removeClass : addClass )(next, "qunit-collapsed");
});
addEvent(b, "dblclick", function (e) {
var target = e && e.target ? e.target : window.event.srcElement;
if (target.nodeName.toLowerCase() === "span" || target.nodeName.toLowerCase() === "b") {
target = target.parentNode;
}
if (window.location && target.nodeName.toLowerCase() === "strong") {
window.location = QUnit.url({ testNumber: test.testNumber });
}
});
// `time` initialized at top of scope
time = document.createElement("span");
time.className = "runtime";
time.innerHTML = this.runtime + " ms";
// `li` initialized at top of scope
li = id(this.id);
li.className = bad ? "fail" : "pass";
li.removeChild(li.firstChild);
a = li.firstChild;
li.appendChild(b);
li.appendChild(a);
li.appendChild(time);
li.appendChild(ol);
} else {
for (i = 0; i < this.assertions.length; i++) {
if (!this.assertions[i].result) {
bad++;
config.stats.bad++;
config.moduleStats.bad++;
}
}
}
runLoggingCallbacks("testDone", QUnit, {
name: this.testName,
module: this.module,
failed: bad,
passed: this.assertions.length - bad,
total: this.assertions.length,
duration: this.runtime
});
QUnit.reset();
config.current = undefined;
},
queue: function () {
var bad,
test = this;
synchronize(function () {
test.init();
});
function run() {
// each of these can by async
synchronize(function () {
test.setup();
});
synchronize(function () {
test.run();
});
synchronize(function () {
test.teardown();
});
synchronize(function () {
test.finish();
});
}
// `bad` initialized at top of scope
// defer when previous test run passed, if storage is available
bad = QUnit.config.reorder && defined.sessionStorage && +sessionStorage.getItem("qunit-test-" + this.module + "-" + this.testName);
if (bad) {
run();
} else {
synchronize(run, true);
}
}
};
// Root QUnit object.
// `QUnit` initialized at top of scope
QUnit = {
// call on start of module test to prepend name to all tests
module: function (name, testEnvironment) {
config.currentModule = name;
config.currentModuleTestEnvironment = testEnvironment;
config.modules[name] = true;
},
asyncTest: function (testName, expected, callback) {
if (arguments.length === 2) {
callback = expected;
expected = null;
}
QUnit.test(testName, expected, callback, true);
},
test: function (testName, expected, callback, async) {
var test,
nameHtml = "<span class='test-name'>" + escapeText(testName) + "</span>";
if (arguments.length === 2) {
callback = expected;
expected = null;
}
if (config.currentModule) {
nameHtml = "<span class='module-name'>" + escapeText(config.currentModule) + "</span>: " + nameHtml;
}
test = new Test({
nameHtml: nameHtml,
testName: testName,
expected: expected,
async: async,
callback: callback,
module: config.currentModule,
moduleTestEnvironment: config.currentModuleTestEnvironment,
stack: sourceFromStacktrace(2)
});
if (!validTest(test)) {
return;
}
test.queue();
},
// Specify the number of expected assertions to guarantee that failed test (no assertions are run at all) don't slip through.
expect: function (asserts) {
if (arguments.length === 1) {
config.current.expected = asserts;
} else {
return config.current.expected;
}
},
start: function (count) {
// QUnit hasn't been initialized yet.
// Note: RequireJS (et al) may delay onLoad
if (config.semaphore === undefined) {
QUnit.begin(function () {
// This is triggered at the top of QUnit.load, push start() to the event loop, to allow QUnit.load to finish first
setTimeout(function () {
QUnit.start(count);
});
});
return;
}
config.semaphore -= count || 1;
// don't start until equal number of stop-calls
if (config.semaphore > 0) {
return;
}
// ignore if start is called more often then stop
if (config.semaphore < 0) {
config.semaphore = 0;
QUnit.pushFailure("Called start() while already started (QUnit.config.semaphore was 0 already)", null, sourceFromStacktrace(2));
return;
}
// A slight delay, to avoid any current callbacks
if (defined.setTimeout) {
setTimeout(function () {
if (config.semaphore > 0) {
return;
}
if (config.timeout) {
clearTimeout(config.timeout);
}
config.blocking = false;
process(true);
}, 13);
} else {
config.blocking = false;
process(true);
}
},
stop: function (count) {
config.semaphore += count || 1;
config.blocking = true;
if (config.testTimeout && defined.setTimeout) {
clearTimeout(config.timeout);
config.timeout = setTimeout(function () {
QUnit.ok(false, "Test timed out");
config.semaphore = 1;
QUnit.start();
}, config.testTimeout);
}
}
};
// `assert` initialized at top of scope
// Assert helpers
// All of these must either call QUnit.push() or manually do:
// - runLoggingCallbacks( "log", .. );
// - config.current.assertions.push({ .. });
// We attach it to the QUnit object *after* we expose the public API,
// otherwise `assert` will become a global variable in browsers (#341).
assert = {
/**
* Asserts rough true-ish result.
* @name ok
* @function
* @example ok( "asdfasdf".length > 5, "There must be at least 5 chars" );
*/
ok: function (result, msg) {
if (!config.current) {
throw new Error("ok() assertion outside test context, was " + sourceFromStacktrace(2));
}
result = !!result;
msg = msg || (result ? "okay" : "failed" );
var source,
details = {
module: config.current.module,
name: config.current.testName,
result: result,
message: msg
};
msg = "<span class='test-message'>" + escapeText(msg) + "</span>";
if (!result) {
source = sourceFromStacktrace(2);
if (source) {
details.source = source;
msg += "<table><tr class='test-source'><th>Source: </th><td><pre>" + escapeText(source) + "</pre></td></tr></table>";
}
}
runLoggingCallbacks("log", QUnit, details);
config.current.assertions.push({
result: result,
message: msg
});
},
/**
* Assert that the first two arguments are equal, with an optional message.
* Prints out both actual and expected values.
* @name equal
* @function
* @example equal( format( "Received {0} bytes.", 2), "Received 2 bytes.", "format() replaces {0} with next argument" );
*/
equal: function (actual, expected, message) {
/*jshint eqeqeq:false */
QUnit.push(expected == actual, actual, expected, message);
},
/**
* @name notEqual
* @function
*/
notEqual: function (actual, expected, message) {
/*jshint eqeqeq:false */
QUnit.push(expected != actual, actual, expected, message);
},
/**
* @name propEqual
* @function
*/
propEqual: function (actual, expected, message) {
actual = objectValues(actual);
expected = objectValues(expected);
QUnit.push(QUnit.equiv(actual, expected), actual, expected, message);
},
/**
* @name notPropEqual
* @function
*/
notPropEqual: function (actual, expected, message) {
actual = objectValues(actual);
expected = objectValues(expected);
QUnit.push(!QUnit.equiv(actual, expected), actual, expected, message);
},
/**
* @name deepEqual
* @function
*/
deepEqual: function (actual, expected, message) {
QUnit.push(QUnit.equiv(actual, expected), actual, expected, message);
},
/**
* @name notDeepEqual
* @function
*/
notDeepEqual: function (actual, expected, message) {
QUnit.push(!QUnit.equiv(actual, expected), actual, expected, message);
},
/**
* @name strictEqual
* @function
*/
strictEqual: function (actual, expected, message) {
QUnit.push(expected === actual, actual, expected, message);
},
/**
* @name notStrictEqual
* @function
*/
notStrictEqual: function (actual, expected, message) {
QUnit.push(expected !== actual, actual, expected, message);
},
"throws": function (block, expected, message) {
var actual,
expectedOutput = expected,
ok = false;
// 'expected' is optional
if (typeof expected === "string") {
message = expected;
expected = null;
}
config.current.ignoreGlobalErrors = true;
try {
block.call(config.current.testEnvironment);
} catch (e) {
actual = e;
}
config.current.ignoreGlobalErrors = false;
if (actual) {
// we don't want to validate thrown error
if (!expected) {
ok = true;
expectedOutput = null;
// expected is a regexp
} else if (QUnit.objectType(expected) === "regexp") {
ok = expected.test(errorString(actual));
// expected is a constructor
} else if (actual instanceof expected) {
ok = true;
// expected is a validation function which returns true is validation passed
} else if (expected.call({}, actual) === true) {
expectedOutput = null;
ok = true;
}
QUnit.push(ok, actual, expectedOutput, message);
} else {
QUnit.pushFailure(message, null, "No exception was thrown.");
}
}
};
/**
* @deprecated since 1.8.0
* Kept assertion helpers in root for backwards compatibility.
*/
extend(QUnit, assert);
/**
* @deprecated since 1.9.0
* Kept root "raises()" for backwards compatibility.
* (Note that we don't introduce assert.raises).
*/
QUnit.raises = assert[ "throws" ];
/**
* @deprecated since 1.0.0, replaced with error pushes since 1.3.0
* Kept to avoid TypeErrors for undefined methods.
*/
QUnit.equals = function () {
QUnit.push(false, false, false, "QUnit.equals has been deprecated since 2009 (e88049a0), use QUnit.equal instead");
};
QUnit.same = function () {
QUnit.push(false, false, false, "QUnit.same has been deprecated since 2009 (e88049a0), use QUnit.deepEqual instead");
};
// We want access to the constructor's prototype
(function () {
function F() {
}
F.prototype = QUnit;
QUnit = new F();
// Make F QUnit's constructor so that we can add to the prototype later
QUnit.constructor = F;
}());
/**
* Config object: Maintain internal state
* Later exposed as QUnit.config
* `config` initialized at top of scope
*/
config = {
// The queue of tests to run
queue: [],
// block until document ready
blocking: true,
// when enabled, show only failing tests
// gets persisted through sessionStorage and can be changed in UI via checkbox
hidepassed: false,
// by default, run previously failed tests first
// very useful in combination with "Hide passed tests" checked
reorder: true,
// by default, modify document.title when suite is done
altertitle: true,
// when enabled, all tests must call expect()
requireExpects: false,
// add checkboxes that are persisted in the query-string
// when enabled, the id is set to `true` as a `QUnit.config` property
urlConfig: [
{
id: "noglobals",
label: "Check for Globals",
tooltip: "Enabling this will test if any test introduces new properties on the `window` object. Stored as query-strings."
},
{
id: "notrycatch",
label: "No try-catch",
tooltip: "Enabling this will run tests outside of a try-catch block. Makes debugging exceptions in IE reasonable. Stored as query-strings."
}
],
// Set of all modules.
modules: {},
// logging callback queues
begin: [],
done: [],
log: [],
testStart: [],
testDone: [],
moduleStart: [],
moduleDone: []
};
// Export global variables, unless an 'exports' object exists,
// in that case we assume we're in CommonJS (dealt with on the bottom of the script)
if (typeof exports === "undefined") {
extend(window, QUnit.constructor.prototype);
// Expose QUnit object
window.QUnit = QUnit;
}
// Initialize more QUnit.config and QUnit.urlParams
(function () {
var i,
location = window.location || { search: "", protocol: "file:" },
params = location.search.slice(1).split("&"),
length = params.length,
urlParams = {},
current;
if (params[ 0 ]) {
for (i = 0; i < length; i++) {
current = params[ i ].split("=");
current[ 0 ] = decodeURIComponent(current[ 0 ]);
// allow just a key to turn on a flag, e.g., test.html?noglobals
current[ 1 ] = current[ 1 ] ? decodeURIComponent(current[ 1 ]) : true;
urlParams[ current[ 0 ] ] = current[ 1 ];
}
}
QUnit.urlParams = urlParams;
// String search anywhere in moduleName+testName
config.filter = urlParams.filter;
// Exact match of the module name
config.module = urlParams.module;
config.testNumber = parseInt(urlParams.testNumber, 10) || null;
// Figure out if we're running the tests from a server or not
QUnit.isLocal = location.protocol === "file:";
}());
// Extend QUnit object,
// these after set here because they should not be exposed as global functions
extend(QUnit, {
assert: assert,
config: config,
// Initialize the configuration options
init: function () {
extend(config, {
stats: { all: 0, bad: 0 },
moduleStats: { all: 0, bad: 0 },
started: +new Date(),
updateRate: 1000,
blocking: false,
autostart: true,
autorun: false,
filter: "",
queue: [],
semaphore: 1
});
var tests, banner, result,
qunit = id("qunit");
if (qunit) {
qunit.innerHTML =
"<h1 id='qunit-header'>" + escapeText(document.title) + "</h1>" +
"<h2 id='qunit-banner'></h2>" +
"<div id='qunit-testrunner-toolbar'></div>" +
"<h2 id='qunit-userAgent'></h2>" +
"<ol id='qunit-tests'></ol>";
}
tests = id("qunit-tests");
banner = id("qunit-banner");
result = id("qunit-testresult");
if (tests) {
tests.innerHTML = "";
}
if (banner) {
banner.className = "";
}
if (result) {
result.parentNode.removeChild(result);
}
if (tests) {
result = document.createElement("p");
result.id = "qunit-testresult";
result.className = "result";
tests.parentNode.insertBefore(result, tests);
result.innerHTML = "Running...<br/> ";
}
},
// Resets the test setup. Useful for tests that modify the DOM.
/*
DEPRECATED: Use multiple tests instead of resetting inside a test.
Use testStart or testDone for custom cleanup.
This method will throw an error in 2.0, and will be removed in 2.1
*/
reset: function () {
var fixture = id("qunit-fixture");
if (fixture) {
fixture.innerHTML = config.fixture;
}
},
// Trigger an event on an element.
// @example triggerEvent( document.body, "click" );
triggerEvent: function (elem, type, event) {
if (document.createEvent) {
event = document.createEvent("MouseEvents");
event.initMouseEvent(type, true, true, elem.ownerDocument.defaultView,
0, 0, 0, 0, 0, false, false, false, false, 0, null);
elem.dispatchEvent(event);
} else if (elem.fireEvent) {
elem.fireEvent("on" + type);
}
},
// Safe object type checking
is: function (type, obj) {
return QUnit.objectType(obj) === type;
},
objectType: function (obj) {
if (typeof obj === "undefined") {
return "undefined";
// consider: typeof null === object
}
if (obj === null) {
return "null";
}
var match = toString.call(obj).match(/^\[object\s(.*)\]$/),
type = match && match[1] || "";
switch (type) {
case "Number":
if (isNaN(obj)) {
return "nan";
}
return "number";
case "String":
case "Boolean":
case "Array":
case "Date":
case "RegExp":
case "Function":
return type.toLowerCase();
}
if (typeof obj === "object") {
return "object";
}
return undefined;
},
push: function (result, actual, expected, message) {
if (!config.current) {
throw new Error("assertion outside test context, was " + sourceFromStacktrace());
}
var output, source,
details = {
module: config.current.module,
name: config.current.testName,
result: result,
message: message,
actual: actual,
expected: expected
};
message = escapeText(message) || ( result ? "okay" : "failed" );
message = "<span class='test-message'>" + message + "</span>";
output = message;
if (!result) {
expected = escapeText(QUnit.jsDump.parse(expected));
actual = escapeText(QUnit.jsDump.parse(actual));
output += "<table><tr class='test-expected'><th>Expected: </th><td><pre>" + expected + "</pre></td></tr>";
if (actual !== expected) {
output += "<tr class='test-actual'><th>Result: </th><td><pre>" + actual + "</pre></td></tr>";
output += "<tr class='test-diff'><th>Diff: </th><td><pre>" + QUnit.diff(expected, actual) + "</pre></td></tr>";
}
source = sourceFromStacktrace();
if (source) {
details.source = source;
output += "<tr class='test-source'><th>Source: </th><td><pre>" + escapeText(source) + "</pre></td></tr>";
}
output += "</table>";
}
runLoggingCallbacks("log", QUnit, details);
config.current.assertions.push({
result: !!result,
message: output
});
},
pushFailure: function (message, source, actual) {
if (!config.current) {
throw new Error("pushFailure() assertion outside test context, was " + sourceFromStacktrace(2));
}
var output,
details = {
module: config.current.module,
name: config.current.testName,
result: false,
message: message
};
message = escapeText(message) || "error";
message = "<span class='test-message'>" + message + "</span>";
output = message;
output += "<table>";
if (actual) {
output += "<tr class='test-actual'><th>Result: </th><td><pre>" + escapeText(actual) + "</pre></td></tr>";
}
if (source) {
details.source = source;
output += "<tr class='test-source'><th>Source: </th><td><pre>" + escapeText(source) + "</pre></td></tr>";
}
output += "</table>";
runLoggingCallbacks("log", QUnit, details);
config.current.assertions.push({
result: false,
message: output
});
},
url: function (params) {
params = extend(extend({}, QUnit.urlParams), params);
var key,
querystring = "?";
for (key in params) {
if (hasOwn.call(params, key)) {
querystring += encodeURIComponent(key) + "=" +
encodeURIComponent(params[ key ]) + "&";
}
}
return window.location.protocol + "//" + window.location.host +
window.location.pathname + querystring.slice(0, -1);
},
extend: extend,
id: id,
addEvent: addEvent,
addClass: addClass,
hasClass: hasClass,
removeClass: removeClass
// load, equiv, jsDump, diff: Attached later
});
/**
* @deprecated: Created for backwards compatibility with test runner that set the hook function
* into QUnit.{hook}, instead of invoking it and passing the hook function.
* QUnit.constructor is set to the empty F() above so that we can add to it's prototype here.
* Doing this allows us to tell if the following methods have been overwritten on the actual
* QUnit object.
*/
extend(QUnit.constructor.prototype, {
// Logging callbacks; all receive a single argument with the listed properties
// run test/logs.html for any related changes
begin: registerLoggingCallback("begin"),
// done: { failed, passed, total, runtime }
done: registerLoggingCallback("done"),
// log: { result, actual, expected, message }
log: registerLoggingCallback("log"),
// testStart: { name }
testStart: registerLoggingCallback("testStart"),
// testDone: { name, failed, passed, total, duration }
testDone: registerLoggingCallback("testDone"),
// moduleStart: { name }
moduleStart: registerLoggingCallback("moduleStart"),
// moduleDone: { name, failed, passed, total }
moduleDone: registerLoggingCallback("moduleDone")
});
if (typeof document === "undefined" || document.readyState === "complete") {
config.autorun = true;
}
QUnit.load = function () {
runLoggingCallbacks("begin", QUnit, {});
// Initialize the config, saving the execution queue
var banner, filter, i, label, len, main, ol, toolbar, userAgent, val,
urlConfigCheckboxesContainer, urlConfigCheckboxes, moduleFilter,
numModules = 0,
moduleNames = [],
moduleFilterHtml = "",
urlConfigHtml = "",
oldconfig = extend({}, config);
QUnit.init();
extend(config, oldconfig);
config.blocking = false;
len = config.urlConfig.length;
for (i = 0; i < len; i++) {
val = config.urlConfig[i];
if (typeof val === "string") {
val = {
id: val,
label: val,
tooltip: "[no tooltip available]"
};
}
config[ val.id ] = QUnit.urlParams[ val.id ];
urlConfigHtml += "<input id='qunit-urlconfig-" + escapeText(val.id) +
"' name='" + escapeText(val.id) +
"' type='checkbox'" + ( config[ val.id ] ? " checked='checked'" : "" ) +
" title='" + escapeText(val.tooltip) +
"'><label for='qunit-urlconfig-" + escapeText(val.id) +
"' title='" + escapeText(val.tooltip) + "'>" + val.label + "</label>";
}
for (i in config.modules) {
if (config.modules.hasOwnProperty(i)) {
moduleNames.push(i);
}
}
numModules = moduleNames.length;
moduleNames.sort(function (a, b) {
return a.localeCompare(b);
});
moduleFilterHtml += "<label for='qunit-modulefilter'>Module: </label><select id='qunit-modulefilter' name='modulefilter'><option value='' " +
( config.module === undefined ? "selected='selected'" : "" ) +
">< All Modules ></option>";
for (i = 0; i < numModules; i++) {
moduleFilterHtml += "<option value='" + escapeText(encodeURIComponent(moduleNames[i])) + "' " +
( config.module === moduleNames[i] ? "selected='selected'" : "" ) +
">" + escapeText(moduleNames[i]) + "</option>";
}
moduleFilterHtml += "</select>";
// `userAgent` initialized at top of scope
userAgent = id("qunit-userAgent");
if (userAgent) {
userAgent.innerHTML = navigator.userAgent;
}
// `banner` initialized at top of scope
banner = id("qunit-header");
if (banner) {
banner.innerHTML = "<a href='" + QUnit.url({ filter: undefined, module: undefined, testNumber: undefined }) + "'>" + banner.innerHTML + "</a> ";
}
// `toolbar` initialized at top of scope
toolbar = id("qunit-testrunner-toolbar");
if (toolbar) {
// `filter` initialized at top of scope
filter = document.createElement("input");
filter.type = "checkbox";
filter.id = "qunit-filter-pass";
addEvent(filter, "click", function () {
var tmp,
ol = document.getElementById("qunit-tests");
if (filter.checked) {
ol.className = ol.className + " hidepass";
} else {
tmp = " " + ol.className.replace(/[\n\t\r]/g, " ") + " ";
ol.className = tmp.replace(/ hidepass /, " ");
}
if (defined.sessionStorage) {
if (filter.checked) {
sessionStorage.setItem("qunit-filter-passed-tests", "true");
} else {
sessionStorage.removeItem("qunit-filter-passed-tests");
}
}
});
if (config.hidepassed || defined.sessionStorage && sessionStorage.getItem("qunit-filter-passed-tests")) {
filter.checked = true;
// `ol` initialized at top of scope
ol = document.getElementById("qunit-tests");
ol.className = ol.className + " hidepass";
}
toolbar.appendChild(filter);
// `label` initialized at top of scope
label = document.createElement("label");
label.setAttribute("for", "qunit-filter-pass");
label.setAttribute("title", "Only show tests and assertions that fail. Stored in sessionStorage.");
label.innerHTML = "Hide passed tests";
toolbar.appendChild(label);
urlConfigCheckboxesContainer = document.createElement("span");
urlConfigCheckboxesContainer.innerHTML = urlConfigHtml;
urlConfigCheckboxes = urlConfigCheckboxesContainer.getElementsByTagName("input");
// For oldIE support:
// * Add handlers to the individual elements instead of the container
// * Use "click" instead of "change"
// * Fallback from event.target to event.srcElement
addEvents(urlConfigCheckboxes, "click", function (event) {
var params = {},
target = event.target || event.srcElement;
params[ target.name ] = target.checked ? true : undefined;
window.location = QUnit.url(params);
});
toolbar.appendChild(urlConfigCheckboxesContainer);
if (numModules > 1) {
moduleFilter = document.createElement("span");
moduleFilter.setAttribute("id", "qunit-modulefilter-container");
moduleFilter.innerHTML = moduleFilterHtml;
addEvent(moduleFilter.lastChild, "change", function () {
var selectBox = moduleFilter.getElementsByTagName("select")[0],
selectedModule = decodeURIComponent(selectBox.options[selectBox.selectedIndex].value);
window.location = QUnit.url({
module: ( selectedModule === "" ) ? undefined : selectedModule,
// Remove any existing filters
filter: undefined,
testNumber: undefined
});
});
toolbar.appendChild(moduleFilter);
}
}
// `main` initialized at top of scope
main = id("qunit-fixture");
if (main) {
config.fixture = main.innerHTML;
}
if (config.autostart) {
QUnit.start();
}
};
addEvent(window, "load", QUnit.load);
// `onErrorFnPrev` initialized at top of scope
// Preserve other handlers
onErrorFnPrev = window.onerror;
// Cover uncaught exceptions
// Returning true will suppress the default browser handler,
// returning false will let it run.
window.onerror = function (error, filePath, linerNr) {
var ret = false;
if (onErrorFnPrev) {
ret = onErrorFnPrev(error, filePath, linerNr);
}
// Treat return value as window.onerror itself does,
// Only do our handling if not suppressed.
if (ret !== true) {
if (QUnit.config.current) {
if (QUnit.config.current.ignoreGlobalErrors) {
return true;
}
QUnit.pushFailure(error, filePath + ":" + linerNr);
} else {
QUnit.test("global failure", extend(function () {
QUnit.pushFailure(error, filePath + ":" + linerNr);
}, { validTest: validTest }));
}
return false;
}
return ret;
};
function done() {
config.autorun = true;
// Log the last module results
if (config.currentModule) {
runLoggingCallbacks("moduleDone", QUnit, {
name: config.currentModule,
failed: config.moduleStats.bad,
passed: config.moduleStats.all - config.moduleStats.bad,
total: config.moduleStats.all
});
}
delete config.previousModule;
var i, key,
banner = id("qunit-banner"),
tests = id("qunit-tests"),
runtime = +new Date() - config.started,
passed = config.stats.all - config.stats.bad,
html = [
"Tests completed in ",
runtime,
" milliseconds.<br/>",
"<span class='passed'>",
passed,
"</span> assertions of <span class='total'>",
config.stats.all,
"</span> passed, <span class='failed'>",
config.stats.bad,
"</span> failed."
].join("");
if (banner) {
banner.className = ( config.stats.bad ? "qunit-fail" : "qunit-pass" );
}
if (tests) {
id("qunit-testresult").innerHTML = html;
}
if (config.altertitle && typeof document !== "undefined" && document.title) {
// show ✖ for good, ✔ for bad suite result in title
// use escape sequences in case file gets loaded with non-utf-8-charset
document.title = [
( config.stats.bad ? "\u2716" : "\u2714" ),
document.title.replace(/^[\u2714\u2716] /i, "")
].join(" ");
}
// clear own sessionStorage items if all tests passed
if (config.reorder && defined.sessionStorage && config.stats.bad === 0) {
// `key` & `i` initialized at top of scope
for (i = 0; i < sessionStorage.length; i++) {
key = sessionStorage.key(i++);
if (key.indexOf("qunit-test-") === 0) {
sessionStorage.removeItem(key);
}
}
}
// scroll back to top to show results
if (window.scrollTo) {
window.scrollTo(0, 0);
}
runLoggingCallbacks("done", QUnit, {
failed: config.stats.bad,
passed: passed,
total: config.stats.all,
runtime: runtime
});
}
/** @return Boolean: true if this test should be ran */
function validTest(test) {
var include,
filter = config.filter && config.filter.toLowerCase(),
module = config.module && config.module.toLowerCase(),
fullName = (test.module + ": " + test.testName).toLowerCase();
// Internally-generated tests are always valid
if (test.callback && test.callback.validTest === validTest) {
delete test.callback.validTest;
return true;
}
if (config.testNumber) {
return test.testNumber === config.testNumber;
}
if (module && ( !test.module || test.module.toLowerCase() !== module )) {
return false;
}
if (!filter) {
return true;
}
include = filter.charAt(0) !== "!";
if (!include) {
filter = filter.slice(1);
}
// If the filter matches, we need to honour include
if (fullName.indexOf(filter) !== -1) {
return include;
}
// Otherwise, do the opposite
return !include;
}
// so far supports only Firefox, Chrome and Opera (buggy), Safari (for real exceptions)
// Later Safari and IE10 are supposed to support error.stack as well
// See also https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Error/Stack
function extractStacktrace(e, offset) {
offset = offset === undefined ? 3 : offset;
var stack, include, i;
if (e.stacktrace) {
// Opera
return e.stacktrace.split("\n")[ offset + 3 ];
} else if (e.stack) {
// Firefox, Chrome
stack = e.stack.split("\n");
if (/^error$/i.test(stack[0])) {
stack.shift();
}
if (fileName) {
include = [];
for (i = offset; i < stack.length; i++) {
if (stack[ i ].indexOf(fileName) !== -1) {
break;
}
include.push(stack[ i ]);
}
if (include.length) {
return include.join("\n");
}
}
return stack[ offset ];
} else if (e.sourceURL) {
// Safari, PhantomJS
// hopefully one day Safari provides actual stacktraces
// exclude useless self-reference for generated Error objects
if (/qunit.js$/.test(e.sourceURL)) {
return;
}
// for actual exceptions, this is useful
return e.sourceURL + ":" + e.line;
}
}
function sourceFromStacktrace(offset) {
try {
throw new Error();
} catch (e) {
return extractStacktrace(e, offset);
}
}
/**
* Escape text for attribute or text content.
*/
function escapeText(s) {
if (!s) {
return "";
}
s = s + "";
// Both single quotes and double quotes (for attributes)
return s.replace(/['"<>&]/g, function (s) {
switch (s) {
case "'":
return "'";
case "\"":
return """;
case "<":
return "<";
case ">":
return ">";
case "&":
return "&";
}
});
}
function synchronize(callback, last) {
config.queue.push(callback);
if (config.autorun && !config.blocking) {
process(last);
}
}
function process(last) {
function next() {
process(last);
}
var start = new Date().getTime();
config.depth = config.depth ? config.depth + 1 : 1;
while (config.queue.length && !config.blocking) {
if (!defined.setTimeout || config.updateRate <= 0 || ( ( new Date().getTime() - start ) < config.updateRate )) {
config.queue.shift()();
} else {
setTimeout(next, 13);
break;
}
}
config.depth--;
if (last && !config.blocking && !config.queue.length && config.depth === 0) {
done();
}
}
function saveGlobal() {
config.pollution = [];
if (config.noglobals) {
for (var key in window) {
if (hasOwn.call(window, key)) {
// in Opera sometimes DOM element ids show up here, ignore them
if (/^qunit-test-output/.test(key)) {
continue;
}
config.pollution.push(key);
}
}
}
}
function checkPollution() {
var newGlobals,
deletedGlobals,
old = config.pollution;
saveGlobal();
newGlobals = diff(config.pollution, old);
if (newGlobals.length > 0) {
QUnit.pushFailure("Introduced global variable(s): " + newGlobals.join(", "));
}
deletedGlobals = diff(old, config.pollution);
if (deletedGlobals.length > 0) {
QUnit.pushFailure("Deleted global variable(s): " + deletedGlobals.join(", "));
}
}
// returns a new Array with the elements that are in a but not in b
function diff(a, b) {
var i, j,
result = a.slice();
for (i = 0; i < result.length; i++) {
for (j = 0; j < b.length; j++) {
if (result[i] === b[j]) {
result.splice(i, 1);
i--;
break;
}
}
}
return result;
}
function extend(a, b) {
for (var prop in b) {
if (hasOwn.call(b, prop)) {
// Avoid "Member not found" error in IE8 caused by messing with window.constructor
if (!( prop === "constructor" && a === window )) {
if (b[ prop ] === undefined) {
delete a[ prop ];
} else {
a[ prop ] = b[ prop ];
}
}
}
}
return a;
}
/**
* @param {HTMLElement} elem
* @param {string} type
* @param {Function} fn
*/
function addEvent(elem, type, fn) {
// Standards-based browsers
if (elem.addEventListener) {
elem.addEventListener(type, fn, false);
// IE
} else {
elem.attachEvent("on" + type, fn);
}
}
/**
* @param {Array|NodeList} elems
* @param {string} type
* @param {Function} fn
*/
function addEvents(elems, type, fn) {
var i = elems.length;
while (i--) {
addEvent(elems[i], type, fn);
}
}
function hasClass(elem, name) {
return (" " + elem.className + " ").indexOf(" " + name + " ") > -1;
}
function addClass(elem, name) {
if (!hasClass(elem, name)) {
elem.className += (elem.className ? " " : "") + name;
}
}
function removeClass(elem, name) {
var set = " " + elem.className + " ";
// Class name may appear multiple times
while (set.indexOf(" " + name + " ") > -1) {
set = set.replace(" " + name + " ", " ");
}
// If possible, trim it for prettiness, but not necessarily
elem.className = typeof set.trim === "function" ? set.trim() : set.replace(/^\s+|\s+$/g, "");
}
function id(name) {
return !!( typeof document !== "undefined" && document && document.getElementById ) &&
document.getElementById(name);
}
function registerLoggingCallback(key) {
return function (callback) {
config[key].push(callback);
};
}
// Supports deprecated method of completely overwriting logging callbacks
function runLoggingCallbacks(key, scope, args) {
var i, callbacks;
if (QUnit.hasOwnProperty(key)) {
QUnit[ key ].call(scope, args);
} else {
callbacks = config[ key ];
for (i = 0; i < callbacks.length; i++) {
callbacks[ i ].call(scope, args);
}
}
}
// Test for equality any JavaScript type.
// Author: Philippe Rathé <prathe@gmail.com>
QUnit.equiv = (function () {
// Call the o related callback with the given arguments.
function bindCallbacks(o, callbacks, args) {
var prop = QUnit.objectType(o);
if (prop) {
if (QUnit.objectType(callbacks[ prop ]) === "function") {
return callbacks[ prop ].apply(callbacks, args);
} else {
return callbacks[ prop ]; // or undefined
}
}
}
// the real equiv function
var innerEquiv,
// stack to decide between skip/abort functions
callers = [],
// stack to avoiding loops from circular referencing
parents = [],
parentsB = [],
getProto = Object.getPrototypeOf || function (obj) {
/*jshint camelcase:false */
return obj.__proto__;
},
callbacks = (function () {
// for string, boolean, number and null
function useStrictEquality(b, a) {
/*jshint eqeqeq:false */
if (b instanceof a.constructor || a instanceof b.constructor) {
// to catch short annotation VS 'new' annotation of a
// declaration
// e.g. var i = 1;
// var j = new Number(1);
return a == b;
} else {
return a === b;
}
}
return {
"string": useStrictEquality,
"boolean": useStrictEquality,
"number": useStrictEquality,
"null": useStrictEquality,
"undefined": useStrictEquality,
"nan": function (b) {
return isNaN(b);
},
"date": function (b, a) {
return QUnit.objectType(b) === "date" && a.valueOf() === b.valueOf();
},
"regexp": function (b, a) {
return QUnit.objectType(b) === "regexp" &&
// the regex itself
a.source === b.source &&
// and its modifiers
a.global === b.global &&
// (gmi) ...
a.ignoreCase === b.ignoreCase &&
a.multiline === b.multiline &&
a.sticky === b.sticky;
},
// - skip when the property is a method of an instance (OOP)
// - abort otherwise,
// initial === would have catch identical references anyway
"function": function () {
var caller = callers[callers.length - 1];
return caller !== Object && typeof caller !== "undefined";
},
"array": function (b, a) {
var i, j, len, loop, aCircular, bCircular;
// b could be an object literal here
if (QUnit.objectType(b) !== "array") {
return false;
}
len = a.length;
if (len !== b.length) {
// safe and faster
return false;
}
// track reference to avoid circular references
parents.push(a);
parentsB.push(b);
for (i = 0; i < len; i++) {
loop = false;
for (j = 0; j < parents.length; j++) {
aCircular = parents[j] === a[i];
bCircular = parentsB[j] === b[i];
if (aCircular || bCircular) {
if (a[i] === b[i] || aCircular && bCircular) {
loop = true;
} else {
parents.pop();
parentsB.pop();
return false;
}
}
}
if (!loop && !innerEquiv(a[i], b[i])) {
parents.pop();
parentsB.pop();
return false;
}
}
parents.pop();
parentsB.pop();
return true;
},
"object": function (b, a) {
/*jshint forin:false */
var i, j, loop, aCircular, bCircular,
// Default to true
eq = true,
aProperties = [],
bProperties = [];
// comparing constructors is more strict than using
// instanceof
if (a.constructor !== b.constructor) {
// Allow objects with no prototype to be equivalent to
// objects with Object as their constructor.
if (!(( getProto(a) === null && getProto(b) === Object.prototype ) ||
( getProto(b) === null && getProto(a) === Object.prototype ) )) {
return false;
}
}
// stack constructor before traversing properties
callers.push(a.constructor);
// track reference to avoid circular references
parents.push(a);
parentsB.push(b);
// be strict: don't ensure hasOwnProperty and go deep
for (i in a) {
loop = false;
for (j = 0; j < parents.length; j++) {
aCircular = parents[j] === a[i];
bCircular = parentsB[j] === b[i];
if (aCircular || bCircular) {
if (a[i] === b[i] || aCircular && bCircular) {
loop = true;
} else {
eq = false;
break;
}
}
}
aProperties.push(i);
if (!loop && !innerEquiv(a[i], b[i])) {
eq = false;
break;
}
}
parents.pop();
parentsB.pop();
callers.pop(); // unstack, we are done
for (i in b) {
bProperties.push(i); // collect b's properties
}
// Ensures identical properties name
return eq && innerEquiv(aProperties.sort(), bProperties.sort());
}
};
}());
innerEquiv = function () { // can take multiple arguments
var args = [].slice.apply(arguments);
if (args.length < 2) {
return true; // end transition
}
return (function (a, b) {
if (a === b) {
return true; // catch the most you can
} else if (a === null || b === null || typeof a === "undefined" ||
typeof b === "undefined" ||
QUnit.objectType(a) !== QUnit.objectType(b)) {
return false; // don't lose time with error prone cases
} else {
return bindCallbacks(a, callbacks, [ b, a ]);
}
// apply transition with (1..n) arguments
}(args[0], args[1]) && innerEquiv.apply(this, args.splice(1, args.length - 1)) );
};
return innerEquiv;
}());
/**
* jsDump Copyright (c) 2008 Ariel Flesler - aflesler(at)gmail(dot)com |
* http://flesler.blogspot.com Licensed under BSD
* (http://www.opensource.org/licenses/bsd-license.php) Date: 5/15/2008
*
* @projectDescription Advanced and extensible data dumping for Javascript.
* @version 1.0.0
* @author Ariel Flesler
* @link {http://flesler.blogspot.com/2008/05/jsdump-pretty-dump-of-any-javascript.html}
*/
QUnit.jsDump = (function () {
function quote(str) {
return "\"" + str.toString().replace(/"/g, "\\\"") + "\"";
}
function literal(o) {
return o + "";
}
function join(pre, arr, post) {
var s = jsDump.separator(),
base = jsDump.indent(),
inner = jsDump.indent(1);
if (arr.join) {
arr = arr.join("," + s + inner);
}
if (!arr) {
return pre + post;
}
return [ pre, inner + arr, base + post ].join(s);
}
function array(arr, stack) {
var i = arr.length, ret = new Array(i);
this.up();
while (i--) {
ret[i] = this.parse(arr[i], undefined, stack);
}
this.down();
return join("[", ret, "]");
}
var reName = /^function (\w+)/,
jsDump = {
// type is used mostly internally, you can fix a (custom)type in advance
parse: function (obj, type, stack) {
stack = stack || [ ];
var inStack, res,
parser = this.parsers[ type || this.typeOf(obj) ];
type = typeof parser;
inStack = inArray(obj, stack);
if (inStack !== -1) {
return "recursion(" + (inStack - stack.length) + ")";
}
if (type === "function") {
stack.push(obj);
res = parser.call(this, obj, stack);
stack.pop();
return res;
}
return ( type === "string" ) ? parser : this.parsers.error;
},
typeOf: function (obj) {
var type;
if (obj === null) {
type = "null";
} else if (typeof obj === "undefined") {
type = "undefined";
} else if (QUnit.is("regexp", obj)) {
type = "regexp";
} else if (QUnit.is("date", obj)) {
type = "date";
} else if (QUnit.is("function", obj)) {
type = "function";
} else if (typeof obj.setInterval !== undefined && typeof obj.document !== "undefined" && typeof obj.nodeType === "undefined") {
type = "window";
} else if (obj.nodeType === 9) {
type = "document";
} else if (obj.nodeType) {
type = "node";
} else if (
// native arrays
toString.call(obj) === "[object Array]" ||
// NodeList objects
( typeof obj.length === "number" && typeof obj.item !== "undefined" && ( obj.length ? obj.item(0) === obj[0] : ( obj.item(0) === null && typeof obj[0] === "undefined" ) ) )
) {
type = "array";
} else if (obj.constructor === Error.prototype.constructor) {
type = "error";
} else {
type = typeof obj;
}
return type;
},
separator: function () {
return this.multiline ? this.HTML ? "<br />" : "\n" : this.HTML ? " " : " ";
},
// extra can be a number, shortcut for increasing-calling-decreasing
indent: function (extra) {
if (!this.multiline) {
return "";
}
var chr = this.indentChar;
if (this.HTML) {
chr = chr.replace(/\t/g, " ").replace(/ /g, " ");
}
return new Array(this.depth + ( extra || 0 )).join(chr);
},
up: function (a) {
this.depth += a || 1;
},
down: function (a) {
this.depth -= a || 1;
},
setParser: function (name, parser) {
this.parsers[name] = parser;
},
// The next 3 are exposed so you can use them
quote: quote,
literal: literal,
join: join,
//
depth: 1,
// This is the list of parsers, to modify them, use jsDump.setParser
parsers: {
window: "[Window]",
document: "[Document]",
error: function (error) {
return "Error(\"" + error.message + "\")";
},
unknown: "[Unknown]",
"null": "null",
"undefined": "undefined",
"function": function (fn) {
var ret = "function",
// functions never have name in IE
name = "name" in fn ? fn.name : (reName.exec(fn) || [])[1];
if (name) {
ret += " " + name;
}
ret += "( ";
ret = [ ret, QUnit.jsDump.parse(fn, "functionArgs"), "){" ].join("");
return join(ret, QUnit.jsDump.parse(fn, "functionCode"), "}");
},
array: array,
nodelist: array,
"arguments": array,
object: function (map, stack) {
/*jshint forin:false */
var ret = [ ], keys, key, val, i;
QUnit.jsDump.up();
keys = [];
for (key in map) {
keys.push(key);
}
keys.sort();
for (i = 0; i < keys.length; i++) {
key = keys[ i ];
val = map[ key ];
ret.push(QUnit.jsDump.parse(key, "key") + ": " + QUnit.jsDump.parse(val, undefined, stack));
}
QUnit.jsDump.down();
return join("{", ret, "}");
},
node: function (node) {
var len, i, val,
open = QUnit.jsDump.HTML ? "<" : "<",
close = QUnit.jsDump.HTML ? ">" : ">",
tag = node.nodeName.toLowerCase(),
ret = open + tag,
attrs = node.attributes;
if (attrs) {
for (i = 0, len = attrs.length; i < len; i++) {
val = attrs[i].nodeValue;
// IE6 includes all attributes in .attributes, even ones not explicitly set.
// Those have values like undefined, null, 0, false, "" or "inherit".
if (val && val !== "inherit") {
ret += " " + attrs[i].nodeName + "=" + QUnit.jsDump.parse(val, "attribute");
}
}
}
ret += close;
// Show content of TextNode or CDATASection
if (node.nodeType === 3 || node.nodeType === 4) {
ret += node.nodeValue;
}
return ret + open + "/" + tag + close;
},
// function calls it internally, it's the arguments part of the function
functionArgs: function (fn) {
var args,
l = fn.length;
if (!l) {
return "";
}
args = new Array(l);
while (l--) {
// 97 is 'a'
args[l] = String.fromCharCode(97 + l);
}
return " " + args.join(", ") + " ";
},
// object calls it internally, the key part of an item in a map
key: quote,
// function calls it internally, it's the content of the function
functionCode: "[code]",
// node calls it internally, it's an html attribute value
attribute: quote,
string: quote,
date: quote,
regexp: literal,
number: literal,
"boolean": literal
},
// if true, entities are escaped ( <, >, \t, space and \n )
HTML: false,
// indentation unit
indentChar: " ",
// if true, items in a collection, are separated by a \n, else just a space.
multiline: true
};
return jsDump;
}());
// from jquery.js
function inArray(elem, array) {
if (array.indexOf) {
return array.indexOf(elem);
}
for (var i = 0, length = array.length; i < length; i++) {
if (array[ i ] === elem) {
return i;
}
}
return -1;
}
/*
* Javascript Diff Algorithm
* By John Resig (http://ejohn.org/)
* Modified by Chu Alan "sprite"
*
* Released under the MIT license.
*
* More Info:
* http://ejohn.org/projects/javascript-diff-algorithm/
*
* Usage: QUnit.diff(expected, actual)
*
* QUnit.diff( "the quick brown fox jumped over", "the quick fox jumps over" ) == "the quick <del>brown </del> fox <del>jumped </del><ins>jumps </ins> over"
*/
QUnit.diff = (function () {
/*jshint eqeqeq:false, eqnull:true */
function diff(o, n) {
var i,
ns = {},
os = {};
for (i = 0; i < n.length; i++) {
if (!hasOwn.call(ns, n[i])) {
ns[ n[i] ] = {
rows: [],
o: null
};
}
ns[ n[i] ].rows.push(i);
}
for (i = 0; i < o.length; i++) {
if (!hasOwn.call(os, o[i])) {
os[ o[i] ] = {
rows: [],
n: null
};
}
os[ o[i] ].rows.push(i);
}
for (i in ns) {
if (hasOwn.call(ns, i)) {
if (ns[i].rows.length === 1 && hasOwn.call(os, i) && os[i].rows.length === 1) {
n[ ns[i].rows[0] ] = {
text: n[ ns[i].rows[0] ],
row: os[i].rows[0]
};
o[ os[i].rows[0] ] = {
text: o[ os[i].rows[0] ],
row: ns[i].rows[0]
};
}
}
}
for (i = 0; i < n.length - 1; i++) {
if (n[i].text != null && n[ i + 1 ].text == null && n[i].row + 1 < o.length && o[ n[i].row + 1 ].text == null &&
n[ i + 1 ] == o[ n[i].row + 1 ]) {
n[ i + 1 ] = {
text: n[ i + 1 ],
row: n[i].row + 1
};
o[ n[i].row + 1 ] = {
text: o[ n[i].row + 1 ],
row: i + 1
};
}
}
for (i = n.length - 1; i > 0; i--) {
if (n[i].text != null && n[ i - 1 ].text == null && n[i].row > 0 && o[ n[i].row - 1 ].text == null &&
n[ i - 1 ] == o[ n[i].row - 1 ]) {
n[ i - 1 ] = {
text: n[ i - 1 ],
row: n[i].row - 1
};
o[ n[i].row - 1 ] = {
text: o[ n[i].row - 1 ],
row: i - 1
};
}
}
return {
o: o,
n: n
};
}
return function (o, n) {
o = o.replace(/\s+$/, "");
n = n.replace(/\s+$/, "");
var i, pre,
str = "",
out = diff(o === "" ? [] : o.split(/\s+/), n === "" ? [] : n.split(/\s+/)),
oSpace = o.match(/\s+/g),
nSpace = n.match(/\s+/g);
if (oSpace == null) {
oSpace = [ " " ];
}
else {
oSpace.push(" ");
}
if (nSpace == null) {
nSpace = [ " " ];
}
else {
nSpace.push(" ");
}
if (out.n.length === 0) {
for (i = 0; i < out.o.length; i++) {
str += "<del>" + out.o[i] + oSpace[i] + "</del>";
}
}
else {
if (out.n[0].text == null) {
for (n = 0; n < out.o.length && out.o[n].text == null; n++) {
str += "<del>" + out.o[n] + oSpace[n] + "</del>";
}
}
for (i = 0; i < out.n.length; i++) {
if (out.n[i].text == null) {
str += "<ins>" + out.n[i] + nSpace[i] + "</ins>";
}
else {
// `pre` initialized at top of scope
pre = "";
for (n = out.n[i].row + 1; n < out.o.length && out.o[n].text == null; n++) {
pre += "<del>" + out.o[n] + oSpace[n] + "</del>";
}
str += " " + out.n[i].text + nSpace[i] + pre;
}
}
}
return str;
};
}());
// for CommonJS environments, export everything
if (typeof exports !== "undefined") {
extend(exports, QUnit.constructor.prototype);
}
// get at whatever the global object is, like window in browsers
}((function () {
return this;
}.call())));