123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389 |
- 'use strict';
- /* eslint-env browser */
- /**
- * @module HTML
- */
- /**
- * Module dependencies.
- */
- var Base = require('./base');
- var utils = require('../utils');
- var Progress = require('../browser/progress');
- var escapeRe = require('escape-string-regexp');
- var constants = require('../runner').constants;
- var EVENT_TEST_PASS = constants.EVENT_TEST_PASS;
- var EVENT_TEST_FAIL = constants.EVENT_TEST_FAIL;
- var EVENT_SUITE_BEGIN = constants.EVENT_SUITE_BEGIN;
- var EVENT_SUITE_END = constants.EVENT_SUITE_END;
- var EVENT_TEST_PENDING = constants.EVENT_TEST_PENDING;
- var escape = utils.escape;
- /**
- * Save timer references to avoid Sinon interfering (see GH-237).
- */
- var Date = global.Date;
- /**
- * Expose `HTML`.
- */
- exports = module.exports = HTML;
- /**
- * Stats template.
- */
- var statsTemplate =
- '<ul id="mocha-stats">' +
- '<li class="progress"><canvas width="40" height="40"></canvas></li>' +
- '<li class="passes"><a href="javascript:void(0);">passes:</a> <em>0</em></li>' +
- '<li class="failures"><a href="javascript:void(0);">failures:</a> <em>0</em></li>' +
- '<li class="duration">duration: <em>0</em>s</li>' +
- '</ul>';
- var playIcon = '‣';
- /**
- * Initialize a new `HTML` reporter.
- *
- * @public
- * @class
- * @memberof Mocha.reporters
- * @extends Mocha.reporters.Base
- * @param {Runner} runner
- */
- function HTML(runner) {
- Base.call(this, runner);
- var self = this;
- var stats = this.stats;
- var stat = fragment(statsTemplate);
- var items = stat.getElementsByTagName('li');
- var passes = items[1].getElementsByTagName('em')[0];
- var passesLink = items[1].getElementsByTagName('a')[0];
- var failures = items[2].getElementsByTagName('em')[0];
- var failuresLink = items[2].getElementsByTagName('a')[0];
- var duration = items[3].getElementsByTagName('em')[0];
- var canvas = stat.getElementsByTagName('canvas')[0];
- var report = fragment('<ul id="mocha-report"></ul>');
- var stack = [report];
- var progress;
- var ctx;
- var root = document.getElementById('mocha');
- if (canvas.getContext) {
- var ratio = window.devicePixelRatio || 1;
- canvas.style.width = canvas.width;
- canvas.style.height = canvas.height;
- canvas.width *= ratio;
- canvas.height *= ratio;
- ctx = canvas.getContext('2d');
- ctx.scale(ratio, ratio);
- progress = new Progress();
- }
- if (!root) {
- return error('#mocha div missing, add it to your document');
- }
- // pass toggle
- on(passesLink, 'click', function(evt) {
- evt.preventDefault();
- unhide();
- var name = /pass/.test(report.className) ? '' : ' pass';
- report.className = report.className.replace(/fail|pass/g, '') + name;
- if (report.className.trim()) {
- hideSuitesWithout('test pass');
- }
- });
- // failure toggle
- on(failuresLink, 'click', function(evt) {
- evt.preventDefault();
- unhide();
- var name = /fail/.test(report.className) ? '' : ' fail';
- report.className = report.className.replace(/fail|pass/g, '') + name;
- if (report.className.trim()) {
- hideSuitesWithout('test fail');
- }
- });
- root.appendChild(stat);
- root.appendChild(report);
- if (progress) {
- progress.size(40);
- }
- runner.on(EVENT_SUITE_BEGIN, function(suite) {
- if (suite.root) {
- return;
- }
- // suite
- var url = self.suiteURL(suite);
- var el = fragment(
- '<li class="suite"><h1><a href="%s">%s</a></h1></li>',
- url,
- escape(suite.title)
- );
- // container
- stack[0].appendChild(el);
- stack.unshift(document.createElement('ul'));
- el.appendChild(stack[0]);
- });
- runner.on(EVENT_SUITE_END, function(suite) {
- if (suite.root) {
- updateStats();
- return;
- }
- stack.shift();
- });
- runner.on(EVENT_TEST_PASS, function(test) {
- var url = self.testURL(test);
- var markup =
- '<li class="test pass %e"><h2>%e<span class="duration">%ems</span> ' +
- '<a href="%s" class="replay">' +
- playIcon +
- '</a></h2></li>';
- var el = fragment(markup, test.speed, test.title, test.duration, url);
- self.addCodeToggle(el, test.body);
- appendToStack(el);
- updateStats();
- });
- runner.on(EVENT_TEST_FAIL, function(test) {
- var el = fragment(
- '<li class="test fail"><h2>%e <a href="%e" class="replay">' +
- playIcon +
- '</a></h2></li>',
- test.title,
- self.testURL(test)
- );
- var stackString; // Note: Includes leading newline
- var message = test.err.toString();
- // <=IE7 stringifies to [Object Error]. Since it can be overloaded, we
- // check for the result of the stringifying.
- if (message === '[object Error]') {
- message = test.err.message;
- }
- if (test.err.stack) {
- var indexOfMessage = test.err.stack.indexOf(test.err.message);
- if (indexOfMessage === -1) {
- stackString = test.err.stack;
- } else {
- stackString = test.err.stack.substr(
- test.err.message.length + indexOfMessage
- );
- }
- } else if (test.err.sourceURL && test.err.line !== undefined) {
- // Safari doesn't give you a stack. Let's at least provide a source line.
- stackString = '\n(' + test.err.sourceURL + ':' + test.err.line + ')';
- }
- stackString = stackString || '';
- if (test.err.htmlMessage && stackString) {
- el.appendChild(
- fragment(
- '<div class="html-error">%s\n<pre class="error">%e</pre></div>',
- test.err.htmlMessage,
- stackString
- )
- );
- } else if (test.err.htmlMessage) {
- el.appendChild(
- fragment('<div class="html-error">%s</div>', test.err.htmlMessage)
- );
- } else {
- el.appendChild(
- fragment('<pre class="error">%e%e</pre>', message, stackString)
- );
- }
- self.addCodeToggle(el, test.body);
- appendToStack(el);
- updateStats();
- });
- runner.on(EVENT_TEST_PENDING, function(test) {
- var el = fragment(
- '<li class="test pass pending"><h2>%e</h2></li>',
- test.title
- );
- appendToStack(el);
- updateStats();
- });
- function appendToStack(el) {
- // Don't call .appendChild if #mocha-report was already .shift()'ed off the stack.
- if (stack[0]) {
- stack[0].appendChild(el);
- }
- }
- function updateStats() {
- // TODO: add to stats
- var percent = ((stats.tests / runner.total) * 100) | 0;
- if (progress) {
- progress.update(percent).draw(ctx);
- }
- // update stats
- var ms = new Date() - stats.start;
- text(passes, stats.passes);
- text(failures, stats.failures);
- text(duration, (ms / 1000).toFixed(2));
- }
- }
- /**
- * Makes a URL, preserving querystring ("search") parameters.
- *
- * @param {string} s
- * @return {string} A new URL.
- */
- function makeUrl(s) {
- var search = window.location.search;
- // Remove previous grep query parameter if present
- if (search) {
- search = search.replace(/[?&]grep=[^&\s]*/g, '').replace(/^&/, '?');
- }
- return (
- window.location.pathname +
- (search ? search + '&' : '?') +
- 'grep=' +
- encodeURIComponent(escapeRe(s))
- );
- }
- /**
- * Provide suite URL.
- *
- * @param {Object} [suite]
- */
- HTML.prototype.suiteURL = function(suite) {
- return makeUrl(suite.fullTitle());
- };
- /**
- * Provide test URL.
- *
- * @param {Object} [test]
- */
- HTML.prototype.testURL = function(test) {
- return makeUrl(test.fullTitle());
- };
- /**
- * Adds code toggle functionality for the provided test's list element.
- *
- * @param {HTMLLIElement} el
- * @param {string} contents
- */
- HTML.prototype.addCodeToggle = function(el, contents) {
- var h2 = el.getElementsByTagName('h2')[0];
- on(h2, 'click', function() {
- pre.style.display = pre.style.display === 'none' ? 'block' : 'none';
- });
- var pre = fragment('<pre><code>%e</code></pre>', utils.clean(contents));
- el.appendChild(pre);
- pre.style.display = 'none';
- };
- /**
- * Display error `msg`.
- *
- * @param {string} msg
- */
- function error(msg) {
- document.body.appendChild(fragment('<div id="mocha-error">%s</div>', msg));
- }
- /**
- * Return a DOM fragment from `html`.
- *
- * @param {string} html
- */
- function fragment(html) {
- var args = arguments;
- var div = document.createElement('div');
- var i = 1;
- div.innerHTML = html.replace(/%([se])/g, function(_, type) {
- switch (type) {
- case 's':
- return String(args[i++]);
- case 'e':
- return escape(args[i++]);
- // no default
- }
- });
- return div.firstChild;
- }
- /**
- * Check for suites that do not have elements
- * with `classname`, and hide them.
- *
- * @param {text} classname
- */
- function hideSuitesWithout(classname) {
- var suites = document.getElementsByClassName('suite');
- for (var i = 0; i < suites.length; i++) {
- var els = suites[i].getElementsByClassName(classname);
- if (!els.length) {
- suites[i].className += ' hidden';
- }
- }
- }
- /**
- * Unhide .hidden suites.
- */
- function unhide() {
- var els = document.getElementsByClassName('suite hidden');
- for (var i = 0; i < els.length; ++i) {
- els[i].className = els[i].className.replace('suite hidden', 'suite');
- }
- }
- /**
- * Set an element's text contents.
- *
- * @param {HTMLElement} el
- * @param {string} contents
- */
- function text(el, contents) {
- if (el.textContent) {
- el.textContent = contents;
- } else {
- el.innerText = contents;
- }
- }
- /**
- * Listen on `event` with callback `fn`.
- */
- function on(el, event, fn) {
- if (el.addEventListener) {
- el.addEventListener(event, fn, false);
- } else {
- el.attachEvent('on' + event, fn);
- }
- }
- HTML.browserOnly = true;
|