From ff00e9fd33a3a056611e28186f4de3b0d12b3912 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 10:23:54 +0100 Subject: [PATCH 01/17] Add JetBrains stuff to .gitignore --- .gitignore | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index 1cc5b6e..90e2c4d 100644 --- a/.gitignore +++ b/.gitignore @@ -7,4 +7,5 @@ src/*.map *.diff *.patch .DS_Store -settings.json \ No newline at end of file +settings.json +.idea/ From 5d0681a1f5d5a1caad1727bc8da2fccede113dd4 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 10:31:12 +0100 Subject: [PATCH 02/17] Fix build error in jshint --- Gruntfile.js | 1 + 1 file changed, 1 insertion(+) diff --git a/Gruntfile.js b/Gruntfile.js index e584c33..05f20fe 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -125,6 +125,7 @@ module.exports = function (grunt) undef: true, eqnull: true, browser: true, + reporterOutput: "", globals: { jQuery: true, $: true, From d91aa2392c41ee61321fe326acf42798e3a40596 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 10:38:27 +0100 Subject: [PATCH 03/17] Format package.json --- package.json | 23 +++++++++++++---------- 1 file changed, 13 insertions(+), 10 deletions(-) diff --git a/package.json b/package.json index 2528243..ef1d91f 100644 --- a/package.json +++ b/package.json @@ -35,28 +35,31 @@ "Accessibility" ], "licenses": [ - { "type": "MIT", "url": "http://www.opensource.org/licenses/MIT" } + { + "type": "MIT", + "url": "http://www.opensource.org/licenses/MIT" + } ], "dependencies": { "jquery": ">=1.9.0", "bootstrap": ">=3.1.1" }, "devDependencies": { - "grunt": "~0.4.5", + "grunt": "^0.4.5", "grunt-cli": "~0.1.13", - "grunt-contrib-less": "~0.11.0", + "grunt-contrib-clean": "~0.5.0", + "grunt-contrib-compress": "~0.8.0", + "grunt-contrib-concat": "~0.4.0", "grunt-contrib-csslint": "~0.3.1", "grunt-contrib-cssmin": "~0.10.0", - "grunt-contrib-qunit": "~0.4.0", "grunt-contrib-jshint": "~0.10.0", + "grunt-contrib-less": "~0.11.0", + "grunt-contrib-qunit": "~0.4.0", "grunt-contrib-uglify": "~0.9.2", - "grunt-contrib-concat": "~0.4.0", "grunt-contrib-yuidoc": "~0.5.2", - "grunt-contrib-clean": "~0.5.0", - "grunt-contrib-compress": "~0.8.0", + "grunt-exec": "~0.4.5", "grunt-nuget": "~0.1.4", - "grunt-regex-replace": "~0.2.6", - "grunt-exec": "~0.4.5" + "grunt-regex-replace": "~0.2.6" }, "readmeFilename": "README.md" -} \ No newline at end of file +} From cb51a55453607a8f3770bf672e0238584efaa96c Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 10:40:29 +0100 Subject: [PATCH 04/17] Fix license var in package.json The old style is deprecated. --- Gruntfile.js | 23 +++++++++-------------- package.json | 7 +------ 2 files changed, 10 insertions(+), 20 deletions(-) diff --git a/Gruntfile.js b/Gruntfile.js index 05f20fe..9ea7e55 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -1,6 +1,5 @@ /*jshint node:true*/ -module.exports = function (grunt) -{ +module.exports = function (grunt) { "use strict"; /* Hint: Using grunt-strip-code to remove comments from the release file */ @@ -10,7 +9,7 @@ module.exports = function (grunt) fontawesome: 'fa', banner: '/*! <%= "\\r\\n * " + pkg.title %> v<%= pkg.version %> - <%= grunt.template.today("mm/dd/yyyy") + "\\r\\n" %>' + ' * Copyright (c) 2014-<%= grunt.template.today("yyyy") %> <%= pkg.author.name %> <%= (pkg.homepage ? "(" + pkg.homepage + ")" : "") + "\\r\\n" %>' + - ' * Licensed under <%= pkg.licenses[0].type + " " + pkg.licenses[0].url + "\\r\\n */\\r\\n" %>', + ' * Licensed under the <%= pkg.license + " license. See LICENSE.txt for more details." + "\\r\\n */\\r\\n" %>', folders: { dist: "dist", docs: "docs", @@ -57,20 +56,17 @@ module.exports = function (grunt) separator: '\r\n\r\n', banner: '<%= banner %>;(function ($, window, undefined)\r\n{\r\n /*jshint validthis: true */\r\n "use strict";\r\n\r\n', footer: '\r\n})(jQuery, window);', - process: function(src, filepath) - { + process: function (src, filepath) { var result = src.trim().replace(/(.+?\r\n)/gm, ' $1'), end = [0, ""], lastChar = result[result.length - 1]; - if (lastChar === ";") - { + if (lastChar === ";") { end = (result[result.length - 2] === ")") ? - (result[result.length - 2] === "}") ? - [3, " });"] : [2, ");"] : [2, " };"]; + (result[result.length - 2] === "}") ? + [3, " });"] : [2, ");"] : [2, " };"]; } - else if (lastChar === "}") - { + else if (lastChar === "}") { end = [1, " }"]; } @@ -211,7 +207,7 @@ module.exports = function (grunt) files: [ { flatten: true, - expand: true, + expand: true, src: ['<%= folders.dist %>/*.js', '<%= folders.dist %>/*.css'], dest: '/' } ] @@ -248,8 +244,7 @@ module.exports = function (grunt) grunt.loadNpmTasks('grunt-nuget'); grunt.loadNpmTasks('grunt-regex-replace'); - grunt.registerMultiTask('version', 'sets version tag', function () - { + grunt.registerMultiTask('version', 'sets version tag', function () { var pkg = grunt.file.readJSON(this.data.src); pkg["version"] = this.data.options.version; grunt.file.write(this.data.src, JSON.stringify(pkg, null, 4)); diff --git a/package.json b/package.json index ef1d91f..880db84 100644 --- a/package.json +++ b/package.json @@ -34,12 +34,7 @@ "HTML5", "Accessibility" ], - "licenses": [ - { - "type": "MIT", - "url": "http://www.opensource.org/licenses/MIT" - } - ], + "license": "MIT", "dependencies": { "jquery": ">=1.9.0", "bootstrap": ">=3.1.1" From 45e841225a98cea70ca5c8c70306a7ba467a1928 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 11:01:30 +0100 Subject: [PATCH 05/17] Lock jquery and Bootstrap versions to then-current major --- package.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 880db84..1062d61 100644 --- a/package.json +++ b/package.json @@ -34,10 +34,10 @@ "HTML5", "Accessibility" ], - "license": "MIT", + "license": "MIT", "dependencies": { - "jquery": ">=1.9.0", - "bootstrap": ">=3.1.1" + "jquery": "^1.9", + "bootstrap": "3.1.1 - 3.3" }, "devDependencies": { "grunt": "^0.4.5", From 54fd7eceb1bfb18b2e347d1c78e493ac701fe7d7 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 11:10:54 +0100 Subject: [PATCH 06/17] Update repository URIs and copyright statements --- Gruntfile.js | 2 +- LICENSE.txt | 3 ++- README.md | 7 +++++-- demo/index.htm | 1 + demo/send.htm | 1 + jQuery.Bootgrid.nuspec | 4 ++-- package.json | 4 ++-- 7 files changed, 14 insertions(+), 8 deletions(-) diff --git a/Gruntfile.js b/Gruntfile.js index 9ea7e55..3af122a 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -8,7 +8,7 @@ module.exports = function (grunt) { pkg: grunt.file.readJSON('package.json'), fontawesome: 'fa', banner: '/*! <%= "\\r\\n * " + pkg.title %> v<%= pkg.version %> - <%= grunt.template.today("mm/dd/yyyy") + "\\r\\n" %>' + - ' * Copyright (c) 2014-<%= grunt.template.today("yyyy") %> <%= pkg.author.name %> <%= (pkg.homepage ? "(" + pkg.homepage + ")" : "") + "\\r\\n" %>' + + ' * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-<%= grunt.template.today("yyyy") %> <%= pkg.author.name %> <%= (pkg.homepage ? "(" + pkg.homepage + ")" : "") + "\\r\\n" %>' + ' * Licensed under the <%= pkg.license + " license. See LICENSE.txt for more details." + "\\r\\n */\\r\\n" %>', folders: { dist: "dist", diff --git a/LICENSE.txt b/LICENSE.txt index 6c71285..12bcaaa 100644 --- a/LICENSE.txt +++ b/LICENSE.txt @@ -1,6 +1,7 @@ The MIT License (MIT) -Copyright (c) 2014-2015 Rafael J. Staib +Copyright © 2014-2015 Rafael J. Staib +Copyright © 2018 Deciso Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/README.md b/README.md index 9ef1416..5c0a613 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -jQuery Bootgrid Plugin [![Build Status](http://img.shields.io/travis/rstaib/jquery-bootgrid/master.svg?style=flat-square)](https://travis-ci.org/rstaib/jquery-bootgrid) ![Bower version](http://img.shields.io/bower/v/jquery.bootgrid.svg?style=flat-square) ![NuGet version](http://img.shields.io/nuget/v/jquery.bootgrid.svg?style=flat-square) ![NPM version](http://img.shields.io/npm/v/jquery-bootgrid.svg?style=flat-square) ![Gratipay](http://img.shields.io/gratipay/RafaelStaib.svg?style=flat-square) +jQuery Bootgrid Plugin [![Build Status](http://img.shields.io/travis/opnsense/jquery-bootgrid/master.svg?style=flat-square)](https://travis-ci.org/opnsense/jquery-bootgrid) ![Bower version](http://img.shields.io/bower/v/jquery.bootgrid.svg?style=flat-square) ![NuGet version](http://img.shields.io/nuget/v/jquery.bootgrid.svg?style=flat-square) ![NPM version](http://img.shields.io/npm/v/jquery-bootgrid.svg?style=flat-square) ![Gratipay](http://img.shields.io/gratipay/RafaelStaib.svg?style=flat-square) ============ Nice, sleek and intuitive. A grid control especially designed for bootstrap. @@ -59,4 +59,7 @@ Instructions will follow soon! ## License -Copyright (c) 2014-2015 Rafael J. Staib Licensed under the [MIT license](https://github.com/rstaib/jquery-bootgrid/blob/master/LICENSE.txt). +Copyright © 2014-2015 Rafael J. Staib +Copyright © 2018 Deciso B.V. + +Licensed under the [MIT license](https://github.com/opnsense/jquery-bootgrid/blob/master/LICENSE.txt). diff --git a/demo/index.htm b/demo/index.htm index ae13225..a4244a5 100644 --- a/demo/index.htm +++ b/demo/index.htm @@ -171,6 +171,7 @@
© Copyright 2014-2015, Rafael Staib + © Copyright 2018 Deciso B.V.
diff --git a/demo/send.htm b/demo/send.htm index 1299d18..8c760be 100644 --- a/demo/send.htm +++ b/demo/send.htm @@ -133,6 +133,7 @@
© Copyright 2014-2015, Rafael Staib + © Copyright 2018 Deciso B.V.
diff --git a/jQuery.Bootgrid.nuspec b/jQuery.Bootgrid.nuspec index be85755..b95678a 100644 --- a/jQuery.Bootgrid.nuspec +++ b/jQuery.Bootgrid.nuspec @@ -6,13 +6,13 @@ jQuery Bootgrid Rafael Staib r_staib - https://github.com/rstaib/jquery-bootgrid/blob/master/LICENSE.txt + https://github.com/opnsense/jquery-bootgrid/blob/master/LICENSE.txt http://www.jquery-bootgrid.com http://www.jquery-bootgrid.com/icon.png true Nice, sleek and intuitive. A grid control especially designed for bootstrap. Nice, sleek and intuitive. A grid control especially designed for bootstrap. - � Copyright 2014-2015, Rafael Staib + Copyright © 2014-2015, Rafael Staib; Copyright © 2018, Deciso B.V. jQuery, Grid, Table, Bootstrap, Accessibility, HTML5, Sorting, Filtering, UI diff --git a/package.json b/package.json index 1062d61..c6a3b49 100644 --- a/package.json +++ b/package.json @@ -10,13 +10,13 @@ "email": "me@rafaelstaib.com", "url": "http://www.rafaelstaib.com" }, - "bugs": "https://github.com/rstaib/jquery-bootgrid/issues", + "bugs": "https://github.com/opnsense/jquery-bootgrid/issues", "scripts": { "test": "grunt" }, "repository": { "type": "git", - "url": "https://github.com/rstaib/jquery-bootgrid.git" + "url": "https://github.com/opnsense/jquery-bootgrid.git" }, "keywords": [ "jquery-plugin", From 6ead150cffbbc59eae4349a818087cc6e9a26034 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 11:14:14 +0100 Subject: [PATCH 07/17] Fix filename of modernizr --- demo/js/{moderniz.2.8.1.js => modernizr-2.8.1.js} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename demo/js/{moderniz.2.8.1.js => modernizr-2.8.1.js} (100%) diff --git a/demo/js/moderniz.2.8.1.js b/demo/js/modernizr-2.8.1.js similarity index 100% rename from demo/js/moderniz.2.8.1.js rename to demo/js/modernizr-2.8.1.js From d2938a6d4cecd9b15b130a35325cc88129ec9045 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 12:31:20 +0100 Subject: [PATCH 08/17] Unlock jQuery version again --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index c6a3b49..5572922 100644 --- a/package.json +++ b/package.json @@ -36,7 +36,7 @@ ], "license": "MIT", "dependencies": { - "jquery": "^1.9", + "jquery": ">=1.9", "bootstrap": "3.1.1 - 3.3" }, "devDependencies": { From e28223c809c7821787e16ea9eee556d85cf9056e Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 12:58:15 +0100 Subject: [PATCH 09/17] Update qunit tests to 2.8.0 --- test/index.html | 4 +- test/qunit/qunit-1.11.0.css | 244 -- test/qunit/qunit-1.11.0.js | 2152 ------------ test/qunit/qunit-2.8.0.css | 436 +++ test/qunit/qunit-2.8.0.js | 6566 +++++++++++++++++++++++++++++++++++ test/tests-extensions.js | 14 +- test/tests-internal.js | 70 +- test/tests-rendering.js | 22 +- 8 files changed, 7071 insertions(+), 2437 deletions(-) delete mode 100644 test/qunit/qunit-1.11.0.css delete mode 100644 test/qunit/qunit-1.11.0.js create mode 100644 test/qunit/qunit-2.8.0.css create mode 100644 test/qunit/qunit-2.8.0.js diff --git a/test/index.html b/test/index.html index cdc0425..9808489 100644 --- a/test/index.html +++ b/test/index.html @@ -3,7 +3,7 @@ jQuery Bootgrid Test Suite - +

@@ -18,7 +18,7 @@

    - + diff --git a/test/qunit/qunit-1.11.0.css b/test/qunit/qunit-1.11.0.css deleted file mode 100644 index d7fc0c8..0000000 --- a/test/qunit/qunit-1.11.0.css +++ /dev/null @@ -1,244 +0,0 @@ -/** - * QUnit v1.11.0 - A JavaScript Unit Testing Framework - * - * http://qunitjs.com - * - * Copyright 2012 jQuery Foundation and other contributors - * Released under the MIT license. - * http://jquery.org/license - */ - -/** Font Family and Sizes */ - -#qunit-tests, #qunit-header, #qunit-banner, #qunit-testrunner-toolbar, #qunit-userAgent, #qunit-testresult { - font-family: "Helvetica Neue Light", "HelveticaNeue-Light", "Helvetica Neue", Calibri, Helvetica, Arial, sans-serif; -} - -#qunit-testrunner-toolbar, #qunit-userAgent, #qunit-testresult, #qunit-tests li { font-size: small; } -#qunit-tests { font-size: smaller; } - - -/** Resets */ - -#qunit-tests, #qunit-header, #qunit-banner, #qunit-userAgent, #qunit-testresult, #qunit-modulefilter { - margin: 0; - padding: 0; -} - - -/** Header */ - -#qunit-header { - padding: 0.5em 0 0.5em 1em; - - color: #8699a4; - background-color: #0d3349; - - font-size: 1.5em; - line-height: 1em; - font-weight: normal; - - border-radius: 5px 5px 0 0; - -moz-border-radius: 5px 5px 0 0; - -webkit-border-top-right-radius: 5px; - -webkit-border-top-left-radius: 5px; -} - -#qunit-header a { - text-decoration: none; - color: #c2ccd1; -} - -#qunit-header a:hover, -#qunit-header a:focus { - color: #fff; -} - -#qunit-testrunner-toolbar label { - display: inline-block; - padding: 0 .5em 0 .1em; -} - -#qunit-banner { - height: 5px; -} - -#qunit-testrunner-toolbar { - padding: 0.5em 0 0.5em 2em; - color: #5E740B; - background-color: #eee; - overflow: hidden; -} - -#qunit-userAgent { - padding: 0.5em 0 0.5em 2.5em; - background-color: #2b81af; - color: #fff; - text-shadow: rgba(0, 0, 0, 0.5) 2px 2px 1px; -} - -#qunit-modulefilter-container { - float: right; -} - -/** Tests: Pass/Fail */ - -#qunit-tests { - list-style-position: inside; -} - -#qunit-tests li { - padding: 0.4em 0.5em 0.4em 2.5em; - border-bottom: 1px solid #fff; - list-style-position: inside; -} - -#qunit-tests.hidepass li.pass, #qunit-tests.hidepass li.running { - display: none; -} - -#qunit-tests li strong { - cursor: pointer; -} - -#qunit-tests li a { - padding: 0.5em; - color: #c2ccd1; - text-decoration: none; -} -#qunit-tests li a:hover, -#qunit-tests li a:focus { - color: #000; -} - -#qunit-tests li .runtime { - float: right; - font-size: smaller; -} - -.qunit-assert-list { - margin-top: 0.5em; - padding: 0.5em; - - background-color: #fff; - - border-radius: 5px; - -moz-border-radius: 5px; - -webkit-border-radius: 5px; -} - -.qunit-collapsed { - display: none; -} - -#qunit-tests table { - border-collapse: collapse; - margin-top: .2em; -} - -#qunit-tests th { - text-align: right; - vertical-align: top; - padding: 0 .5em 0 0; -} - -#qunit-tests td { - vertical-align: top; -} - -#qunit-tests pre { - margin: 0; - white-space: pre-wrap; - word-wrap: break-word; -} - -#qunit-tests del { - background-color: #e0f2be; - color: #374e0c; - text-decoration: none; -} - -#qunit-tests ins { - background-color: #ffcaca; - color: #500; - text-decoration: none; -} - -/*** Test Counts */ - -#qunit-tests b.counts { color: black; } -#qunit-tests b.passed { color: #5E740B; } -#qunit-tests b.failed { color: #710909; } - -#qunit-tests li li { - padding: 5px; - background-color: #fff; - border-bottom: none; - list-style-position: inside; -} - -/*** Passing Styles */ - -#qunit-tests li li.pass { - color: #3c510c; - background-color: #fff; - border-left: 10px solid #C6E746; -} - -#qunit-tests .pass { color: #528CE0; background-color: #D2E0E6; } -#qunit-tests .pass .test-name { color: #366097; } - -#qunit-tests .pass .test-actual, -#qunit-tests .pass .test-expected { color: #999999; } - -#qunit-banner.qunit-pass { background-color: #C6E746; } - -/*** Failing Styles */ - -#qunit-tests li li.fail { - color: #710909; - background-color: #fff; - border-left: 10px solid #EE5757; - white-space: pre; -} - -#qunit-tests > li:last-child { - border-radius: 0 0 5px 5px; - -moz-border-radius: 0 0 5px 5px; - -webkit-border-bottom-right-radius: 5px; - -webkit-border-bottom-left-radius: 5px; -} - -#qunit-tests .fail { color: #000000; background-color: #EE5757; } -#qunit-tests .fail .test-name, -#qunit-tests .fail .module-name { color: #000000; } - -#qunit-tests .fail .test-actual { color: #EE5757; } -#qunit-tests .fail .test-expected { color: green; } - -#qunit-banner.qunit-fail { background-color: #EE5757; } - - -/** Result */ - -#qunit-testresult { - padding: 0.5em 0.5em 0.5em 2.5em; - - color: #2b81af; - background-color: #D2E0E6; - - border-bottom: 1px solid white; -} -#qunit-testresult .module-name { - font-weight: bold; -} - -/** Fixture */ - -#qunit-fixture { - position: absolute; - top: -10000px; - left: -10000px; - width: 1000px; - height: 1000px; -} diff --git a/test/qunit/qunit-1.11.0.js b/test/qunit/qunit-1.11.0.js deleted file mode 100644 index 302545f..0000000 --- a/test/qunit/qunit-1.11.0.js +++ /dev/null @@ -1,2152 +0,0 @@ -/** - * QUnit v1.11.0 - A JavaScript Unit Testing Framework - * - * http://qunitjs.com - * - * Copyright 2012 jQuery Foundation and other contributors - * Released under the MIT license. - * http://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, - 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 ( this.module !== config.previousModule ) { - if ( 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 - }); - } else if ( config.autorun ) { - 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 - }); - - // allow utility functions to access the current test environment - // TODO why?? - QUnit.current_testEnvironment = this.testEnvironment; - - if ( !config.pollution ) { - saveGlobal(); - } - if ( config.notrycatch ) { - this.testEnvironment.setup.call( this.testEnvironment ); - return; - } - try { - this.testEnvironment.setup.call( this.testEnvironment ); - } 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:
    " + 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 ); - return; - } else { - try { - this.testEnvironment.teardown.call( this.testEnvironment ); - } 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 + " (" + bad + ", " + good + ", " + this.assertions.length + ")"; - - 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 = "" + escapeText( testName ) + ""; - - if ( arguments.length === 2 ) { - callback = expected; - expected = null; - } - - if ( config.currentModule ) { - nameHtml = "" + escapeText( config.currentModule ) + ": " + 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 gurantee 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 ) { - window.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 = window.setTimeout(function() { - QUnit.ok( false, "Test timed out" ); - config.semaphore = 1; - QUnit.start(); - }, config.testTimeout ); - } - } -}; - -// `assert` initialized at top of scope -// Asssert 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; - - var source, - details = { - module: config.current.module, - name: config.current.testName, - result: result, - message: msg - }; - - msg = escapeText( msg || (result ? "okay" : "failed" ) ); - msg = "" + msg + ""; - - if ( !result ) { - source = sourceFromStacktrace( 2 ); - if ( source ) { - details.source = source; - msg += "
    Source:
    " + escapeText( source ) + "
    "; - } - } - 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.' ); - } - } -}; - -/** - * @deprecate 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 ); - - // 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 = - "

    " + escapeText( document.title ) + "

    " + - "

    " + - "
    " + - "

    " + - "
      "; - } - - 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...
       "; - } - }, - - // Resets the test setup. Useful for tests that modify the DOM. - 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 = "" + message + ""; - output = message; - - if ( !result ) { - expected = escapeText( QUnit.jsDump.parse(expected) ); - actual = escapeText( QUnit.jsDump.parse(actual) ); - output += ""; - - if ( actual !== expected ) { - output += ""; - output += ""; - } - - source = sourceFromStacktrace(); - - if ( source ) { - details.source = source; - output += ""; - } - - output += "
      Expected:
      " + expected + "
      Result:
      " + actual + "
      Diff:
      " + QUnit.diff( expected, actual ) + "
      Source:
      " + escapeText( source ) + "
      "; - } - - 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 = "" + message + ""; - output = message; - - output += ""; - - if ( actual ) { - output += ""; - } - - if ( source ) { - details.source = source; - output += ""; - } - - output += "
      Result:
      " + escapeText( actual ) + "
      Source:
      " + escapeText( source ) + "
      "; - - 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 ) ) { - continue; - } - 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 - // 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, - 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 += ""; - } - - moduleFilterHtml += ""; - - // `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 = "" + banner.innerHTML + " "; - } - - // `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 assertons 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 } ); - }); - 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 surpress 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 surpressed. - 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 - }); - } - - 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.
      ", - "", - passed, - " assertions of ", - config.stats.all, - " passed, ", - config.stats.bad, - " 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 { - window.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 ) { - // in Opera sometimes DOM element ids show up here, ignore them - if ( !hasOwn.call( window, key ) || /^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 ( b[ prop ] === undefined ) { - delete a[ prop ]; - - // Avoid "Member not found" error in IE8 caused by setting window.constructor - } else if ( prop !== "constructor" || a !== window ) { - 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 neccecarily - elem.className = window.jQuery ? jQuery.trim( set ) : ( set.trim ? set.trim() : set ); -} - -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é -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 = [], - - getProto = Object.getPrototypeOf || function ( obj ) { - 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 annotaion 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 modifers - 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; - - // 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 ); - for ( i = 0; i < len; i++ ) { - loop = false; - for ( j = 0; j < parents.length; j++ ) { - if ( parents[j] === a[i] ) { - loop = true;// dont rewalk array - } - } - if ( !loop && !innerEquiv(a[i], b[i]) ) { - parents.pop(); - return false; - } - } - parents.pop(); - return true; - }, - - "object": function( b, a ) { - var i, j, loop, - // 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 ); - - for ( i in a ) { // be strict: don't ensures hasOwnProperty - // and go deep - loop = false; - for ( j = 0; j < parents.length; j++ ) { - if ( parents[j] === a[i] ) { - // don't go down the same path twice - loop = true; - } - } - aProperties.push(i); // collect a's properties - - if (!loop && !innerEquiv( a[i], b[i] ) ) { - eq = false; - break; - } - } - - callers.pop(); // unstack, we are done - parents.pop(); - - 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] ) && arguments.callee.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 ? "
      " : "\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 ) { - 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 brown fox jumped jumps 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 ) ) { - continue; - } - 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 += "" + out.o[i] + oSpace[i] + ""; - } - } - else { - if ( out.n[0].text == null ) { - for ( n = 0; n < out.o.length && out.o[n].text == null; n++ ) { - str += "" + out.o[n] + oSpace[n] + ""; - } - } - - for ( i = 0; i < out.n.length; i++ ) { - if (out.n[i].text == null) { - str += "" + out.n[i] + nSpace[i] + ""; - } - 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 += "" + out.o[n] + oSpace[n] + ""; - } - str += " " + out.n[i].text + nSpace[i] + pre; - } - } - } - - return str; - }; -}()); - -// for CommonJS enviroments, export everything -if ( typeof exports !== "undefined" ) { - extend( exports, QUnit ); -} - -// get at whatever the global object is, like window in browsers -}( (function() {return this;}.call()) )); diff --git a/test/qunit/qunit-2.8.0.css b/test/qunit/qunit-2.8.0.css new file mode 100644 index 0000000..9bf0381 --- /dev/null +++ b/test/qunit/qunit-2.8.0.css @@ -0,0 +1,436 @@ +/*! + * QUnit 2.8.0 + * https://qunitjs.com/ + * + * Copyright jQuery Foundation and other contributors + * Released under the MIT license + * https://jquery.org/license + * + * Date: 2018-11-02T16:17Z + */ + +/** Font Family and Sizes */ + +#qunit-tests, #qunit-header, #qunit-banner, #qunit-testrunner-toolbar, #qunit-filteredTest, #qunit-userAgent, #qunit-testresult { + font-family: "Helvetica Neue Light", "HelveticaNeue-Light", "Helvetica Neue", Calibri, Helvetica, Arial, sans-serif; +} + +#qunit-testrunner-toolbar, #qunit-filteredTest, #qunit-userAgent, #qunit-testresult, #qunit-tests li { font-size: small; } +#qunit-tests { font-size: smaller; } + + +/** Resets */ + +#qunit-tests, #qunit-header, #qunit-banner, #qunit-filteredTest, #qunit-userAgent, #qunit-testresult, #qunit-modulefilter { + margin: 0; + padding: 0; +} + + +/** Header (excluding toolbar) */ + +#qunit-header { + padding: 0.5em 0 0.5em 1em; + + color: #8699A4; + background-color: #0D3349; + + font-size: 1.5em; + line-height: 1em; + font-weight: 400; + + border-radius: 5px 5px 0 0; +} + +#qunit-header a { + text-decoration: none; + color: #C2CCD1; +} + +#qunit-header a:hover, +#qunit-header a:focus { + color: #FFF; +} + +#qunit-banner { + height: 5px; +} + +#qunit-filteredTest { + padding: 0.5em 1em 0.5em 1em; + color: #366097; + background-color: #F4FF77; +} + +#qunit-userAgent { + padding: 0.5em 1em 0.5em 1em; + color: #FFF; + background-color: #2B81AF; + text-shadow: rgba(0, 0, 0, 0.5) 2px 2px 1px; +} + + +/** Toolbar */ + +#qunit-testrunner-toolbar { + padding: 0.5em 1em 0.5em 1em; + color: #5E740B; + background-color: #EEE; +} + +#qunit-testrunner-toolbar .clearfix { + height: 0; + clear: both; +} + +#qunit-testrunner-toolbar label { + display: inline-block; +} + +#qunit-testrunner-toolbar input[type=checkbox], +#qunit-testrunner-toolbar input[type=radio] { + margin: 3px; + vertical-align: -2px; +} + +#qunit-testrunner-toolbar input[type=text] { + box-sizing: border-box; + height: 1.6em; +} + +.qunit-url-config, +.qunit-filter, +#qunit-modulefilter { + display: inline-block; + line-height: 2.1em; +} + +.qunit-filter, +#qunit-modulefilter { + float: right; + position: relative; + margin-left: 1em; +} + +.qunit-url-config label { + margin-right: 0.5em; +} + +#qunit-modulefilter-search { + box-sizing: border-box; + width: 400px; +} + +#qunit-modulefilter-search-container:after { + position: absolute; + right: 0.3em; + content: "\25bc"; + color: black; +} + +#qunit-modulefilter-dropdown { + /* align with #qunit-modulefilter-search */ + box-sizing: border-box; + width: 400px; + position: absolute; + right: 0; + top: 50%; + margin-top: 0.8em; + + border: 1px solid #D3D3D3; + border-top: none; + border-radius: 0 0 .25em .25em; + color: #000; + background-color: #F5F5F5; + z-index: 99; +} + +#qunit-modulefilter-dropdown a { + color: inherit; + text-decoration: none; +} + +#qunit-modulefilter-dropdown .clickable.checked { + font-weight: bold; + color: #000; + background-color: #D2E0E6; +} + +#qunit-modulefilter-dropdown .clickable:hover { + color: #FFF; + background-color: #0D3349; +} + +#qunit-modulefilter-actions { + display: block; + overflow: auto; + + /* align with #qunit-modulefilter-dropdown-list */ + font: smaller/1.5em sans-serif; +} + +#qunit-modulefilter-dropdown #qunit-modulefilter-actions > * { + box-sizing: border-box; + max-height: 2.8em; + display: block; + padding: 0.4em; +} + +#qunit-modulefilter-dropdown #qunit-modulefilter-actions > button { + float: right; + font: inherit; +} + +#qunit-modulefilter-dropdown #qunit-modulefilter-actions > :last-child { + /* insert padding to align with checkbox margins */ + padding-left: 3px; +} + +#qunit-modulefilter-dropdown-list { + max-height: 200px; + overflow-y: auto; + margin: 0; + border-top: 2px groove threedhighlight; + padding: 0.4em 0 0; + font: smaller/1.5em sans-serif; +} + +#qunit-modulefilter-dropdown-list li { + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; +} + +#qunit-modulefilter-dropdown-list .clickable { + display: block; + padding-left: 0.15em; +} + + +/** Tests: Pass/Fail */ + +#qunit-tests { + list-style-position: inside; +} + +#qunit-tests li { + padding: 0.4em 1em 0.4em 1em; + border-bottom: 1px solid #FFF; + list-style-position: inside; +} + +#qunit-tests > li { + display: none; +} + +#qunit-tests li.running, +#qunit-tests li.pass, +#qunit-tests li.fail, +#qunit-tests li.skipped, +#qunit-tests li.aborted { + display: list-item; +} + +#qunit-tests.hidepass { + position: relative; +} + +#qunit-tests.hidepass li.running, +#qunit-tests.hidepass li.pass:not(.todo) { + visibility: hidden; + position: absolute; + width: 0; + height: 0; + padding: 0; + border: 0; + margin: 0; +} + +#qunit-tests li strong { + cursor: pointer; +} + +#qunit-tests li.skipped strong { + cursor: default; +} + +#qunit-tests li a { + padding: 0.5em; + color: #C2CCD1; + text-decoration: none; +} + +#qunit-tests li p a { + padding: 0.25em; + color: #6B6464; +} +#qunit-tests li a:hover, +#qunit-tests li a:focus { + color: #000; +} + +#qunit-tests li .runtime { + float: right; + font-size: smaller; +} + +.qunit-assert-list { + margin-top: 0.5em; + padding: 0.5em; + + background-color: #FFF; + + border-radius: 5px; +} + +.qunit-source { + margin: 0.6em 0 0.3em; +} + +.qunit-collapsed { + display: none; +} + +#qunit-tests table { + border-collapse: collapse; + margin-top: 0.2em; +} + +#qunit-tests th { + text-align: right; + vertical-align: top; + padding: 0 0.5em 0 0; +} + +#qunit-tests td { + vertical-align: top; +} + +#qunit-tests pre { + margin: 0; + white-space: pre-wrap; + word-wrap: break-word; +} + +#qunit-tests del { + color: #374E0C; + background-color: #E0F2BE; + text-decoration: none; +} + +#qunit-tests ins { + color: #500; + background-color: #FFCACA; + text-decoration: none; +} + +/*** Test Counts */ + +#qunit-tests b.counts { color: #000; } +#qunit-tests b.passed { color: #5E740B; } +#qunit-tests b.failed { color: #710909; } + +#qunit-tests li li { + padding: 5px; + background-color: #FFF; + border-bottom: none; + list-style-position: inside; +} + +/*** Passing Styles */ + +#qunit-tests li li.pass { + color: #3C510C; + background-color: #FFF; + border-left: 10px solid #C6E746; +} + +#qunit-tests .pass { color: #528CE0; background-color: #D2E0E6; } +#qunit-tests .pass .test-name { color: #366097; } + +#qunit-tests .pass .test-actual, +#qunit-tests .pass .test-expected { color: #999; } + +#qunit-banner.qunit-pass { background-color: #C6E746; } + +/*** Failing Styles */ + +#qunit-tests li li.fail { + color: #710909; + background-color: #FFF; + border-left: 10px solid #EE5757; + white-space: pre; +} + +#qunit-tests > li:last-child { + border-radius: 0 0 5px 5px; +} + +#qunit-tests .fail { color: #000; background-color: #EE5757; } +#qunit-tests .fail .test-name, +#qunit-tests .fail .module-name { color: #000; } + +#qunit-tests .fail .test-actual { color: #EE5757; } +#qunit-tests .fail .test-expected { color: #008000; } + +#qunit-banner.qunit-fail { background-color: #EE5757; } + + +/*** Aborted tests */ +#qunit-tests .aborted { color: #000; background-color: orange; } +/*** Skipped tests */ + +#qunit-tests .skipped { + background-color: #EBECE9; +} + +#qunit-tests .qunit-todo-label, +#qunit-tests .qunit-skipped-label { + background-color: #F4FF77; + display: inline-block; + font-style: normal; + color: #366097; + line-height: 1.8em; + padding: 0 0.5em; + margin: -0.4em 0.4em -0.4em 0; +} + +#qunit-tests .qunit-todo-label { + background-color: #EEE; +} + +/** Result */ + +#qunit-testresult { + color: #2B81AF; + background-color: #D2E0E6; + + border-bottom: 1px solid #FFF; +} +#qunit-testresult .clearfix { + height: 0; + clear: both; +} +#qunit-testresult .module-name { + font-weight: 700; +} +#qunit-testresult-display { + padding: 0.5em 1em 0.5em 1em; + width: 85%; + float:left; +} +#qunit-testresult-controls { + padding: 0.5em 1em 0.5em 1em; + width: 10%; + float:left; +} + +/** Fixture */ + +#qunit-fixture { + position: absolute; + top: -10000px; + left: -10000px; + width: 1000px; + height: 1000px; +} diff --git a/test/qunit/qunit-2.8.0.js b/test/qunit/qunit-2.8.0.js new file mode 100644 index 0000000..e30bb19 --- /dev/null +++ b/test/qunit/qunit-2.8.0.js @@ -0,0 +1,6566 @@ +/*! + * QUnit 2.8.0 + * https://qunitjs.com/ + * + * Copyright jQuery Foundation and other contributors + * Released under the MIT license + * https://jquery.org/license + * + * Date: 2018-11-02T16:17Z + */ +(function (global$1) { + 'use strict'; + + global$1 = global$1 && global$1.hasOwnProperty('default') ? global$1['default'] : global$1; + + var window$1 = global$1.window; + var self$1 = global$1.self; + var console = global$1.console; + var setTimeout$1 = global$1.setTimeout; + var clearTimeout = global$1.clearTimeout; + + var document$1 = window$1 && window$1.document; + var navigator = window$1 && window$1.navigator; + + var localSessionStorage = function () { + var x = "qunit-test-string"; + try { + global$1.sessionStorage.setItem(x, x); + global$1.sessionStorage.removeItem(x); + return global$1.sessionStorage; + } catch (e) { + return undefined; + } + }(); + + /** + * Returns a function that proxies to the given method name on the globals + * console object. The proxy will also detect if the console doesn't exist and + * will appropriately no-op. This allows support for IE9, which doesn't have a + * console if the developer tools are not open. + */ + function consoleProxy(method) { + return function () { + if (console) { + console[method].apply(console, arguments); + } + }; + } + + var Logger = { + warn: consoleProxy("warn") + }; + + var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { + return typeof obj; + } : function (obj) { + return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; + }; + + + + + + + + + + + + var classCallCheck = function (instance, Constructor) { + if (!(instance instanceof Constructor)) { + throw new TypeError("Cannot call a class as a function"); + } + }; + + var createClass = function () { + function defineProperties(target, props) { + for (var i = 0; i < props.length; i++) { + var descriptor = props[i]; + descriptor.enumerable = descriptor.enumerable || false; + descriptor.configurable = true; + if ("value" in descriptor) descriptor.writable = true; + Object.defineProperty(target, descriptor.key, descriptor); + } + } + + return function (Constructor, protoProps, staticProps) { + if (protoProps) defineProperties(Constructor.prototype, protoProps); + if (staticProps) defineProperties(Constructor, staticProps); + return Constructor; + }; + }(); + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + var toConsumableArray = function (arr) { + if (Array.isArray(arr)) { + for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) arr2[i] = arr[i]; + + return arr2; + } else { + return Array.from(arr); + } + }; + + var toString = Object.prototype.toString; + var hasOwn = Object.prototype.hasOwnProperty; + var now = Date.now || function () { + return new Date().getTime(); + }; + + var hasPerformanceApi = detectPerformanceApi(); + var performance = hasPerformanceApi ? window$1.performance : undefined; + var performanceNow = hasPerformanceApi ? performance.now.bind(performance) : now; + + function detectPerformanceApi() { + return window$1 && typeof window$1.performance !== "undefined" && typeof window$1.performance.mark === "function" && typeof window$1.performance.measure === "function"; + } + + function measure(comment, startMark, endMark) { + + // `performance.measure` may fail if the mark could not be found. + // reasons a specific mark could not be found include: outside code invoking `performance.clearMarks()` + try { + performance.measure(comment, startMark, endMark); + } catch (ex) { + Logger.warn("performance.measure could not be executed because of ", ex.message); + } + } + + var defined = { + document: window$1 && window$1.document !== undefined, + setTimeout: setTimeout$1 !== undefined + }; + + // 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; + } + + /** + * Determines whether an element exists in a given array or not. + * + * @method inArray + * @param {Any} elem + * @param {Array} array + * @return {Boolean} + */ + function inArray(elem, array) { + return array.indexOf(elem) !== -1; + } + + /** + * 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). + */ + function objectValues(obj) { + var key, + val, + vals = 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 extend(a, b, undefOnly) { + for (var prop in b) { + if (hasOwn.call(b, prop)) { + if (b[prop] === undefined) { + delete a[prop]; + } else if (!(undefOnly && typeof a[prop] !== "undefined")) { + a[prop] = b[prop]; + } + } + } + + return a; + } + + function objectType(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 "Set": + case "Map": + case "Date": + case "RegExp": + case "Function": + case "Symbol": + return type.toLowerCase(); + default: + return typeof obj === "undefined" ? "undefined" : _typeof(obj); + } + } + + // Safe object type checking + function is(type, obj) { + return objectType(obj) === type; + } + + // Based on Java's String.hashCode, a simple but not + // rigorously collision resistant hashing function + function generateHash(module, testName) { + var str = module + "\x1C" + testName; + var hash = 0; + + for (var i = 0; i < str.length; i++) { + hash = (hash << 5) - hash + str.charCodeAt(i); + hash |= 0; + } + + // Convert the possibly negative integer hash code into an 8 character hex string, which isn't + // strictly necessary but increases user understanding that the id is a SHA-like hash + var hex = (0x100000000 + hash).toString(16); + if (hex.length < 8) { + hex = "0000000" + hex; + } + + return hex.slice(-8); + } + + // Test for equality any JavaScript type. + // Authors: Philippe Rathé , David Chan + var equiv = (function () { + + // Value pairs queued for comparison. Used for breadth-first processing order, recursion + // detection and avoiding repeated comparison (see below for details). + // Elements are { a: val, b: val }. + var pairs = []; + + var getProto = Object.getPrototypeOf || function (obj) { + return obj.__proto__; + }; + + function useStrictEquality(a, b) { + + // This only gets called if a and b are not strict equal, and is used to compare on + // the primitive values inside object wrappers. For example: + // `var i = 1;` + // `var j = new Number(1);` + // Neither a nor b can be null, as a !== b and they have the same type. + if ((typeof a === "undefined" ? "undefined" : _typeof(a)) === "object") { + a = a.valueOf(); + } + if ((typeof b === "undefined" ? "undefined" : _typeof(b)) === "object") { + b = b.valueOf(); + } + + return a === b; + } + + function compareConstructors(a, b) { + var protoA = getProto(a); + var protoB = getProto(b); + + // Comparing constructors is more strict than using `instanceof` + if (a.constructor === b.constructor) { + return true; + } + + // Ref #851 + // If the obj prototype descends from a null constructor, treat it + // as a null prototype. + if (protoA && protoA.constructor === null) { + protoA = null; + } + if (protoB && protoB.constructor === null) { + protoB = null; + } + + // Allow objects with no prototype to be equivalent to + // objects with Object as their constructor. + if (protoA === null && protoB === Object.prototype || protoB === null && protoA === Object.prototype) { + return true; + } + + return false; + } + + function getRegExpFlags(regexp) { + return "flags" in regexp ? regexp.flags : regexp.toString().match(/[gimuy]*$/)[0]; + } + + function isContainer(val) { + return ["object", "array", "map", "set"].indexOf(objectType(val)) !== -1; + } + + function breadthFirstCompareChild(a, b) { + + // If a is a container not reference-equal to b, postpone the comparison to the + // end of the pairs queue -- unless (a, b) has been seen before, in which case skip + // over the pair. + if (a === b) { + return true; + } + if (!isContainer(a)) { + return typeEquiv(a, b); + } + if (pairs.every(function (pair) { + return pair.a !== a || pair.b !== b; + })) { + + // Not yet started comparing this pair + pairs.push({ a: a, b: b }); + } + return true; + } + + var callbacks = { + "string": useStrictEquality, + "boolean": useStrictEquality, + "number": useStrictEquality, + "null": useStrictEquality, + "undefined": useStrictEquality, + "symbol": useStrictEquality, + "date": useStrictEquality, + + "nan": function nan() { + return true; + }, + + "regexp": function regexp(a, b) { + return a.source === b.source && + + // Include flags in the comparison + getRegExpFlags(a) === getRegExpFlags(b); + }, + + // abort (identical references / instance methods were skipped earlier) + "function": function _function() { + return false; + }, + + "array": function array(a, b) { + var i, len; + + len = a.length; + if (len !== b.length) { + + // Safe and faster + return false; + } + + for (i = 0; i < len; i++) { + + // Compare non-containers; queue non-reference-equal containers + if (!breadthFirstCompareChild(a[i], b[i])) { + return false; + } + } + return true; + }, + + // Define sets a and b to be equivalent if for each element aVal in a, there + // is some element bVal in b such that aVal and bVal are equivalent. Element + // repetitions are not counted, so these are equivalent: + // a = new Set( [ {}, [], [] ] ); + // b = new Set( [ {}, {}, [] ] ); + "set": function set$$1(a, b) { + var innerEq, + outerEq = true; + + if (a.size !== b.size) { + + // This optimization has certain quirks because of the lack of + // repetition counting. For instance, adding the same + // (reference-identical) element to two equivalent sets can + // make them non-equivalent. + return false; + } + + a.forEach(function (aVal) { + + // Short-circuit if the result is already known. (Using for...of + // with a break clause would be cleaner here, but it would cause + // a syntax error on older Javascript implementations even if + // Set is unused) + if (!outerEq) { + return; + } + + innerEq = false; + + b.forEach(function (bVal) { + var parentPairs; + + // Likewise, short-circuit if the result is already known + if (innerEq) { + return; + } + + // Swap out the global pairs list, as the nested call to + // innerEquiv will clobber its contents + parentPairs = pairs; + if (innerEquiv(bVal, aVal)) { + innerEq = true; + } + + // Replace the global pairs list + pairs = parentPairs; + }); + + if (!innerEq) { + outerEq = false; + } + }); + + return outerEq; + }, + + // Define maps a and b to be equivalent if for each key-value pair (aKey, aVal) + // in a, there is some key-value pair (bKey, bVal) in b such that + // [ aKey, aVal ] and [ bKey, bVal ] are equivalent. Key repetitions are not + // counted, so these are equivalent: + // a = new Map( [ [ {}, 1 ], [ {}, 1 ], [ [], 1 ] ] ); + // b = new Map( [ [ {}, 1 ], [ [], 1 ], [ [], 1 ] ] ); + "map": function map(a, b) { + var innerEq, + outerEq = true; + + if (a.size !== b.size) { + + // This optimization has certain quirks because of the lack of + // repetition counting. For instance, adding the same + // (reference-identical) key-value pair to two equivalent maps + // can make them non-equivalent. + return false; + } + + a.forEach(function (aVal, aKey) { + + // Short-circuit if the result is already known. (Using for...of + // with a break clause would be cleaner here, but it would cause + // a syntax error on older Javascript implementations even if + // Map is unused) + if (!outerEq) { + return; + } + + innerEq = false; + + b.forEach(function (bVal, bKey) { + var parentPairs; + + // Likewise, short-circuit if the result is already known + if (innerEq) { + return; + } + + // Swap out the global pairs list, as the nested call to + // innerEquiv will clobber its contents + parentPairs = pairs; + if (innerEquiv([bVal, bKey], [aVal, aKey])) { + innerEq = true; + } + + // Replace the global pairs list + pairs = parentPairs; + }); + + if (!innerEq) { + outerEq = false; + } + }); + + return outerEq; + }, + + "object": function object(a, b) { + var i, + aProperties = [], + bProperties = []; + + if (compareConstructors(a, b) === false) { + return false; + } + + // Be strict: don't ensure hasOwnProperty and go deep + for (i in a) { + + // Collect a's properties + aProperties.push(i); + + // Skip OOP methods that look the same + if (a.constructor !== Object && typeof a.constructor !== "undefined" && typeof a[i] === "function" && typeof b[i] === "function" && a[i].toString() === b[i].toString()) { + continue; + } + + // Compare non-containers; queue non-reference-equal containers + if (!breadthFirstCompareChild(a[i], b[i])) { + return false; + } + } + + for (i in b) { + + // Collect b's properties + bProperties.push(i); + } + + // Ensures identical properties name + return typeEquiv(aProperties.sort(), bProperties.sort()); + } + }; + + function typeEquiv(a, b) { + var type = objectType(a); + + // Callbacks for containers will append to the pairs queue to achieve breadth-first + // search order. The pairs queue is also used to avoid reprocessing any pair of + // containers that are reference-equal to a previously visited pair (a special case + // this being recursion detection). + // + // Because of this approach, once typeEquiv returns a false value, it should not be + // called again without clearing the pair queue else it may wrongly report a visited + // pair as being equivalent. + return objectType(b) === type && callbacks[type](a, b); + } + + function innerEquiv(a, b) { + var i, pair; + + // We're done when there's nothing more to compare + if (arguments.length < 2) { + return true; + } + + // Clear the global pair queue and add the top-level values being compared + pairs = [{ a: a, b: b }]; + + for (i = 0; i < pairs.length; i++) { + pair = pairs[i]; + + // Perform type-specific comparison on any pairs that are not strictly + // equal. For container types, that comparison will postpone comparison + // of any sub-container pair to the end of the pair queue. This gives + // breadth-first search order. It also avoids the reprocessing of + // reference-equal siblings, cousins etc, which can have a significant speed + // impact when comparing a container of small objects each of which has a + // reference to the same (singleton) large object. + if (pair.a !== pair.b && !typeEquiv(pair.a, pair.b)) { + return false; + } + } + + // ...across all consecutive argument pairs + return arguments.length === 2 || innerEquiv.apply(this, [].slice.call(arguments, 1)); + } + + return function () { + var result = innerEquiv.apply(undefined, arguments); + + // Release any retained objects + pairs.length = 0; + return result; + }; + })(); + + /** + * Config object: Maintain internal state + * Later exposed as QUnit.config + * `config` initialized at top of scope + */ + var config = { + + // The queue of tests to run + queue: [], + + // Block until document ready + blocking: true, + + // 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, + + // HTML Reporter: collapse every test except the first failing test + // If false, all failing tests will be expanded + collapse: true, + + // By default, scroll to top of the page when suite is done + scrolltop: true, + + // Depth up-to which object will be dumped + maxDepth: 5, + + // When enabled, all tests must call expect() + requireExpects: false, + + // Placeholder for user-configurable form-exposed URL parameters + urlConfig: [], + + // Set of all modules. + modules: [], + + // The first unnamed module + currentModule: { + name: "", + tests: [], + childModules: [], + testsRun: 0, + unskippedTestsRun: 0, + hooks: { + before: [], + beforeEach: [], + afterEach: [], + after: [] + } + }, + + callbacks: {}, + + // The storage module to use for reordering tests + storage: localSessionStorage + }; + + // take a predefined QUnit.config and extend the defaults + var globalConfig = window$1 && window$1.QUnit && window$1.QUnit.config; + + // only extend the global config if there is no QUnit overload + if (window$1 && window$1.QUnit && !window$1.QUnit.version) { + extend(config, globalConfig); + } + + // Push a loose unnamed module to the modules collection + config.modules.push(config.currentModule); + + // Based on jsDump by Ariel Flesler + // http://flesler.blogspot.com/2008/05/jsdump-pretty-dump-of-any-javascript.html + var dump = (function () { + function quote(str) { + return "\"" + str.toString().replace(/\\/g, "\\\\").replace(/"/g, "\\\"") + "\""; + } + function literal(o) { + return o + ""; + } + function join(pre, arr, post) { + var s = dump.separator(), + base = dump.indent(), + inner = dump.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); + + if (dump.maxDepth && dump.depth > dump.maxDepth) { + return "[object Array]"; + } + + this.up(); + while (i--) { + ret[i] = this.parse(arr[i], undefined, stack); + } + this.down(); + return join("[", ret, "]"); + } + + function isArray(obj) { + return ( + + //Native Arrays + toString.call(obj) === "[object Array]" || + + // NodeList objects + typeof obj.length === "number" && obj.item !== undefined && (obj.length ? obj.item(0) === obj[0] : obj.item(0) === null && obj[0] === undefined) + ); + } + + var reName = /^function (\w+)/, + dump = { + + // The objType is used mostly internally, you can fix a (custom) type in advance + parse: function parse(obj, objType, stack) { + stack = stack || []; + var res, + parser, + parserType, + objIndex = stack.indexOf(obj); + + if (objIndex !== -1) { + return "recursion(" + (objIndex - stack.length) + ")"; + } + + objType = objType || this.typeOf(obj); + parser = this.parsers[objType]; + parserType = typeof parser === "undefined" ? "undefined" : _typeof(parser); + + if (parserType === "function") { + stack.push(obj); + res = parser.call(this, obj, stack); + stack.pop(); + return res; + } + return parserType === "string" ? parser : this.parsers.error; + }, + typeOf: function typeOf(obj) { + var type; + + if (obj === null) { + type = "null"; + } else if (typeof obj === "undefined") { + type = "undefined"; + } else if (is("regexp", obj)) { + type = "regexp"; + } else if (is("date", obj)) { + type = "date"; + } else if (is("function", obj)) { + type = "function"; + } else if (obj.setInterval !== undefined && obj.document !== undefined && obj.nodeType === undefined) { + type = "window"; + } else if (obj.nodeType === 9) { + type = "document"; + } else if (obj.nodeType) { + type = "node"; + } else if (isArray(obj)) { + type = "array"; + } else if (obj.constructor === Error.prototype.constructor) { + type = "error"; + } else { + type = typeof obj === "undefined" ? "undefined" : _typeof(obj); + } + return type; + }, + + separator: function separator() { + if (this.multiline) { + return this.HTML ? "
      " : "\n"; + } else { + return this.HTML ? " " : " "; + } + }, + + // Extra can be a number, shortcut for increasing-calling-decreasing + indent: function indent(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 up(a) { + this.depth += a || 1; + }, + down: function down(a) { + this.depth -= a || 1; + }, + setParser: function setParser(name, parser) { + this.parsers[name] = parser; + }, + + // The next 3 are exposed so you can use them + quote: quote, + literal: literal, + join: join, + depth: 1, + maxDepth: config.maxDepth, + + // This is the list of parsers, to modify them, use dump.setParser + parsers: { + window: "[Window]", + document: "[Document]", + error: function error(_error) { + return "Error(\"" + _error.message + "\")"; + }, + unknown: "[Unknown]", + "null": "null", + "undefined": "undefined", + "function": 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, dump.parse(fn, "functionArgs"), "){"].join(""); + return join(ret, dump.parse(fn, "functionCode"), "}"); + }, + array: array, + nodelist: array, + "arguments": array, + object: function object(map, stack) { + var keys, + key, + val, + i, + nonEnumerableProperties, + ret = []; + + if (dump.maxDepth && dump.depth > dump.maxDepth) { + return "[object Object]"; + } + + dump.up(); + keys = []; + for (key in map) { + keys.push(key); + } + + // Some properties are not always enumerable on Error objects. + nonEnumerableProperties = ["message", "name"]; + for (i in nonEnumerableProperties) { + key = nonEnumerableProperties[i]; + if (key in map && !inArray(key, keys)) { + keys.push(key); + } + } + keys.sort(); + for (i = 0; i < keys.length; i++) { + key = keys[i]; + val = map[key]; + ret.push(dump.parse(key, "key") + ": " + dump.parse(val, undefined, stack)); + } + dump.down(); + return join("{", ret, "}"); + }, + node: function node(_node) { + var len, + i, + val, + open = dump.HTML ? "<" : "<", + close = dump.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 + "=" + dump.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 functionArgs(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 a html attribute value + attribute: quote, + string: quote, + date: quote, + regexp: literal, + number: literal, + "boolean": literal, + symbol: function symbol(sym) { + return sym.toString(); + } + }, + + // 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 dump; + })(); + + var SuiteReport = function () { + function SuiteReport(name, parentSuite) { + classCallCheck(this, SuiteReport); + + this.name = name; + this.fullName = parentSuite ? parentSuite.fullName.concat(name) : []; + + this.tests = []; + this.childSuites = []; + + if (parentSuite) { + parentSuite.pushChildSuite(this); + } + } + + createClass(SuiteReport, [{ + key: "start", + value: function start(recordTime) { + if (recordTime) { + this._startTime = performanceNow(); + + if (performance) { + var suiteLevel = this.fullName.length; + performance.mark("qunit_suite_" + suiteLevel + "_start"); + } + } + + return { + name: this.name, + fullName: this.fullName.slice(), + tests: this.tests.map(function (test) { + return test.start(); + }), + childSuites: this.childSuites.map(function (suite) { + return suite.start(); + }), + testCounts: { + total: this.getTestCounts().total + } + }; + } + }, { + key: "end", + value: function end(recordTime) { + if (recordTime) { + this._endTime = performanceNow(); + + if (performance) { + var suiteLevel = this.fullName.length; + performance.mark("qunit_suite_" + suiteLevel + "_end"); + + var suiteName = this.fullName.join(" – "); + + measure(suiteLevel === 0 ? "QUnit Test Run" : "QUnit Test Suite: " + suiteName, "qunit_suite_" + suiteLevel + "_start", "qunit_suite_" + suiteLevel + "_end"); + } + } + + return { + name: this.name, + fullName: this.fullName.slice(), + tests: this.tests.map(function (test) { + return test.end(); + }), + childSuites: this.childSuites.map(function (suite) { + return suite.end(); + }), + testCounts: this.getTestCounts(), + runtime: this.getRuntime(), + status: this.getStatus() + }; + } + }, { + key: "pushChildSuite", + value: function pushChildSuite(suite) { + this.childSuites.push(suite); + } + }, { + key: "pushTest", + value: function pushTest(test) { + this.tests.push(test); + } + }, { + key: "getRuntime", + value: function getRuntime() { + return this._endTime - this._startTime; + } + }, { + key: "getTestCounts", + value: function getTestCounts() { + var counts = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : { passed: 0, failed: 0, skipped: 0, todo: 0, total: 0 }; + + counts = this.tests.reduce(function (counts, test) { + if (test.valid) { + counts[test.getStatus()]++; + counts.total++; + } + + return counts; + }, counts); + + return this.childSuites.reduce(function (counts, suite) { + return suite.getTestCounts(counts); + }, counts); + } + }, { + key: "getStatus", + value: function getStatus() { + var _getTestCounts = this.getTestCounts(), + total = _getTestCounts.total, + failed = _getTestCounts.failed, + skipped = _getTestCounts.skipped, + todo = _getTestCounts.todo; + + if (failed) { + return "failed"; + } else { + if (skipped === total) { + return "skipped"; + } else if (todo === total) { + return "todo"; + } else { + return "passed"; + } + } + } + }]); + return SuiteReport; + }(); + + var focused = false; + + var moduleStack = []; + + function createModule(name, testEnvironment, modifiers) { + var parentModule = moduleStack.length ? moduleStack.slice(-1)[0] : null; + var moduleName = parentModule !== null ? [parentModule.name, name].join(" > ") : name; + var parentSuite = parentModule ? parentModule.suiteReport : globalSuite; + + var skip = parentModule !== null && parentModule.skip || modifiers.skip; + var todo = parentModule !== null && parentModule.todo || modifiers.todo; + + var module = { + name: moduleName, + parentModule: parentModule, + tests: [], + moduleId: generateHash(moduleName), + testsRun: 0, + unskippedTestsRun: 0, + childModules: [], + suiteReport: new SuiteReport(name, parentSuite), + + // Pass along `skip` and `todo` properties from parent module, in case + // there is one, to childs. And use own otherwise. + // This property will be used to mark own tests and tests of child suites + // as either `skipped` or `todo`. + skip: skip, + todo: skip ? false : todo + }; + + var env = {}; + if (parentModule) { + parentModule.childModules.push(module); + extend(env, parentModule.testEnvironment); + } + extend(env, testEnvironment); + module.testEnvironment = env; + + config.modules.push(module); + return module; + } + + function processModule(name, options, executeNow) { + var modifiers = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : {}; + + if (objectType(options) === "function") { + executeNow = options; + options = undefined; + } + + var module = createModule(name, options, modifiers); + + // Move any hooks to a 'hooks' object + var testEnvironment = module.testEnvironment; + var hooks = module.hooks = {}; + + setHookFromEnvironment(hooks, testEnvironment, "before"); + setHookFromEnvironment(hooks, testEnvironment, "beforeEach"); + setHookFromEnvironment(hooks, testEnvironment, "afterEach"); + setHookFromEnvironment(hooks, testEnvironment, "after"); + + var moduleFns = { + before: setHookFunction(module, "before"), + beforeEach: setHookFunction(module, "beforeEach"), + afterEach: setHookFunction(module, "afterEach"), + after: setHookFunction(module, "after") + }; + + var currentModule = config.currentModule; + if (objectType(executeNow) === "function") { + moduleStack.push(module); + config.currentModule = module; + executeNow.call(module.testEnvironment, moduleFns); + moduleStack.pop(); + module = module.parentModule || currentModule; + } + + config.currentModule = module; + + function setHookFromEnvironment(hooks, environment, name) { + var potentialHook = environment[name]; + hooks[name] = typeof potentialHook === "function" ? [potentialHook] : []; + delete environment[name]; + } + + function setHookFunction(module, hookName) { + return function setHook(callback) { + module.hooks[hookName].push(callback); + }; + } + } + + function module$1(name, options, executeNow) { + if (focused) { + return; + } + + processModule(name, options, executeNow); + } + + module$1.only = function () { + if (focused) { + return; + } + + config.modules.length = 0; + config.queue.length = 0; + + module$1.apply(undefined, arguments); + + focused = true; + }; + + module$1.skip = function (name, options, executeNow) { + if (focused) { + return; + } + + processModule(name, options, executeNow, { skip: true }); + }; + + module$1.todo = function (name, options, executeNow) { + if (focused) { + return; + } + + processModule(name, options, executeNow, { todo: true }); + }; + + var LISTENERS = Object.create(null); + var SUPPORTED_EVENTS = ["runStart", "suiteStart", "testStart", "assertion", "testEnd", "suiteEnd", "runEnd"]; + + /** + * Emits an event with the specified data to all currently registered listeners. + * Callbacks will fire in the order in which they are registered (FIFO). This + * function is not exposed publicly; it is used by QUnit internals to emit + * logging events. + * + * @private + * @method emit + * @param {String} eventName + * @param {Object} data + * @return {Void} + */ + function emit(eventName, data) { + if (objectType(eventName) !== "string") { + throw new TypeError("eventName must be a string when emitting an event"); + } + + // Clone the callbacks in case one of them registers a new callback + var originalCallbacks = LISTENERS[eventName]; + var callbacks = originalCallbacks ? [].concat(toConsumableArray(originalCallbacks)) : []; + + for (var i = 0; i < callbacks.length; i++) { + callbacks[i](data); + } + } + + /** + * Registers a callback as a listener to the specified event. + * + * @public + * @method on + * @param {String} eventName + * @param {Function} callback + * @return {Void} + */ + function on(eventName, callback) { + if (objectType(eventName) !== "string") { + throw new TypeError("eventName must be a string when registering a listener"); + } else if (!inArray(eventName, SUPPORTED_EVENTS)) { + var events = SUPPORTED_EVENTS.join(", "); + throw new Error("\"" + eventName + "\" is not a valid event; must be one of: " + events + "."); + } else if (objectType(callback) !== "function") { + throw new TypeError("callback must be a function when registering a listener"); + } + + if (!LISTENERS[eventName]) { + LISTENERS[eventName] = []; + } + + // Don't register the same callback more than once + if (!inArray(callback, LISTENERS[eventName])) { + LISTENERS[eventName].push(callback); + } + } + + function objectOrFunction(x) { + var type = typeof x === 'undefined' ? 'undefined' : _typeof(x); + return x !== null && (type === 'object' || type === 'function'); + } + + function isFunction(x) { + return typeof x === 'function'; + } + + + + var _isArray = void 0; + if (Array.isArray) { + _isArray = Array.isArray; + } else { + _isArray = function _isArray(x) { + return Object.prototype.toString.call(x) === '[object Array]'; + }; + } + + var isArray = _isArray; + + var len = 0; + var vertxNext = void 0; + var customSchedulerFn = void 0; + + var asap = function asap(callback, arg) { + queue[len] = callback; + queue[len + 1] = arg; + len += 2; + if (len === 2) { + // If len is 2, that means that we need to schedule an async flush. + // If additional callbacks are queued before the queue is flushed, they + // will be processed by this flush that we are scheduling. + if (customSchedulerFn) { + customSchedulerFn(flush); + } else { + scheduleFlush(); + } + } + }; + + function setScheduler(scheduleFn) { + customSchedulerFn = scheduleFn; + } + + function setAsap(asapFn) { + asap = asapFn; + } + + var browserWindow = typeof window !== 'undefined' ? window : undefined; + var browserGlobal = browserWindow || {}; + var BrowserMutationObserver = browserGlobal.MutationObserver || browserGlobal.WebKitMutationObserver; + var isNode = typeof self === 'undefined' && typeof process !== 'undefined' && {}.toString.call(process) === '[object process]'; + + // test for web worker but not in IE10 + var isWorker = typeof Uint8ClampedArray !== 'undefined' && typeof importScripts !== 'undefined' && typeof MessageChannel !== 'undefined'; + + // node + function useNextTick() { + // node version 0.10.x displays a deprecation warning when nextTick is used recursively + // see https://github.com/cujojs/when/issues/410 for details + return function () { + return process.nextTick(flush); + }; + } + + // vertx + function useVertxTimer() { + if (typeof vertxNext !== 'undefined') { + return function () { + vertxNext(flush); + }; + } + + return useSetTimeout(); + } + + function useMutationObserver() { + var iterations = 0; + var observer = new BrowserMutationObserver(flush); + var node = document.createTextNode(''); + observer.observe(node, { characterData: true }); + + return function () { + node.data = iterations = ++iterations % 2; + }; + } + + // web worker + function useMessageChannel() { + var channel = new MessageChannel(); + channel.port1.onmessage = flush; + return function () { + return channel.port2.postMessage(0); + }; + } + + function useSetTimeout() { + // Store setTimeout reference so es6-promise will be unaffected by + // other code modifying setTimeout (like sinon.useFakeTimers()) + var globalSetTimeout = setTimeout; + return function () { + return globalSetTimeout(flush, 1); + }; + } + + var queue = new Array(1000); + function flush() { + for (var i = 0; i < len; i += 2) { + var callback = queue[i]; + var arg = queue[i + 1]; + + callback(arg); + + queue[i] = undefined; + queue[i + 1] = undefined; + } + + len = 0; + } + + function attemptVertx() { + try { + var vertx = Function('return this')().require('vertx'); + vertxNext = vertx.runOnLoop || vertx.runOnContext; + return useVertxTimer(); + } catch (e) { + return useSetTimeout(); + } + } + + var scheduleFlush = void 0; + // Decide what async method to use to triggering processing of queued callbacks: + if (isNode) { + scheduleFlush = useNextTick(); + } else if (BrowserMutationObserver) { + scheduleFlush = useMutationObserver(); + } else if (isWorker) { + scheduleFlush = useMessageChannel(); + } else if (browserWindow === undefined && typeof require === 'function') { + scheduleFlush = attemptVertx(); + } else { + scheduleFlush = useSetTimeout(); + } + + function then(onFulfillment, onRejection) { + var parent = this; + + var child = new this.constructor(noop); + + if (child[PROMISE_ID] === undefined) { + makePromise(child); + } + + var _state = parent._state; + + + if (_state) { + var callback = arguments[_state - 1]; + asap(function () { + return invokeCallback(_state, child, callback, parent._result); + }); + } else { + subscribe(parent, child, onFulfillment, onRejection); + } + + return child; + } + + /** + `Promise.resolve` returns a promise that will become resolved with the + passed `value`. It is shorthand for the following: + + ```javascript + let promise = new Promise(function(resolve, reject){ + resolve(1); + }); + + promise.then(function(value){ + // value === 1 + }); + ``` + + Instead of writing the above, your code now simply becomes the following: + + ```javascript + let promise = Promise.resolve(1); + + promise.then(function(value){ + // value === 1 + }); + ``` + + @method resolve + @static + @param {Any} value value that the returned promise will be resolved with + Useful for tooling. + @return {Promise} a promise that will become fulfilled with the given + `value` + */ + function resolve$1(object) { + /*jshint validthis:true */ + var Constructor = this; + + if (object && (typeof object === 'undefined' ? 'undefined' : _typeof(object)) === 'object' && object.constructor === Constructor) { + return object; + } + + var promise = new Constructor(noop); + resolve(promise, object); + return promise; + } + + var PROMISE_ID = Math.random().toString(36).substring(2); + + function noop() {} + + var PENDING = void 0; + var FULFILLED = 1; + var REJECTED = 2; + + var TRY_CATCH_ERROR = { error: null }; + + function selfFulfillment() { + return new TypeError("You cannot resolve a promise with itself"); + } + + function cannotReturnOwn() { + return new TypeError('A promises callback cannot return that same promise.'); + } + + function getThen(promise) { + try { + return promise.then; + } catch (error) { + TRY_CATCH_ERROR.error = error; + return TRY_CATCH_ERROR; + } + } + + function tryThen(then$$1, value, fulfillmentHandler, rejectionHandler) { + try { + then$$1.call(value, fulfillmentHandler, rejectionHandler); + } catch (e) { + return e; + } + } + + function handleForeignThenable(promise, thenable, then$$1) { + asap(function (promise) { + var sealed = false; + var error = tryThen(then$$1, thenable, function (value) { + if (sealed) { + return; + } + sealed = true; + if (thenable !== value) { + resolve(promise, value); + } else { + fulfill(promise, value); + } + }, function (reason) { + if (sealed) { + return; + } + sealed = true; + + reject(promise, reason); + }, 'Settle: ' + (promise._label || ' unknown promise')); + + if (!sealed && error) { + sealed = true; + reject(promise, error); + } + }, promise); + } + + function handleOwnThenable(promise, thenable) { + if (thenable._state === FULFILLED) { + fulfill(promise, thenable._result); + } else if (thenable._state === REJECTED) { + reject(promise, thenable._result); + } else { + subscribe(thenable, undefined, function (value) { + return resolve(promise, value); + }, function (reason) { + return reject(promise, reason); + }); + } + } + + function handleMaybeThenable(promise, maybeThenable, then$$1) { + if (maybeThenable.constructor === promise.constructor && then$$1 === then && maybeThenable.constructor.resolve === resolve$1) { + handleOwnThenable(promise, maybeThenable); + } else { + if (then$$1 === TRY_CATCH_ERROR) { + reject(promise, TRY_CATCH_ERROR.error); + TRY_CATCH_ERROR.error = null; + } else if (then$$1 === undefined) { + fulfill(promise, maybeThenable); + } else if (isFunction(then$$1)) { + handleForeignThenable(promise, maybeThenable, then$$1); + } else { + fulfill(promise, maybeThenable); + } + } + } + + function resolve(promise, value) { + if (promise === value) { + reject(promise, selfFulfillment()); + } else if (objectOrFunction(value)) { + handleMaybeThenable(promise, value, getThen(value)); + } else { + fulfill(promise, value); + } + } + + function publishRejection(promise) { + if (promise._onerror) { + promise._onerror(promise._result); + } + + publish(promise); + } + + function fulfill(promise, value) { + if (promise._state !== PENDING) { + return; + } + + promise._result = value; + promise._state = FULFILLED; + + if (promise._subscribers.length !== 0) { + asap(publish, promise); + } + } + + function reject(promise, reason) { + if (promise._state !== PENDING) { + return; + } + promise._state = REJECTED; + promise._result = reason; + + asap(publishRejection, promise); + } + + function subscribe(parent, child, onFulfillment, onRejection) { + var _subscribers = parent._subscribers; + var length = _subscribers.length; + + + parent._onerror = null; + + _subscribers[length] = child; + _subscribers[length + FULFILLED] = onFulfillment; + _subscribers[length + REJECTED] = onRejection; + + if (length === 0 && parent._state) { + asap(publish, parent); + } + } + + function publish(promise) { + var subscribers = promise._subscribers; + var settled = promise._state; + + if (subscribers.length === 0) { + return; + } + + var child = void 0, + callback = void 0, + detail = promise._result; + + for (var i = 0; i < subscribers.length; i += 3) { + child = subscribers[i]; + callback = subscribers[i + settled]; + + if (child) { + invokeCallback(settled, child, callback, detail); + } else { + callback(detail); + } + } + + promise._subscribers.length = 0; + } + + function tryCatch(callback, detail) { + try { + return callback(detail); + } catch (e) { + TRY_CATCH_ERROR.error = e; + return TRY_CATCH_ERROR; + } + } + + function invokeCallback(settled, promise, callback, detail) { + var hasCallback = isFunction(callback), + value = void 0, + error = void 0, + succeeded = void 0, + failed = void 0; + + if (hasCallback) { + value = tryCatch(callback, detail); + + if (value === TRY_CATCH_ERROR) { + failed = true; + error = value.error; + value.error = null; + } else { + succeeded = true; + } + + if (promise === value) { + reject(promise, cannotReturnOwn()); + return; + } + } else { + value = detail; + succeeded = true; + } + + if (promise._state !== PENDING) { + // noop + } else if (hasCallback && succeeded) { + resolve(promise, value); + } else if (failed) { + reject(promise, error); + } else if (settled === FULFILLED) { + fulfill(promise, value); + } else if (settled === REJECTED) { + reject(promise, value); + } + } + + function initializePromise(promise, resolver) { + try { + resolver(function resolvePromise(value) { + resolve(promise, value); + }, function rejectPromise(reason) { + reject(promise, reason); + }); + } catch (e) { + reject(promise, e); + } + } + + var id = 0; + function nextId() { + return id++; + } + + function makePromise(promise) { + promise[PROMISE_ID] = id++; + promise._state = undefined; + promise._result = undefined; + promise._subscribers = []; + } + + function validationError() { + return new Error('Array Methods must be provided an Array'); + } + + var Enumerator = function () { + function Enumerator(Constructor, input) { + classCallCheck(this, Enumerator); + + this._instanceConstructor = Constructor; + this.promise = new Constructor(noop); + + if (!this.promise[PROMISE_ID]) { + makePromise(this.promise); + } + + if (isArray(input)) { + this.length = input.length; + this._remaining = input.length; + + this._result = new Array(this.length); + + if (this.length === 0) { + fulfill(this.promise, this._result); + } else { + this.length = this.length || 0; + this._enumerate(input); + if (this._remaining === 0) { + fulfill(this.promise, this._result); + } + } + } else { + reject(this.promise, validationError()); + } + } + + createClass(Enumerator, [{ + key: '_enumerate', + value: function _enumerate(input) { + for (var i = 0; this._state === PENDING && i < input.length; i++) { + this._eachEntry(input[i], i); + } + } + }, { + key: '_eachEntry', + value: function _eachEntry(entry, i) { + var c = this._instanceConstructor; + var resolve$$1 = c.resolve; + + + if (resolve$$1 === resolve$1) { + var _then = getThen(entry); + + if (_then === then && entry._state !== PENDING) { + this._settledAt(entry._state, i, entry._result); + } else if (typeof _then !== 'function') { + this._remaining--; + this._result[i] = entry; + } else if (c === Promise$2) { + var promise = new c(noop); + handleMaybeThenable(promise, entry, _then); + this._willSettleAt(promise, i); + } else { + this._willSettleAt(new c(function (resolve$$1) { + return resolve$$1(entry); + }), i); + } + } else { + this._willSettleAt(resolve$$1(entry), i); + } + } + }, { + key: '_settledAt', + value: function _settledAt(state, i, value) { + var promise = this.promise; + + + if (promise._state === PENDING) { + this._remaining--; + + if (state === REJECTED) { + reject(promise, value); + } else { + this._result[i] = value; + } + } + + if (this._remaining === 0) { + fulfill(promise, this._result); + } + } + }, { + key: '_willSettleAt', + value: function _willSettleAt(promise, i) { + var enumerator = this; + + subscribe(promise, undefined, function (value) { + return enumerator._settledAt(FULFILLED, i, value); + }, function (reason) { + return enumerator._settledAt(REJECTED, i, reason); + }); + } + }]); + return Enumerator; + }(); + + /** + `Promise.all` accepts an array of promises, and returns a new promise which + is fulfilled with an array of fulfillment values for the passed promises, or + rejected with the reason of the first passed promise to be rejected. It casts all + elements of the passed iterable to promises as it runs this algorithm. + + Example: + + ```javascript + let promise1 = resolve(1); + let promise2 = resolve(2); + let promise3 = resolve(3); + let promises = [ promise1, promise2, promise3 ]; + + Promise.all(promises).then(function(array){ + // The array here would be [ 1, 2, 3 ]; + }); + ``` + + If any of the `promises` given to `all` are rejected, the first promise + that is rejected will be given as an argument to the returned promises's + rejection handler. For example: + + Example: + + ```javascript + let promise1 = resolve(1); + let promise2 = reject(new Error("2")); + let promise3 = reject(new Error("3")); + let promises = [ promise1, promise2, promise3 ]; + + Promise.all(promises).then(function(array){ + // Code here never runs because there are rejected promises! + }, function(error) { + // error.message === "2" + }); + ``` + + @method all + @static + @param {Array} entries array of promises + @param {String} label optional string for labeling the promise. + Useful for tooling. + @return {Promise} promise that is fulfilled when all `promises` have been + fulfilled, or rejected if any of them become rejected. + @static + */ + function all(entries) { + return new Enumerator(this, entries).promise; + } + + /** + `Promise.race` returns a new promise which is settled in the same way as the + first passed promise to settle. + + Example: + + ```javascript + let promise1 = new Promise(function(resolve, reject){ + setTimeout(function(){ + resolve('promise 1'); + }, 200); + }); + + let promise2 = new Promise(function(resolve, reject){ + setTimeout(function(){ + resolve('promise 2'); + }, 100); + }); + + Promise.race([promise1, promise2]).then(function(result){ + // result === 'promise 2' because it was resolved before promise1 + // was resolved. + }); + ``` + + `Promise.race` is deterministic in that only the state of the first + settled promise matters. For example, even if other promises given to the + `promises` array argument are resolved, but the first settled promise has + become rejected before the other promises became fulfilled, the returned + promise will become rejected: + + ```javascript + let promise1 = new Promise(function(resolve, reject){ + setTimeout(function(){ + resolve('promise 1'); + }, 200); + }); + + let promise2 = new Promise(function(resolve, reject){ + setTimeout(function(){ + reject(new Error('promise 2')); + }, 100); + }); + + Promise.race([promise1, promise2]).then(function(result){ + // Code here never runs + }, function(reason){ + // reason.message === 'promise 2' because promise 2 became rejected before + // promise 1 became fulfilled + }); + ``` + + An example real-world use case is implementing timeouts: + + ```javascript + Promise.race([ajax('foo.json'), timeout(5000)]) + ``` + + @method race + @static + @param {Array} promises array of promises to observe + Useful for tooling. + @return {Promise} a promise which settles in the same way as the first passed + promise to settle. + */ + function race(entries) { + /*jshint validthis:true */ + var Constructor = this; + + if (!isArray(entries)) { + return new Constructor(function (_, reject) { + return reject(new TypeError('You must pass an array to race.')); + }); + } else { + return new Constructor(function (resolve, reject) { + var length = entries.length; + for (var i = 0; i < length; i++) { + Constructor.resolve(entries[i]).then(resolve, reject); + } + }); + } + } + + /** + `Promise.reject` returns a promise rejected with the passed `reason`. + It is shorthand for the following: + + ```javascript + let promise = new Promise(function(resolve, reject){ + reject(new Error('WHOOPS')); + }); + + promise.then(function(value){ + // Code here doesn't run because the promise is rejected! + }, function(reason){ + // reason.message === 'WHOOPS' + }); + ``` + + Instead of writing the above, your code now simply becomes the following: + + ```javascript + let promise = Promise.reject(new Error('WHOOPS')); + + promise.then(function(value){ + // Code here doesn't run because the promise is rejected! + }, function(reason){ + // reason.message === 'WHOOPS' + }); + ``` + + @method reject + @static + @param {Any} reason value that the returned promise will be rejected with. + Useful for tooling. + @return {Promise} a promise rejected with the given `reason`. + */ + function reject$1(reason) { + /*jshint validthis:true */ + var Constructor = this; + var promise = new Constructor(noop); + reject(promise, reason); + return promise; + } + + function needsResolver() { + throw new TypeError('You must pass a resolver function as the first argument to the promise constructor'); + } + + function needsNew() { + throw new TypeError("Failed to construct 'Promise': Please use the 'new' operator, this object constructor cannot be called as a function."); + } + + /** + Promise objects represent the eventual result of an asynchronous operation. The + primary way of interacting with a promise is through its `then` method, which + registers callbacks to receive either a promise's eventual value or the reason + why the promise cannot be fulfilled. + + Terminology + ----------- + + - `promise` is an object or function with a `then` method whose behavior conforms to this specification. + - `thenable` is an object or function that defines a `then` method. + - `value` is any legal JavaScript value (including undefined, a thenable, or a promise). + - `exception` is a value that is thrown using the throw statement. + - `reason` is a value that indicates why a promise was rejected. + - `settled` the final resting state of a promise, fulfilled or rejected. + + A promise can be in one of three states: pending, fulfilled, or rejected. + + Promises that are fulfilled have a fulfillment value and are in the fulfilled + state. Promises that are rejected have a rejection reason and are in the + rejected state. A fulfillment value is never a thenable. + + Promises can also be said to *resolve* a value. If this value is also a + promise, then the original promise's settled state will match the value's + settled state. So a promise that *resolves* a promise that rejects will + itself reject, and a promise that *resolves* a promise that fulfills will + itself fulfill. + + + Basic Usage: + ------------ + + ```js + let promise = new Promise(function(resolve, reject) { + // on success + resolve(value); + + // on failure + reject(reason); + }); + + promise.then(function(value) { + // on fulfillment + }, function(reason) { + // on rejection + }); + ``` + + Advanced Usage: + --------------- + + Promises shine when abstracting away asynchronous interactions such as + `XMLHttpRequest`s. + + ```js + function getJSON(url) { + return new Promise(function(resolve, reject){ + let xhr = new XMLHttpRequest(); + + xhr.open('GET', url); + xhr.onreadystatechange = handler; + xhr.responseType = 'json'; + xhr.setRequestHeader('Accept', 'application/json'); + xhr.send(); + + function handler() { + if (this.readyState === this.DONE) { + if (this.status === 200) { + resolve(this.response); + } else { + reject(new Error('getJSON: `' + url + '` failed with status: [' + this.status + ']')); + } + } + }; + }); + } + + getJSON('/posts.json').then(function(json) { + // on fulfillment + }, function(reason) { + // on rejection + }); + ``` + + Unlike callbacks, promises are great composable primitives. + + ```js + Promise.all([ + getJSON('/posts'), + getJSON('/comments') + ]).then(function(values){ + values[0] // => postsJSON + values[1] // => commentsJSON + + return values; + }); + ``` + + @class Promise + @param {Function} resolver + Useful for tooling. + @constructor + */ + + var Promise$2 = function () { + function Promise(resolver) { + classCallCheck(this, Promise); + + this[PROMISE_ID] = nextId(); + this._result = this._state = undefined; + this._subscribers = []; + + if (noop !== resolver) { + typeof resolver !== 'function' && needsResolver(); + this instanceof Promise ? initializePromise(this, resolver) : needsNew(); + } + } + + /** + The primary way of interacting with a promise is through its `then` method, + which registers callbacks to receive either a promise's eventual value or the + reason why the promise cannot be fulfilled. + ```js + findUser().then(function(user){ + // user is available + }, function(reason){ + // user is unavailable, and you are given the reason why + }); + ``` + Chaining + -------- + The return value of `then` is itself a promise. This second, 'downstream' + promise is resolved with the return value of the first promise's fulfillment + or rejection handler, or rejected if the handler throws an exception. + ```js + findUser().then(function (user) { + return user.name; + }, function (reason) { + return 'default name'; + }).then(function (userName) { + // If `findUser` fulfilled, `userName` will be the user's name, otherwise it + // will be `'default name'` + }); + findUser().then(function (user) { + throw new Error('Found user, but still unhappy'); + }, function (reason) { + throw new Error('`findUser` rejected and we're unhappy'); + }).then(function (value) { + // never reached + }, function (reason) { + // if `findUser` fulfilled, `reason` will be 'Found user, but still unhappy'. + // If `findUser` rejected, `reason` will be '`findUser` rejected and we're unhappy'. + }); + ``` + If the downstream promise does not specify a rejection handler, rejection reasons will be propagated further downstream. + ```js + findUser().then(function (user) { + throw new PedagogicalException('Upstream error'); + }).then(function (value) { + // never reached + }).then(function (value) { + // never reached + }, function (reason) { + // The `PedgagocialException` is propagated all the way down to here + }); + ``` + Assimilation + ------------ + Sometimes the value you want to propagate to a downstream promise can only be + retrieved asynchronously. This can be achieved by returning a promise in the + fulfillment or rejection handler. The downstream promise will then be pending + until the returned promise is settled. This is called *assimilation*. + ```js + findUser().then(function (user) { + return findCommentsByAuthor(user); + }).then(function (comments) { + // The user's comments are now available + }); + ``` + If the assimliated promise rejects, then the downstream promise will also reject. + ```js + findUser().then(function (user) { + return findCommentsByAuthor(user); + }).then(function (comments) { + // If `findCommentsByAuthor` fulfills, we'll have the value here + }, function (reason) { + // If `findCommentsByAuthor` rejects, we'll have the reason here + }); + ``` + Simple Example + -------------- + Synchronous Example + ```javascript + let result; + try { + result = findResult(); + // success + } catch(reason) { + // failure + } + ``` + Errback Example + ```js + findResult(function(result, err){ + if (err) { + // failure + } else { + // success + } + }); + ``` + Promise Example; + ```javascript + findResult().then(function(result){ + // success + }, function(reason){ + // failure + }); + ``` + Advanced Example + -------------- + Synchronous Example + ```javascript + let author, books; + try { + author = findAuthor(); + books = findBooksByAuthor(author); + // success + } catch(reason) { + // failure + } + ``` + Errback Example + ```js + function foundBooks(books) { + } + function failure(reason) { + } + findAuthor(function(author, err){ + if (err) { + failure(err); + // failure + } else { + try { + findBoooksByAuthor(author, function(books, err) { + if (err) { + failure(err); + } else { + try { + foundBooks(books); + } catch(reason) { + failure(reason); + } + } + }); + } catch(error) { + failure(err); + } + // success + } + }); + ``` + Promise Example; + ```javascript + findAuthor(). + then(findBooksByAuthor). + then(function(books){ + // found books + }).catch(function(reason){ + // something went wrong + }); + ``` + @method then + @param {Function} onFulfilled + @param {Function} onRejected + Useful for tooling. + @return {Promise} + */ + + /** + `catch` is simply sugar for `then(undefined, onRejection)` which makes it the same + as the catch block of a try/catch statement. + ```js + function findAuthor(){ + throw new Error('couldn't find that author'); + } + // synchronous + try { + findAuthor(); + } catch(reason) { + // something went wrong + } + // async with promises + findAuthor().catch(function(reason){ + // something went wrong + }); + ``` + @method catch + @param {Function} onRejection + Useful for tooling. + @return {Promise} + */ + + + createClass(Promise, [{ + key: 'catch', + value: function _catch(onRejection) { + return this.then(null, onRejection); + } + + /** + `finally` will be invoked regardless of the promise's fate just as native + try/catch/finally behaves + + Synchronous example: + + ```js + findAuthor() { + if (Math.random() > 0.5) { + throw new Error(); + } + return new Author(); + } + + try { + return findAuthor(); // succeed or fail + } catch(error) { + return findOtherAuther(); + } finally { + // always runs + // doesn't affect the return value + } + ``` + + Asynchronous example: + + ```js + findAuthor().catch(function(reason){ + return findOtherAuther(); + }).finally(function(){ + // author was either found, or not + }); + ``` + + @method finally + @param {Function} callback + @return {Promise} + */ + + }, { + key: 'finally', + value: function _finally(callback) { + var promise = this; + var constructor = promise.constructor; + + if (isFunction(callback)) { + return promise.then(function (value) { + return constructor.resolve(callback()).then(function () { + return value; + }); + }, function (reason) { + return constructor.resolve(callback()).then(function () { + throw reason; + }); + }); + } + + return promise.then(callback, callback); + } + }]); + return Promise; + }(); + + Promise$2.prototype.then = then; + Promise$2.all = all; + Promise$2.race = race; + Promise$2.resolve = resolve$1; + Promise$2.reject = reject$1; + Promise$2._setScheduler = setScheduler; + Promise$2._setAsap = setAsap; + Promise$2._asap = asap; + + /*global self*/ + function polyfill() { + var local = void 0; + + if (typeof global !== 'undefined') { + local = global; + } else if (typeof self !== 'undefined') { + local = self; + } else { + try { + local = Function('return this')(); + } catch (e) { + throw new Error('polyfill failed because global object is unavailable in this environment'); + } + } + + var P = local.Promise; + + if (P) { + var promiseToString = null; + try { + promiseToString = Object.prototype.toString.call(P.resolve()); + } catch (e) { + // silently ignored + } + + if (promiseToString === '[object Promise]' && !P.cast) { + return; + } + } + + local.Promise = Promise$2; + } + + // Strange compat.. + Promise$2.polyfill = polyfill; + Promise$2.Promise = Promise$2; + + var Promise$1 = typeof Promise !== "undefined" ? Promise : Promise$2; + + // Register logging callbacks + function registerLoggingCallbacks(obj) { + var i, + l, + key, + callbackNames = ["begin", "done", "log", "testStart", "testDone", "moduleStart", "moduleDone"]; + + function registerLoggingCallback(key) { + var loggingCallback = function loggingCallback(callback) { + if (objectType(callback) !== "function") { + throw new Error("QUnit logging methods require a callback function as their first parameters."); + } + + config.callbacks[key].push(callback); + }; + + return loggingCallback; + } + + for (i = 0, l = callbackNames.length; i < l; i++) { + key = callbackNames[i]; + + // Initialize key collection of logging callback + if (objectType(config.callbacks[key]) === "undefined") { + config.callbacks[key] = []; + } + + obj[key] = registerLoggingCallback(key); + } + } + + function runLoggingCallbacks(key, args) { + var callbacks = config.callbacks[key]; + + // Handling 'log' callbacks separately. Unlike the other callbacks, + // the log callback is not controlled by the processing queue, + // but rather used by asserts. Hence to promisfy the 'log' callback + // would mean promisfying each step of a test + if (key === "log") { + callbacks.map(function (callback) { + return callback(args); + }); + return; + } + + // ensure that each callback is executed serially + return callbacks.reduce(function (promiseChain, callback) { + return promiseChain.then(function () { + return Promise$1.resolve(callback(args)); + }); + }, Promise$1.resolve([])); + } + + // Doesn't support IE9, it will return undefined on these browsers + // See also https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Error/Stack + var fileName = (sourceFromStacktrace(0) || "").replace(/(:\d+)+\)?/, "").replace(/.+\//, ""); + + function extractStacktrace(e, offset) { + offset = offset === undefined ? 4 : offset; + + var stack, include, i; + + if (e && e.stack) { + 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]; + } + } + + function sourceFromStacktrace(offset) { + var error = new Error(); + + // Support: Safari <=7 only, IE <=10 - 11 only + // Not all browsers generate the `stack` property for `new Error()`, see also #636 + if (!error.stack) { + try { + throw error; + } catch (err) { + error = err; + } + } + + return extractStacktrace(error, offset); + } + + var priorityCount = 0; + var unitSampler = void 0; + + // This is a queue of functions that are tasks within a single test. + // After tests are dequeued from config.queue they are expanded into + // a set of tasks in this queue. + var taskQueue = []; + + /** + * Advances the taskQueue to the next task. If the taskQueue is empty, + * process the testQueue + */ + function advance() { + advanceTaskQueue(); + + if (!taskQueue.length && !config.blocking && !config.current) { + advanceTestQueue(); + } + } + + /** + * Advances the taskQueue with an increased depth + */ + function advanceTaskQueue() { + var start = now(); + config.depth = (config.depth || 0) + 1; + + processTaskQueue(start); + + config.depth--; + } + + /** + * Process the first task on the taskQueue as a promise. + * Each task is a function returned by https://github.com/qunitjs/qunit/blob/master/src/test.js#L381 + */ + function processTaskQueue(start) { + if (taskQueue.length && !config.blocking) { + var elapsedTime = now() - start; + + if (!defined.setTimeout || config.updateRate <= 0 || elapsedTime < config.updateRate) { + var task = taskQueue.shift(); + Promise$1.resolve(task()).then(function () { + if (!taskQueue.length) { + advance(); + } else { + processTaskQueue(start); + } + }); + } else { + setTimeout$1(advance); + } + } + } + + /** + * Advance the testQueue to the next test to process. Call done() if testQueue completes. + */ + function advanceTestQueue() { + if (!config.blocking && !config.queue.length && config.depth === 0) { + done(); + return; + } + + var testTasks = config.queue.shift(); + addToTaskQueue(testTasks()); + + if (priorityCount > 0) { + priorityCount--; + } + + advance(); + } + + /** + * Enqueue the tasks for a test into the task queue. + * @param {Array} tasksArray + */ + function addToTaskQueue(tasksArray) { + taskQueue.push.apply(taskQueue, toConsumableArray(tasksArray)); + } + + /** + * Return the number of tasks remaining in the task queue to be processed. + * @return {Number} + */ + function taskQueueLength() { + return taskQueue.length; + } + + /** + * Adds a test to the TestQueue for execution. + * @param {Function} testTasksFunc + * @param {Boolean} prioritize + * @param {String} seed + */ + function addToTestQueue(testTasksFunc, prioritize, seed) { + if (prioritize) { + config.queue.splice(priorityCount++, 0, testTasksFunc); + } else if (seed) { + if (!unitSampler) { + unitSampler = unitSamplerGenerator(seed); + } + + // Insert into a random position after all prioritized items + var index = Math.floor(unitSampler() * (config.queue.length - priorityCount + 1)); + config.queue.splice(priorityCount + index, 0, testTasksFunc); + } else { + config.queue.push(testTasksFunc); + } + } + + /** + * Creates a seeded "sample" generator which is used for randomizing tests. + */ + function unitSamplerGenerator(seed) { + + // 32-bit xorshift, requires only a nonzero seed + // http://excamera.com/sphinx/article-xorshift.html + var sample = parseInt(generateHash(seed), 16) || -1; + return function () { + sample ^= sample << 13; + sample ^= sample >>> 17; + sample ^= sample << 5; + + // ECMAScript has no unsigned number type + if (sample < 0) { + sample += 0x100000000; + } + + return sample / 0x100000000; + }; + } + + /** + * This function is called when the ProcessingQueue is done processing all + * items. It handles emitting the final run events. + */ + function done() { + var storage = config.storage; + + ProcessingQueue.finished = true; + + var runtime = now() - config.started; + var passed = config.stats.all - config.stats.bad; + + if (config.stats.all === 0) { + + if (config.filter && config.filter.length) { + throw new Error("No tests matched the filter \"" + config.filter + "\"."); + } + + if (config.module && config.module.length) { + throw new Error("No tests matched the module \"" + config.module + "\"."); + } + + if (config.moduleId && config.moduleId.length) { + throw new Error("No tests matched the moduleId \"" + config.moduleId + "\"."); + } + + if (config.testId && config.testId.length) { + throw new Error("No tests matched the testId \"" + config.testId + "\"."); + } + + throw new Error("No tests were run."); + } + + emit("runEnd", globalSuite.end(true)); + runLoggingCallbacks("done", { + passed: passed, + failed: config.stats.bad, + total: config.stats.all, + runtime: runtime + }).then(function () { + + // Clear own storage items if all tests passed + if (storage && config.stats.bad === 0) { + for (var i = storage.length - 1; i >= 0; i--) { + var key = storage.key(i); + + if (key.indexOf("qunit-test-") === 0) { + storage.removeItem(key); + } + } + } + }); + } + + var ProcessingQueue = { + finished: false, + add: addToTestQueue, + advance: advance, + taskCount: taskQueueLength + }; + + var TestReport = function () { + function TestReport(name, suite, options) { + classCallCheck(this, TestReport); + + this.name = name; + this.suiteName = suite.name; + this.fullName = suite.fullName.concat(name); + this.runtime = 0; + this.assertions = []; + + this.skipped = !!options.skip; + this.todo = !!options.todo; + + this.valid = options.valid; + + this._startTime = 0; + this._endTime = 0; + + suite.pushTest(this); + } + + createClass(TestReport, [{ + key: "start", + value: function start(recordTime) { + if (recordTime) { + this._startTime = performanceNow(); + if (performance) { + performance.mark("qunit_test_start"); + } + } + + return { + name: this.name, + suiteName: this.suiteName, + fullName: this.fullName.slice() + }; + } + }, { + key: "end", + value: function end(recordTime) { + if (recordTime) { + this._endTime = performanceNow(); + if (performance) { + performance.mark("qunit_test_end"); + + var testName = this.fullName.join(" – "); + + measure("QUnit Test: " + testName, "qunit_test_start", "qunit_test_end"); + } + } + + return extend(this.start(), { + runtime: this.getRuntime(), + status: this.getStatus(), + errors: this.getFailedAssertions(), + assertions: this.getAssertions() + }); + } + }, { + key: "pushAssertion", + value: function pushAssertion(assertion) { + this.assertions.push(assertion); + } + }, { + key: "getRuntime", + value: function getRuntime() { + return this._endTime - this._startTime; + } + }, { + key: "getStatus", + value: function getStatus() { + if (this.skipped) { + return "skipped"; + } + + var testPassed = this.getFailedAssertions().length > 0 ? this.todo : !this.todo; + + if (!testPassed) { + return "failed"; + } else if (this.todo) { + return "todo"; + } else { + return "passed"; + } + } + }, { + key: "getFailedAssertions", + value: function getFailedAssertions() { + return this.assertions.filter(function (assertion) { + return !assertion.passed; + }); + } + }, { + key: "getAssertions", + value: function getAssertions() { + return this.assertions.slice(); + } + + // Remove actual and expected values from assertions. This is to prevent + // leaking memory throughout a test suite. + + }, { + key: "slimAssertions", + value: function slimAssertions() { + this.assertions = this.assertions.map(function (assertion) { + delete assertion.actual; + delete assertion.expected; + return assertion; + }); + } + }]); + return TestReport; + }(); + + var focused$1 = false; + + function Test(settings) { + var i, l; + + ++Test.count; + + this.expected = null; + this.assertions = []; + this.semaphore = 0; + this.module = config.currentModule; + this.stack = sourceFromStacktrace(3); + this.steps = []; + this.timeout = undefined; + + // If a module is skipped, all its tests and the tests of the child suites + // should be treated as skipped even if they are defined as `only` or `todo`. + // As for `todo` module, all its tests will be treated as `todo` except for + // tests defined as `skip` which will be left intact. + // + // So, if a test is defined as `todo` and is inside a skipped module, we should + // then treat that test as if was defined as `skip`. + if (this.module.skip) { + settings.skip = true; + settings.todo = false; + + // Skipped tests should be left intact + } else if (this.module.todo && !settings.skip) { + settings.todo = true; + } + + extend(this, settings); + + this.testReport = new TestReport(settings.testName, this.module.suiteReport, { + todo: settings.todo, + skip: settings.skip, + valid: this.valid() + }); + + // Register unique strings + for (i = 0, l = this.module.tests; i < l.length; i++) { + if (this.module.tests[i].name === this.testName) { + this.testName += " "; + } + } + + this.testId = generateHash(this.module.name, this.testName); + + this.module.tests.push({ + name: this.testName, + testId: this.testId, + skip: !!settings.skip + }); + + if (settings.skip) { + + // Skipped tests will fully ignore any sent callback + this.callback = function () {}; + this.async = false; + this.expected = 0; + } else { + if (typeof this.callback !== "function") { + var method = this.todo ? "todo" : "test"; + + // eslint-disable-next-line max-len + throw new TypeError("You must provide a function as a test callback to QUnit." + method + "(\"" + settings.testName + "\")"); + } + + this.assert = new Assert(this); + } + } + + Test.count = 0; + + function getNotStartedModules(startModule) { + var module = startModule, + modules = []; + + while (module && module.testsRun === 0) { + modules.push(module); + module = module.parentModule; + } + + // The above push modules from the child to the parent + // return a reversed order with the top being the top most parent module + return modules.reverse(); + } + + Test.prototype = { + before: function before() { + var _this = this; + + var module = this.module, + notStartedModules = getNotStartedModules(module); + + // ensure the callbacks are executed serially for each module + var callbackPromises = notStartedModules.reduce(function (promiseChain, startModule) { + return promiseChain.then(function () { + startModule.stats = { all: 0, bad: 0, started: now() }; + emit("suiteStart", startModule.suiteReport.start(true)); + return runLoggingCallbacks("moduleStart", { + name: startModule.name, + tests: startModule.tests + }); + }); + }, Promise$1.resolve([])); + + return callbackPromises.then(function () { + config.current = _this; + + _this.testEnvironment = extend({}, module.testEnvironment); + + _this.started = now(); + emit("testStart", _this.testReport.start(true)); + return runLoggingCallbacks("testStart", { + name: _this.testName, + module: module.name, + testId: _this.testId, + previousFailure: _this.previousFailure + }).then(function () { + if (!config.pollution) { + saveGlobal(); + } + }); + }); + }, + + run: function run() { + var promise; + + config.current = this; + + this.callbackStarted = now(); + + if (config.notrycatch) { + runTest(this); + return; + } + + try { + runTest(this); + } catch (e) { + this.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) { + internalRecover(this); + } + } + + function runTest(test) { + promise = test.callback.call(test.testEnvironment, test.assert); + test.resolvePromise(promise); + + // If the test has a "lock" on it, but the timeout is 0, then we push a + // failure as the test should be synchronous. + if (test.timeout === 0 && test.semaphore !== 0) { + pushFailure("Test did not finish synchronously even though assert.timeout( 0 ) was used.", sourceFromStacktrace(2)); + } + } + }, + + after: function after() { + checkPollution(); + }, + + queueHook: function queueHook(hook, hookName, hookOwner) { + var _this2 = this; + + var callHook = function callHook() { + var promise = hook.call(_this2.testEnvironment, _this2.assert); + _this2.resolvePromise(promise, hookName); + }; + + var runHook = function runHook() { + if (hookName === "before") { + if (hookOwner.unskippedTestsRun !== 0) { + return; + } + + _this2.preserveEnvironment = true; + } + + // The 'after' hook should only execute when there are not tests left and + // when the 'after' and 'finish' tasks are the only tasks left to process + if (hookName === "after" && hookOwner.unskippedTestsRun !== numberOfUnskippedTests(hookOwner) - 1 && (config.queue.length > 0 || ProcessingQueue.taskCount() > 2)) { + return; + } + + config.current = _this2; + if (config.notrycatch) { + callHook(); + return; + } + try { + callHook(); + } catch (error) { + _this2.pushFailure(hookName + " failed on " + _this2.testName + ": " + (error.message || error), extractStacktrace(error, 0)); + } + }; + + return runHook; + }, + + + // Currently only used for module level hooks, can be used to add global level ones + hooks: function hooks(handler) { + var hooks = []; + + function processHooks(test, module) { + if (module.parentModule) { + processHooks(test, module.parentModule); + } + + if (module.hooks[handler].length) { + for (var i = 0; i < module.hooks[handler].length; i++) { + hooks.push(test.queueHook(module.hooks[handler][i], handler, module)); + } + } + } + + // Hooks are ignored on skipped tests + if (!this.skip) { + processHooks(this, this.module); + } + + return hooks; + }, + + + finish: function finish() { + config.current = this; + + // Release the test callback to ensure that anything referenced has been + // released to be garbage collected. + this.callback = undefined; + + if (this.steps.length) { + var stepsList = this.steps.join(", "); + this.pushFailure("Expected assert.verifySteps() to be called before end of test " + ("after using assert.step(). Unverified steps: " + stepsList), this.stack); + } + + if (config.requireExpects && this.expected === null) { + this.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) { + this.pushFailure("Expected " + this.expected + " assertions, but " + this.assertions.length + " were run", this.stack); + } else if (this.expected === null && !this.assertions.length) { + this.pushFailure("Expected at least one assertion, but none were run - call " + "expect(0) to accept zero assertions.", this.stack); + } + + var i, + module = this.module, + moduleName = module.name, + testName = this.testName, + skipped = !!this.skip, + todo = !!this.todo, + bad = 0, + storage = config.storage; + + this.runtime = now() - this.started; + + config.stats.all += this.assertions.length; + module.stats.all += this.assertions.length; + + for (i = 0; i < this.assertions.length; i++) { + if (!this.assertions[i].result) { + bad++; + config.stats.bad++; + module.stats.bad++; + } + } + + notifyTestsRan(module, skipped); + + // Store result when possible + if (storage) { + if (bad) { + storage.setItem("qunit-test-" + moduleName + "-" + testName, bad); + } else { + storage.removeItem("qunit-test-" + moduleName + "-" + testName); + } + } + + // After emitting the js-reporters event we cleanup the assertion data to + // avoid leaking it. It is not used by the legacy testDone callbacks. + emit("testEnd", this.testReport.end(true)); + this.testReport.slimAssertions(); + + return runLoggingCallbacks("testDone", { + name: testName, + module: moduleName, + skipped: skipped, + todo: todo, + failed: bad, + passed: this.assertions.length - bad, + total: this.assertions.length, + runtime: skipped ? 0 : this.runtime, + + // HTML Reporter use + assertions: this.assertions, + testId: this.testId, + + // Source of Test + source: this.stack + }).then(function () { + if (module.testsRun === numberOfTests(module)) { + var completedModules = [module]; + + // Check if the parent modules, iteratively, are done. If that the case, + // we emit the `suiteEnd` event and trigger `moduleDone` callback. + var parent = module.parentModule; + while (parent && parent.testsRun === numberOfTests(parent)) { + completedModules.push(parent); + parent = parent.parentModule; + } + + return completedModules.reduce(function (promiseChain, completedModule) { + return promiseChain.then(function () { + return logSuiteEnd(completedModule); + }); + }, Promise$1.resolve([])); + } + }).then(function () { + config.current = undefined; + }); + + function logSuiteEnd(module) { + + // Reset `module.hooks` to ensure that anything referenced in these hooks + // has been released to be garbage collected. + module.hooks = {}; + + emit("suiteEnd", module.suiteReport.end(true)); + return runLoggingCallbacks("moduleDone", { + name: module.name, + tests: module.tests, + failed: module.stats.bad, + passed: module.stats.all - module.stats.bad, + total: module.stats.all, + runtime: now() - module.stats.started + }); + } + }, + + preserveTestEnvironment: function preserveTestEnvironment() { + if (this.preserveEnvironment) { + this.module.testEnvironment = this.testEnvironment; + this.testEnvironment = extend({}, this.module.testEnvironment); + } + }, + + queue: function queue() { + var test = this; + + if (!this.valid()) { + return; + } + + function runTest() { + return [function () { + return test.before(); + }].concat(toConsumableArray(test.hooks("before")), [function () { + test.preserveTestEnvironment(); + }], toConsumableArray(test.hooks("beforeEach")), [function () { + test.run(); + }], toConsumableArray(test.hooks("afterEach").reverse()), toConsumableArray(test.hooks("after").reverse()), [function () { + test.after(); + }, function () { + return test.finish(); + }]); + } + + var previousFailCount = config.storage && +config.storage.getItem("qunit-test-" + this.module.name + "-" + this.testName); + + // Prioritize previously failed tests, detected from storage + var prioritize = config.reorder && !!previousFailCount; + + this.previousFailure = !!previousFailCount; + + ProcessingQueue.add(runTest, prioritize, config.seed); + + // If the queue has already finished, we manually process the new test + if (ProcessingQueue.finished) { + ProcessingQueue.advance(); + } + }, + + + pushResult: function pushResult(resultInfo) { + if (this !== config.current) { + throw new Error("Assertion occurred after test had finished."); + } + + // Destructure of resultInfo = { result, actual, expected, message, negative } + var source, + details = { + module: this.module.name, + name: this.testName, + result: resultInfo.result, + message: resultInfo.message, + actual: resultInfo.actual, + testId: this.testId, + negative: resultInfo.negative || false, + runtime: now() - this.started, + todo: !!this.todo + }; + + if (hasOwn.call(resultInfo, "expected")) { + details.expected = resultInfo.expected; + } + + if (!resultInfo.result) { + source = resultInfo.source || sourceFromStacktrace(); + + if (source) { + details.source = source; + } + } + + this.logAssertion(details); + + this.assertions.push({ + result: !!resultInfo.result, + message: resultInfo.message + }); + }, + + pushFailure: function pushFailure(message, source, actual) { + if (!(this instanceof Test)) { + throw new Error("pushFailure() assertion outside test context, was " + sourceFromStacktrace(2)); + } + + this.pushResult({ + result: false, + message: message || "error", + actual: actual || null, + source: source + }); + }, + + /** + * Log assertion details using both the old QUnit.log interface and + * QUnit.on( "assertion" ) interface. + * + * @private + */ + logAssertion: function logAssertion(details) { + runLoggingCallbacks("log", details); + + var assertion = { + passed: details.result, + actual: details.actual, + expected: details.expected, + message: details.message, + stack: details.source, + todo: details.todo + }; + this.testReport.pushAssertion(assertion); + emit("assertion", assertion); + }, + + + resolvePromise: function resolvePromise(promise, phase) { + var then, + resume, + message, + test = this; + if (promise != null) { + then = promise.then; + if (objectType(then) === "function") { + resume = internalStop(test); + if (config.notrycatch) { + then.call(promise, function () { + resume(); + }); + } else { + then.call(promise, function () { + resume(); + }, function (error) { + message = "Promise rejected " + (!phase ? "during" : phase.replace(/Each$/, "")) + " \"" + test.testName + "\": " + (error && error.message || error); + test.pushFailure(message, extractStacktrace(error, 0)); + + // Else next test will carry the responsibility + saveGlobal(); + + // Unblock + internalRecover(test); + }); + } + } + } + }, + + valid: function valid() { + var filter = config.filter, + regexFilter = /^(!?)\/([\w\W]*)\/(i?$)/.exec(filter), + module = config.module && config.module.toLowerCase(), + fullName = this.module.name + ": " + this.testName; + + function moduleChainNameMatch(testModule) { + var testModuleName = testModule.name ? testModule.name.toLowerCase() : null; + if (testModuleName === module) { + return true; + } else if (testModule.parentModule) { + return moduleChainNameMatch(testModule.parentModule); + } else { + return false; + } + } + + function moduleChainIdMatch(testModule) { + return inArray(testModule.moduleId, config.moduleId) || testModule.parentModule && moduleChainIdMatch(testModule.parentModule); + } + + // Internally-generated tests are always valid + if (this.callback && this.callback.validTest) { + return true; + } + + if (config.moduleId && config.moduleId.length > 0 && !moduleChainIdMatch(this.module)) { + + return false; + } + + if (config.testId && config.testId.length > 0 && !inArray(this.testId, config.testId)) { + + return false; + } + + if (module && !moduleChainNameMatch(this.module)) { + return false; + } + + if (!filter) { + return true; + } + + return regexFilter ? this.regexFilter(!!regexFilter[1], regexFilter[2], regexFilter[3], fullName) : this.stringFilter(filter, fullName); + }, + + regexFilter: function regexFilter(exclude, pattern, flags, fullName) { + var regex = new RegExp(pattern, flags); + var match = regex.test(fullName); + + return match !== exclude; + }, + + stringFilter: function stringFilter(filter, fullName) { + filter = filter.toLowerCase(); + fullName = fullName.toLowerCase(); + + var 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; + } + }; + + function pushFailure() { + if (!config.current) { + throw new Error("pushFailure() assertion outside test context, in " + sourceFromStacktrace(2)); + } + + // Gets current test obj + var currentTest = config.current; + + return currentTest.pushFailure.apply(currentTest, arguments); + } + + function saveGlobal() { + config.pollution = []; + + if (config.noglobals) { + for (var key in global$1) { + if (hasOwn.call(global$1, 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) { + pushFailure("Introduced global variable(s): " + newGlobals.join(", ")); + } + + deletedGlobals = diff(old, config.pollution); + if (deletedGlobals.length > 0) { + pushFailure("Deleted global variable(s): " + deletedGlobals.join(", ")); + } + } + + // Will be exposed as QUnit.test + function test(testName, callback) { + if (focused$1) { + return; + } + + var newTest = new Test({ + testName: testName, + callback: callback + }); + + newTest.queue(); + } + + function todo(testName, callback) { + if (focused$1) { + return; + } + + var newTest = new Test({ + testName: testName, + callback: callback, + todo: true + }); + + newTest.queue(); + } + + // Will be exposed as QUnit.skip + function skip(testName) { + if (focused$1) { + return; + } + + var test = new Test({ + testName: testName, + skip: true + }); + + test.queue(); + } + + // Will be exposed as QUnit.only + function only(testName, callback) { + if (focused$1) { + return; + } + + config.queue.length = 0; + focused$1 = true; + + var newTest = new Test({ + testName: testName, + callback: callback + }); + + newTest.queue(); + } + + // Put a hold on processing and return a function that will release it. + function internalStop(test) { + test.semaphore += 1; + config.blocking = true; + + // Set a recovery timeout, if so configured. + if (defined.setTimeout) { + var timeoutDuration = void 0; + + if (typeof test.timeout === "number") { + timeoutDuration = test.timeout; + } else if (typeof config.testTimeout === "number") { + timeoutDuration = config.testTimeout; + } + + if (typeof timeoutDuration === "number" && timeoutDuration > 0) { + clearTimeout(config.timeout); + config.timeout = setTimeout$1(function () { + pushFailure("Test took longer than " + timeoutDuration + "ms; test timed out.", sourceFromStacktrace(2)); + internalRecover(test); + }, timeoutDuration); + } + } + + var released = false; + return function resume() { + if (released) { + return; + } + + released = true; + test.semaphore -= 1; + internalStart(test); + }; + } + + // Forcefully release all processing holds. + function internalRecover(test) { + test.semaphore = 0; + internalStart(test); + } + + // Release a processing hold, scheduling a resumption attempt if no holds remain. + function internalStart(test) { + + // If semaphore is non-numeric, throw error + if (isNaN(test.semaphore)) { + test.semaphore = 0; + + pushFailure("Invalid value on test.semaphore", sourceFromStacktrace(2)); + return; + } + + // Don't start until equal number of stop-calls + if (test.semaphore > 0) { + return; + } + + // Throw an Error if start is called more often than stop + if (test.semaphore < 0) { + test.semaphore = 0; + + pushFailure("Tried to restart test while already started (test's semaphore was 0 already)", sourceFromStacktrace(2)); + return; + } + + // Add a slight delay to allow more assertions etc. + if (defined.setTimeout) { + if (config.timeout) { + clearTimeout(config.timeout); + } + config.timeout = setTimeout$1(function () { + if (test.semaphore > 0) { + return; + } + + if (config.timeout) { + clearTimeout(config.timeout); + } + + begin(); + }); + } else { + begin(); + } + } + + function collectTests(module) { + var tests = [].concat(module.tests); + var modules = [].concat(toConsumableArray(module.childModules)); + + // Do a breadth-first traversal of the child modules + while (modules.length) { + var nextModule = modules.shift(); + tests.push.apply(tests, nextModule.tests); + modules.push.apply(modules, toConsumableArray(nextModule.childModules)); + } + + return tests; + } + + function numberOfTests(module) { + return collectTests(module).length; + } + + function numberOfUnskippedTests(module) { + return collectTests(module).filter(function (test) { + return !test.skip; + }).length; + } + + function notifyTestsRan(module, skipped) { + module.testsRun++; + if (!skipped) { + module.unskippedTestsRun++; + } + while (module = module.parentModule) { + module.testsRun++; + if (!skipped) { + module.unskippedTestsRun++; + } + } + } + + var Assert = function () { + function Assert(testContext) { + classCallCheck(this, Assert); + + this.test = testContext; + } + + // Assert helpers + + createClass(Assert, [{ + key: "timeout", + value: function timeout(duration) { + if (typeof duration !== "number") { + throw new Error("You must pass a number as the duration to assert.timeout"); + } + + this.test.timeout = duration; + } + + // Documents a "step", which is a string value, in a test as a passing assertion + + }, { + key: "step", + value: function step(message) { + var assertionMessage = message; + var result = !!message; + + this.test.steps.push(message); + + if (objectType(message) === "undefined" || message === "") { + assertionMessage = "You must provide a message to assert.step"; + } else if (objectType(message) !== "string") { + assertionMessage = "You must provide a string value to assert.step"; + result = false; + } + + this.pushResult({ + result: result, + message: assertionMessage + }); + } + + // Verifies the steps in a test match a given array of string values + + }, { + key: "verifySteps", + value: function verifySteps(steps, message) { + + // Since the steps array is just string values, we can clone with slice + var actualStepsClone = this.test.steps.slice(); + this.deepEqual(actualStepsClone, steps, message); + this.test.steps.length = 0; + } + + // Specify the number of expected assertions to guarantee that failed test + // (no assertions are run at all) don't slip through. + + }, { + key: "expect", + value: function expect(asserts) { + if (arguments.length === 1) { + this.test.expected = asserts; + } else { + return this.test.expected; + } + } + + // Put a hold on processing and return a function that will release it a maximum of once. + + }, { + key: "async", + value: function async(count) { + var test$$1 = this.test; + + var popped = false, + acceptCallCount = count; + + if (typeof acceptCallCount === "undefined") { + acceptCallCount = 1; + } + + var resume = internalStop(test$$1); + + return function done() { + if (config.current !== test$$1) { + throw Error("assert.async callback called after test finished."); + } + + if (popped) { + test$$1.pushFailure("Too many calls to the `assert.async` callback", sourceFromStacktrace(2)); + return; + } + + acceptCallCount -= 1; + if (acceptCallCount > 0) { + return; + } + + popped = true; + resume(); + }; + } + + // Exports test.push() to the user API + // Alias of pushResult. + + }, { + key: "push", + value: function push(result, actual, expected, message, negative) { + Logger.warn("assert.push is deprecated and will be removed in QUnit 3.0." + " Please use assert.pushResult instead (https://api.qunitjs.com/assert/pushResult)."); + + var currentAssert = this instanceof Assert ? this : config.current.assert; + return currentAssert.pushResult({ + result: result, + actual: actual, + expected: expected, + message: message, + negative: negative + }); + } + }, { + key: "pushResult", + value: function pushResult(resultInfo) { + + // Destructure of resultInfo = { result, actual, expected, message, negative } + var assert = this; + var currentTest = assert instanceof Assert && assert.test || config.current; + + // Backwards compatibility fix. + // Allows the direct use of global exported assertions and QUnit.assert.* + // Although, it's use is not recommended as it can leak assertions + // to other tests from async tests, because we only get a reference to the current test, + // not exactly the test where assertion were intended to be called. + if (!currentTest) { + throw new Error("assertion outside test context, in " + sourceFromStacktrace(2)); + } + + if (!(assert instanceof Assert)) { + assert = currentTest.assert; + } + + return assert.test.pushResult(resultInfo); + } + }, { + key: "ok", + value: function ok(result, message) { + if (!message) { + message = result ? "okay" : "failed, expected argument to be truthy, was: " + dump.parse(result); + } + + this.pushResult({ + result: !!result, + actual: result, + expected: true, + message: message + }); + } + }, { + key: "notOk", + value: function notOk(result, message) { + if (!message) { + message = !result ? "okay" : "failed, expected argument to be falsy, was: " + dump.parse(result); + } + + this.pushResult({ + result: !result, + actual: result, + expected: false, + message: message + }); + } + }, { + key: "equal", + value: function equal(actual, expected, message) { + + // eslint-disable-next-line eqeqeq + var result = expected == actual; + + this.pushResult({ + result: result, + actual: actual, + expected: expected, + message: message + }); + } + }, { + key: "notEqual", + value: function notEqual(actual, expected, message) { + + // eslint-disable-next-line eqeqeq + var result = expected != actual; + + this.pushResult({ + result: result, + actual: actual, + expected: expected, + message: message, + negative: true + }); + } + }, { + key: "propEqual", + value: function propEqual(actual, expected, message) { + actual = objectValues(actual); + expected = objectValues(expected); + + this.pushResult({ + result: equiv(actual, expected), + actual: actual, + expected: expected, + message: message + }); + } + }, { + key: "notPropEqual", + value: function notPropEqual(actual, expected, message) { + actual = objectValues(actual); + expected = objectValues(expected); + + this.pushResult({ + result: !equiv(actual, expected), + actual: actual, + expected: expected, + message: message, + negative: true + }); + } + }, { + key: "deepEqual", + value: function deepEqual(actual, expected, message) { + this.pushResult({ + result: equiv(actual, expected), + actual: actual, + expected: expected, + message: message + }); + } + }, { + key: "notDeepEqual", + value: function notDeepEqual(actual, expected, message) { + this.pushResult({ + result: !equiv(actual, expected), + actual: actual, + expected: expected, + message: message, + negative: true + }); + } + }, { + key: "strictEqual", + value: function strictEqual(actual, expected, message) { + this.pushResult({ + result: expected === actual, + actual: actual, + expected: expected, + message: message + }); + } + }, { + key: "notStrictEqual", + value: function notStrictEqual(actual, expected, message) { + this.pushResult({ + result: expected !== actual, + actual: actual, + expected: expected, + message: message, + negative: true + }); + } + }, { + key: "throws", + value: function throws(block, expected, message) { + var actual = void 0, + result = false; + + var currentTest = this instanceof Assert && this.test || config.current; + + // 'expected' is optional unless doing string comparison + if (objectType(expected) === "string") { + if (message == null) { + message = expected; + expected = null; + } else { + throw new Error("throws/raises does not accept a string value for the expected argument.\n" + "Use a non-string object value (e.g. regExp) instead if it's necessary."); + } + } + + currentTest.ignoreGlobalErrors = true; + try { + block.call(currentTest.testEnvironment); + } catch (e) { + actual = e; + } + currentTest.ignoreGlobalErrors = false; + + if (actual) { + var expectedType = objectType(expected); + + // We don't want to validate thrown error + if (!expected) { + result = true; + expected = null; + + // Expected is a regexp + } else if (expectedType === "regexp") { + result = expected.test(errorString(actual)); + + // Expected is a constructor, maybe an Error constructor + } else if (expectedType === "function" && actual instanceof expected) { + result = true; + + // Expected is an Error object + } else if (expectedType === "object") { + result = actual instanceof expected.constructor && actual.name === expected.name && actual.message === expected.message; + + // Expected is a validation function which returns true if validation passed + } else if (expectedType === "function" && expected.call({}, actual) === true) { + expected = null; + result = true; + } + } + + currentTest.assert.pushResult({ + result: result, + actual: actual, + expected: expected, + message: message + }); + } + }, { + key: "rejects", + value: function rejects(promise, expected, message) { + var result = false; + + var currentTest = this instanceof Assert && this.test || config.current; + + // 'expected' is optional unless doing string comparison + if (objectType(expected) === "string") { + if (message === undefined) { + message = expected; + expected = undefined; + } else { + message = "assert.rejects does not accept a string value for the expected " + "argument.\nUse a non-string object value (e.g. validator function) instead " + "if necessary."; + + currentTest.assert.pushResult({ + result: false, + message: message + }); + + return; + } + } + + var then = promise && promise.then; + if (objectType(then) !== "function") { + var _message = "The value provided to `assert.rejects` in " + "\"" + currentTest.testName + "\" was not a promise."; + + currentTest.assert.pushResult({ + result: false, + message: _message, + actual: promise + }); + + return; + } + + var done = this.async(); + + return then.call(promise, function handleFulfillment() { + var message = "The promise returned by the `assert.rejects` callback in " + "\"" + currentTest.testName + "\" did not reject."; + + currentTest.assert.pushResult({ + result: false, + message: message, + actual: promise + }); + + done(); + }, function handleRejection(actual) { + var expectedType = objectType(expected); + + // We don't want to validate + if (expected === undefined) { + result = true; + expected = actual; + + // Expected is a regexp + } else if (expectedType === "regexp") { + result = expected.test(errorString(actual)); + + // Expected is a constructor, maybe an Error constructor + } else if (expectedType === "function" && actual instanceof expected) { + result = true; + + // Expected is an Error object + } else if (expectedType === "object") { + result = actual instanceof expected.constructor && actual.name === expected.name && actual.message === expected.message; + + // Expected is a validation function which returns true if validation passed + } else { + if (expectedType === "function") { + result = expected.call({}, actual) === true; + expected = null; + + // Expected is some other invalid type + } else { + result = false; + message = "invalid expected value provided to `assert.rejects` " + "callback in \"" + currentTest.testName + "\": " + expectedType + "."; + } + } + + currentTest.assert.pushResult({ + result: result, + actual: actual, + expected: expected, + message: message + }); + + done(); + }); + } + }]); + return Assert; + }(); + + // Provide an alternative to assert.throws(), for environments that consider throws a reserved word + // Known to us are: Closure Compiler, Narwhal + // eslint-disable-next-line dot-notation + + + Assert.prototype.raises = Assert.prototype["throws"]; + + /** + * Converts an error into a simple string for comparisons. + * + * @param {Error} error + * @return {String} + */ + function errorString(error) { + var resultErrorString = error.toString(); + + if (resultErrorString.substring(0, 7) === "[object") { + var name = error.name ? error.name.toString() : "Error"; + var 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 resultErrorString; + } + } + + /* global module, exports, define */ + function exportQUnit(QUnit) { + + if (defined.document) { + + // QUnit may be defined when it is preconfigured but then only QUnit and QUnit.config may be defined. + if (window$1.QUnit && window$1.QUnit.version) { + throw new Error("QUnit has already been defined."); + } + + window$1.QUnit = QUnit; + } + + // For nodejs + if (typeof module !== "undefined" && module && module.exports) { + module.exports = QUnit; + + // For consistency with CommonJS environments' exports + module.exports.QUnit = QUnit; + } + + // For CommonJS with exports, but without module.exports, like Rhino + if (typeof exports !== "undefined" && exports) { + exports.QUnit = QUnit; + } + + if (typeof define === "function" && define.amd) { + define(function () { + return QUnit; + }); + QUnit.config.autostart = false; + } + + // For Web/Service Workers + if (self$1 && self$1.WorkerGlobalScope && self$1 instanceof self$1.WorkerGlobalScope) { + self$1.QUnit = QUnit; + } + } + + // Handle an unhandled exception. By convention, returns true if further + // error handling should be suppressed and false otherwise. + // In this case, we will only suppress further error handling if the + // "ignoreGlobalErrors" configuration option is enabled. + function onError(error) { + for (var _len = arguments.length, args = Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { + args[_key - 1] = arguments[_key]; + } + + if (config.current) { + if (config.current.ignoreGlobalErrors) { + return true; + } + pushFailure.apply(undefined, [error.message, error.stacktrace || error.fileName + ":" + error.lineNumber].concat(args)); + } else { + test("global failure", extend(function () { + pushFailure.apply(undefined, [error.message, error.stacktrace || error.fileName + ":" + error.lineNumber].concat(args)); + }, { validTest: true })); + } + + return false; + } + + // Handle an unhandled rejection + function onUnhandledRejection(reason) { + var resultInfo = { + result: false, + message: reason.message || "error", + actual: reason, + source: reason.stack || sourceFromStacktrace(3) + }; + + var currentTest = config.current; + if (currentTest) { + currentTest.assert.pushResult(resultInfo); + } else { + test("global failure", extend(function (assert) { + assert.pushResult(resultInfo); + }, { validTest: true })); + } + } + + var QUnit = {}; + var globalSuite = new SuiteReport(); + + // The initial "currentModule" represents the global (or top-level) module that + // is not explicitly defined by the user, therefore we add the "globalSuite" to + // it since each module has a suiteReport associated with it. + config.currentModule.suiteReport = globalSuite; + + var globalStartCalled = false; + var runStarted = false; + + // Figure out if we're running the tests from a server or not + QUnit.isLocal = !(defined.document && window$1.location.protocol !== "file:"); + + // Expose the current QUnit version + QUnit.version = "2.8.0"; + + extend(QUnit, { + on: on, + + module: module$1, + + test: test, + + todo: todo, + + skip: skip, + + only: only, + + start: function start(count) { + var globalStartAlreadyCalled = globalStartCalled; + + if (!config.current) { + globalStartCalled = true; + + if (runStarted) { + throw new Error("Called start() while test already started running"); + } else if (globalStartAlreadyCalled || count > 1) { + throw new Error("Called start() outside of a test context too many times"); + } else if (config.autostart) { + throw new Error("Called start() outside of a test context when " + "QUnit.config.autostart was true"); + } else if (!config.pageLoaded) { + + // The page isn't completely loaded yet, so we set autostart and then + // load if we're in Node or wait for the browser's load event. + config.autostart = true; + + // Starts from Node even if .load was not previously called. We still return + // early otherwise we'll wind up "beginning" twice. + if (!defined.document) { + QUnit.load(); + } + + return; + } + } else { + throw new Error("QUnit.start cannot be called inside a test context."); + } + + scheduleBegin(); + }, + + config: config, + + is: is, + + objectType: objectType, + + extend: extend, + + load: function load() { + config.pageLoaded = true; + + // Initialize the configuration options + extend(config, { + stats: { all: 0, bad: 0 }, + started: 0, + updateRate: 1000, + autostart: true, + filter: "" + }, true); + + if (!runStarted) { + config.blocking = false; + + if (config.autostart) { + scheduleBegin(); + } + } + }, + + stack: function stack(offset) { + offset = (offset || 0) + 2; + return sourceFromStacktrace(offset); + }, + + onError: onError, + + onUnhandledRejection: onUnhandledRejection + }); + + QUnit.pushFailure = pushFailure; + QUnit.assert = Assert.prototype; + QUnit.equiv = equiv; + QUnit.dump = dump; + + registerLoggingCallbacks(QUnit); + + function scheduleBegin() { + + runStarted = true; + + // Add a slight delay to allow definition of more modules and tests. + if (defined.setTimeout) { + setTimeout$1(function () { + begin(); + }); + } else { + begin(); + } + } + + function unblockAndAdvanceQueue() { + config.blocking = false; + ProcessingQueue.advance(); + } + + function begin() { + var i, + l, + modulesLog = []; + + // If the test run hasn't officially begun yet + if (!config.started) { + + // Record the time of the test run's beginning + config.started = now(); + + // Delete the loose unnamed module if unused. + if (config.modules[0].name === "" && config.modules[0].tests.length === 0) { + config.modules.shift(); + } + + // Avoid unnecessary information by not logging modules' test environments + for (i = 0, l = config.modules.length; i < l; i++) { + modulesLog.push({ + name: config.modules[i].name, + tests: config.modules[i].tests + }); + } + + // The test run is officially beginning now + emit("runStart", globalSuite.start(true)); + runLoggingCallbacks("begin", { + totalTests: Test.count, + modules: modulesLog + }).then(unblockAndAdvanceQueue); + } else { + unblockAndAdvanceQueue(); + } + } + + exportQUnit(QUnit); + + (function () { + + if (typeof window$1 === "undefined" || typeof document$1 === "undefined") { + return; + } + + var config = QUnit.config, + hasOwn = Object.prototype.hasOwnProperty; + + // Stores fixture HTML for resetting later + function storeFixture() { + + // Avoid overwriting user-defined values + if (hasOwn.call(config, "fixture")) { + return; + } + + var fixture = document$1.getElementById("qunit-fixture"); + if (fixture) { + config.fixture = fixture.cloneNode(true); + } + } + + QUnit.begin(storeFixture); + + // Resets the fixture DOM element if available. + function resetFixture() { + if (config.fixture == null) { + return; + } + + var fixture = document$1.getElementById("qunit-fixture"); + var resetFixtureType = _typeof(config.fixture); + if (resetFixtureType === "string") { + + // support user defined values for `config.fixture` + var newFixture = document$1.createElement("div"); + newFixture.setAttribute("id", "qunit-fixture"); + newFixture.innerHTML = config.fixture; + fixture.parentNode.replaceChild(newFixture, fixture); + } else { + var clonedFixture = config.fixture.cloneNode(true); + fixture.parentNode.replaceChild(clonedFixture, fixture); + } + } + + QUnit.testStart(resetFixture); + })(); + + (function () { + + // Only interact with URLs via window.location + var location = typeof window$1 !== "undefined" && window$1.location; + if (!location) { + return; + } + + var urlParams = getUrlParams(); + + QUnit.urlParams = urlParams; + + // Match module/test by inclusion in an array + QUnit.config.moduleId = [].concat(urlParams.moduleId || []); + QUnit.config.testId = [].concat(urlParams.testId || []); + + // Exact case-insensitive match of the module name + QUnit.config.module = urlParams.module; + + // Regular expression or case-insenstive substring match against "moduleName: testName" + QUnit.config.filter = urlParams.filter; + + // Test order randomization + if (urlParams.seed === true) { + + // Generate a random seed if the option is specified without a value + QUnit.config.seed = Math.random().toString(36).slice(2); + } else if (urlParams.seed) { + QUnit.config.seed = urlParams.seed; + } + + // Add URL-parameter-mapped config values with UI form rendering data + QUnit.config.urlConfig.push({ + id: "hidepassed", + label: "Hide passed tests", + tooltip: "Only show tests and assertions that fail. Stored as query-strings." + }, { + id: "noglobals", + label: "Check for Globals", + tooltip: "Enabling this will test if any test introduces new properties on the " + "global object (`window` in Browsers). 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." + }); + + QUnit.begin(function () { + var i, + option, + urlConfig = QUnit.config.urlConfig; + + for (i = 0; i < urlConfig.length; i++) { + + // Options can be either strings or objects with nonempty "id" properties + option = QUnit.config.urlConfig[i]; + if (typeof option !== "string") { + option = option.id; + } + + if (QUnit.config[option] === undefined) { + QUnit.config[option] = urlParams[option]; + } + } + }); + + function getUrlParams() { + var i, param, name, value; + var urlParams = Object.create(null); + var params = location.search.slice(1).split("&"); + var length = params.length; + + for (i = 0; i < length; i++) { + if (params[i]) { + param = params[i].split("="); + name = decodeQueryParam(param[0]); + + // Allow just a key to turn on a flag, e.g., test.html?noglobals + value = param.length === 1 || decodeQueryParam(param.slice(1).join("=")); + if (name in urlParams) { + urlParams[name] = [].concat(urlParams[name], value); + } else { + urlParams[name] = value; + } + } + } + + return urlParams; + } + + function decodeQueryParam(param) { + return decodeURIComponent(param.replace(/\+/g, "%20")); + } + })(); + + var stats = { + passedTests: 0, + failedTests: 0, + skippedTests: 0, + todoTests: 0 + }; + + // 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 () { + + // Don't load the HTML Reporter on non-browser environments + if (typeof window$1 === "undefined" || !window$1.document) { + return; + } + + var config = QUnit.config, + hiddenTests = [], + document = window$1.document, + collapseNext = false, + hasOwn = Object.prototype.hasOwnProperty, + unfilteredUrl = setUrl({ filter: undefined, module: undefined, + moduleId: undefined, testId: undefined }), + modulesList = []; + + function addEvent(elem, type, fn) { + elem.addEventListener(type, fn, false); + } + + function removeEvent(elem, type, fn) { + elem.removeEventListener(type, fn, false); + } + + 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 + " ") >= 0; + } + + function addClass(elem, name) { + if (!hasClass(elem, name)) { + elem.className += (elem.className ? " " : "") + name; + } + } + + function toggleClass(elem, name, force) { + if (force || typeof force === "undefined" && !hasClass(elem, name)) { + addClass(elem, name); + } else { + removeClass(elem, name); + } + } + + function removeClass(elem, name) { + var set = " " + elem.className + " "; + + // Class name may appear multiple times + while (set.indexOf(" " + name + " ") >= 0) { + set = set.replace(" " + name + " ", " "); + } + + // Trim for prettiness + elem.className = typeof set.trim === "function" ? set.trim() : set.replace(/^\s+|\s+$/g, ""); + } + + function id(name) { + return document.getElementById && document.getElementById(name); + } + + function abortTests() { + var abortButton = id("qunit-abort-tests-button"); + if (abortButton) { + abortButton.disabled = true; + abortButton.innerHTML = "Aborting..."; + } + QUnit.config.queue.length = 0; + return false; + } + + function interceptNavigation(ev) { + applyUrlParams(); + + if (ev && ev.preventDefault) { + ev.preventDefault(); + } + + return false; + } + + function getUrlConfigHtml() { + var i, + j, + val, + escaped, + escapedTooltip, + selection = false, + urlConfig = config.urlConfig, + urlConfigHtml = ""; + + for (i = 0; i < urlConfig.length; i++) { + + // Options can be either strings or objects with nonempty "id" properties + val = config.urlConfig[i]; + if (typeof val === "string") { + val = { + id: val, + label: val + }; + } + + escaped = escapeText(val.id); + escapedTooltip = escapeText(val.tooltip); + + if (!val.value || typeof val.value === "string") { + urlConfigHtml += ""; + } else { + urlConfigHtml += ""; + } + } + + return urlConfigHtml; + } + + // Handle "click" events on toolbar checkboxes and "change" for select menus. + // Updates the URL with the new state of `config.urlConfig` values. + function toolbarChanged() { + var updatedUrl, + value, + tests, + field = this, + params = {}; + + // Detect if field is a select menu or a checkbox + if ("selectedIndex" in field) { + value = field.options[field.selectedIndex].value || undefined; + } else { + value = field.checked ? field.defaultValue || true : undefined; + } + + params[field.name] = value; + updatedUrl = setUrl(params); + + // Check if we can apply the change without a page refresh + if ("hidepassed" === field.name && "replaceState" in window$1.history) { + QUnit.urlParams[field.name] = value; + config[field.name] = value || false; + tests = id("qunit-tests"); + if (tests) { + var length = tests.children.length; + var children = tests.children; + + if (field.checked) { + for (var i = 0; i < length; i++) { + var test = children[i]; + + if (test && test.className.indexOf("pass") > -1) { + hiddenTests.push(test); + } + } + + var _iteratorNormalCompletion = true; + var _didIteratorError = false; + var _iteratorError = undefined; + + try { + for (var _iterator = hiddenTests[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { + var hiddenTest = _step.value; + + tests.removeChild(hiddenTest); + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (!_iteratorNormalCompletion && _iterator.return) { + _iterator.return(); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + } else { + while ((test = hiddenTests.pop()) != null) { + tests.appendChild(test); + } + } + } + window$1.history.replaceState(null, "", updatedUrl); + } else { + window$1.location = updatedUrl; + } + } + + function setUrl(params) { + var key, + arrValue, + i, + querystring = "?", + location = window$1.location; + + params = QUnit.extend(QUnit.extend({}, QUnit.urlParams), params); + + for (key in params) { + + // Skip inherited or undefined properties + if (hasOwn.call(params, key) && params[key] !== undefined) { + + // Output a parameter for each value of this key + // (but usually just one) + arrValue = [].concat(params[key]); + for (i = 0; i < arrValue.length; i++) { + querystring += encodeURIComponent(key); + if (arrValue[i] !== true) { + querystring += "=" + encodeURIComponent(arrValue[i]); + } + querystring += "&"; + } + } + } + return location.protocol + "//" + location.host + location.pathname + querystring.slice(0, -1); + } + + function applyUrlParams() { + var i, + selectedModules = [], + modulesList = id("qunit-modulefilter-dropdown-list").getElementsByTagName("input"), + filter = id("qunit-filter-input").value; + + for (i = 0; i < modulesList.length; i++) { + if (modulesList[i].checked) { + selectedModules.push(modulesList[i].value); + } + } + + window$1.location = setUrl({ + filter: filter === "" ? undefined : filter, + moduleId: selectedModules.length === 0 ? undefined : selectedModules, + + // Remove module and testId filter + module: undefined, + testId: undefined + }); + } + + function toolbarUrlConfigContainer() { + var urlConfigContainer = document.createElement("span"); + + urlConfigContainer.innerHTML = getUrlConfigHtml(); + addClass(urlConfigContainer, "qunit-url-config"); + + addEvents(urlConfigContainer.getElementsByTagName("input"), "change", toolbarChanged); + addEvents(urlConfigContainer.getElementsByTagName("select"), "change", toolbarChanged); + + return urlConfigContainer; + } + + function abortTestsButton() { + var button = document.createElement("button"); + button.id = "qunit-abort-tests-button"; + button.innerHTML = "Abort"; + addEvent(button, "click", abortTests); + return button; + } + + function toolbarLooseFilter() { + var filter = document.createElement("form"), + label = document.createElement("label"), + input = document.createElement("input"), + button = document.createElement("button"); + + addClass(filter, "qunit-filter"); + + label.innerHTML = "Filter: "; + + input.type = "text"; + input.value = config.filter || ""; + input.name = "filter"; + input.id = "qunit-filter-input"; + + button.innerHTML = "Go"; + + label.appendChild(input); + + filter.appendChild(label); + filter.appendChild(document.createTextNode(" ")); + filter.appendChild(button); + addEvent(filter, "submit", interceptNavigation); + + return filter; + } + + function moduleListHtml() { + var i, + checked, + html = ""; + + for (i = 0; i < config.modules.length; i++) { + if (config.modules[i].name !== "") { + checked = config.moduleId.indexOf(config.modules[i].moduleId) > -1; + html += "
    1. "; + } + } + + return html; + } + + function toolbarModuleFilter() { + var allCheckbox, + commit, + reset, + moduleFilter = document.createElement("form"), + label = document.createElement("label"), + moduleSearch = document.createElement("input"), + dropDown = document.createElement("div"), + actions = document.createElement("span"), + dropDownList = document.createElement("ul"), + dirty = false; + + moduleSearch.id = "qunit-modulefilter-search"; + moduleSearch.autocomplete = "off"; + addEvent(moduleSearch, "input", searchInput); + addEvent(moduleSearch, "input", searchFocus); + addEvent(moduleSearch, "focus", searchFocus); + addEvent(moduleSearch, "click", searchFocus); + + label.id = "qunit-modulefilter-search-container"; + label.innerHTML = "Module: "; + label.appendChild(moduleSearch); + + actions.id = "qunit-modulefilter-actions"; + actions.innerHTML = "" + "" + ""; + allCheckbox = actions.lastChild.firstChild; + commit = actions.firstChild; + reset = commit.nextSibling; + addEvent(commit, "click", applyUrlParams); + + dropDownList.id = "qunit-modulefilter-dropdown-list"; + dropDownList.innerHTML = moduleListHtml(); + + dropDown.id = "qunit-modulefilter-dropdown"; + dropDown.style.display = "none"; + dropDown.appendChild(actions); + dropDown.appendChild(dropDownList); + addEvent(dropDown, "change", selectionChange); + selectionChange(); + + moduleFilter.id = "qunit-modulefilter"; + moduleFilter.appendChild(label); + moduleFilter.appendChild(dropDown); + addEvent(moduleFilter, "submit", interceptNavigation); + addEvent(moduleFilter, "reset", function () { + + // Let the reset happen, then update styles + window$1.setTimeout(selectionChange); + }); + + // Enables show/hide for the dropdown + function searchFocus() { + if (dropDown.style.display !== "none") { + return; + } + + dropDown.style.display = "block"; + addEvent(document, "click", hideHandler); + addEvent(document, "keydown", hideHandler); + + // Hide on Escape keydown or outside-container click + function hideHandler(e) { + var inContainer = moduleFilter.contains(e.target); + + if (e.keyCode === 27 || !inContainer) { + if (e.keyCode === 27 && inContainer) { + moduleSearch.focus(); + } + dropDown.style.display = "none"; + removeEvent(document, "click", hideHandler); + removeEvent(document, "keydown", hideHandler); + moduleSearch.value = ""; + searchInput(); + } + } + } + + // Processes module search box input + function searchInput() { + var i, + item, + searchText = moduleSearch.value.toLowerCase(), + listItems = dropDownList.children; + + for (i = 0; i < listItems.length; i++) { + item = listItems[i]; + if (!searchText || item.textContent.toLowerCase().indexOf(searchText) > -1) { + item.style.display = ""; + } else { + item.style.display = "none"; + } + } + } + + // Processes selection changes + function selectionChange(evt) { + var i, + item, + checkbox = evt && evt.target || allCheckbox, + modulesList = dropDownList.getElementsByTagName("input"), + selectedNames = []; + + toggleClass(checkbox.parentNode, "checked", checkbox.checked); + + dirty = false; + if (checkbox.checked && checkbox !== allCheckbox) { + allCheckbox.checked = false; + removeClass(allCheckbox.parentNode, "checked"); + } + for (i = 0; i < modulesList.length; i++) { + item = modulesList[i]; + if (!evt) { + toggleClass(item.parentNode, "checked", item.checked); + } else if (checkbox === allCheckbox && checkbox.checked) { + item.checked = false; + removeClass(item.parentNode, "checked"); + } + dirty = dirty || item.checked !== item.defaultChecked; + if (item.checked) { + selectedNames.push(item.parentNode.textContent); + } + } + + commit.style.display = reset.style.display = dirty ? "" : "none"; + moduleSearch.placeholder = selectedNames.join(", ") || allCheckbox.parentNode.textContent; + moduleSearch.title = "Type to filter list. Current selection:\n" + (selectedNames.join("\n") || allCheckbox.parentNode.textContent); + } + + return moduleFilter; + } + + function appendToolbar() { + var toolbar = id("qunit-testrunner-toolbar"); + + if (toolbar) { + toolbar.appendChild(toolbarUrlConfigContainer()); + toolbar.appendChild(toolbarModuleFilter()); + toolbar.appendChild(toolbarLooseFilter()); + toolbar.appendChild(document.createElement("div")).className = "clearfix"; + } + } + + function appendHeader() { + var header = id("qunit-header"); + + if (header) { + header.innerHTML = "" + header.innerHTML + " "; + } + } + + function appendBanner() { + var banner = id("qunit-banner"); + + if (banner) { + banner.className = ""; + } + } + + function appendTestResults() { + var tests = id("qunit-tests"), + result = id("qunit-testresult"), + controls; + + if (result) { + result.parentNode.removeChild(result); + } + + if (tests) { + tests.innerHTML = ""; + result = document.createElement("p"); + result.id = "qunit-testresult"; + result.className = "result"; + tests.parentNode.insertBefore(result, tests); + result.innerHTML = "
      Running...
       
      " + "
      " + "
      "; + controls = id("qunit-testresult-controls"); + } + + if (controls) { + controls.appendChild(abortTestsButton()); + } + } + + function appendFilteredTest() { + var testId = QUnit.config.testId; + if (!testId || testId.length <= 0) { + return ""; + } + return "
      Rerunning selected tests: " + escapeText(testId.join(", ")) + " Run all tests
      "; + } + + function appendUserAgent() { + var userAgent = id("qunit-userAgent"); + + if (userAgent) { + userAgent.innerHTML = ""; + userAgent.appendChild(document.createTextNode("QUnit " + QUnit.version + "; " + navigator.userAgent)); + } + } + + function appendInterface() { + var qunit = id("qunit"); + + if (qunit) { + qunit.innerHTML = "

      " + escapeText(document.title) + "

      " + "

      " + "
      " + appendFilteredTest() + "

      " + "
        "; + } + + appendHeader(); + appendBanner(); + appendTestResults(); + appendUserAgent(); + appendToolbar(); + } + + function appendTest(name, testId, moduleName) { + var title, + rerunTrigger, + testBlock, + assertList, + tests = id("qunit-tests"); + + if (!tests) { + return; + } + + title = document.createElement("strong"); + title.innerHTML = getNameHtml(name, moduleName); + + rerunTrigger = document.createElement("a"); + rerunTrigger.innerHTML = "Rerun"; + rerunTrigger.href = setUrl({ testId: testId }); + + testBlock = document.createElement("li"); + testBlock.appendChild(title); + testBlock.appendChild(rerunTrigger); + testBlock.id = "qunit-test-output-" + testId; + + assertList = document.createElement("ol"); + assertList.className = "qunit-assert-list"; + + testBlock.appendChild(assertList); + + tests.appendChild(testBlock); + } + + // HTML Reporter initialization and load + QUnit.begin(function (details) { + var i, moduleObj; + + // Sort modules by name for the picker + for (i = 0; i < details.modules.length; i++) { + moduleObj = details.modules[i]; + if (moduleObj.name) { + modulesList.push(moduleObj.name); + } + } + modulesList.sort(function (a, b) { + return a.localeCompare(b); + }); + + // Initialize QUnit elements + appendInterface(); + }); + + QUnit.done(function (details) { + var banner = id("qunit-banner"), + tests = id("qunit-tests"), + abortButton = id("qunit-abort-tests-button"), + totalTests = stats.passedTests + stats.skippedTests + stats.todoTests + stats.failedTests, + html = [totalTests, " tests completed in ", details.runtime, " milliseconds, with ", stats.failedTests, " failed, ", stats.skippedTests, " skipped, and ", stats.todoTests, " todo.
        ", "", details.passed, " assertions of ", details.total, " passed, ", details.failed, " failed."].join(""), + test, + assertLi, + assertList; + + // Update remaing tests to aborted + if (abortButton && abortButton.disabled) { + html = "Tests aborted after " + details.runtime + " milliseconds."; + + for (var i = 0; i < tests.children.length; i++) { + test = tests.children[i]; + if (test.className === "" || test.className === "running") { + test.className = "aborted"; + assertList = test.getElementsByTagName("ol")[0]; + assertLi = document.createElement("li"); + assertLi.className = "fail"; + assertLi.innerHTML = "Test aborted."; + assertList.appendChild(assertLi); + } + } + } + + if (banner && (!abortButton || abortButton.disabled === false)) { + banner.className = stats.failedTests ? "qunit-fail" : "qunit-pass"; + } + + if (abortButton) { + abortButton.parentNode.removeChild(abortButton); + } + + if (tests) { + id("qunit-testresult-display").innerHTML = html; + } + + if (config.altertitle && 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 = [stats.failedTests ? "\u2716" : "\u2714", document.title.replace(/^[\u2714\u2716] /i, "")].join(" "); + } + + // Scroll back to top to show results + if (config.scrolltop && window$1.scrollTo) { + window$1.scrollTo(0, 0); + } + }); + + function getNameHtml(name, module) { + var nameHtml = ""; + + if (module) { + nameHtml = "" + escapeText(module) + ": "; + } + + nameHtml += "" + escapeText(name) + ""; + + return nameHtml; + } + + QUnit.testStart(function (details) { + var running, bad; + + appendTest(details.name, details.testId, details.module); + + running = id("qunit-testresult-display"); + + if (running) { + addClass(running, "running"); + + bad = QUnit.config.reorder && details.previousFailure; + + running.innerHTML = [bad ? "Rerunning previously failed test:
        " : "Running:
        ", getNameHtml(details.name, details.module)].join(""); + } + }); + + function stripHtml(string) { + + // Strip tags, html entity and whitespaces + return string.replace(/<\/?[^>]+(>|$)/g, "").replace(/"/g, "").replace(/\s+/g, ""); + } + + QUnit.log(function (details) { + var assertList, + assertLi, + message, + expected, + actual, + diff, + showDiff = false, + testItem = id("qunit-test-output-" + details.testId); + + if (!testItem) { + return; + } + + message = escapeText(details.message) || (details.result ? "okay" : "failed"); + message = "" + message + ""; + message += "@ " + details.runtime + " ms"; + + // The pushFailure doesn't provide details.expected + // when it calls, it's implicit to also not show expected and diff stuff + // Also, we need to check details.expected existence, as it can exist and be undefined + if (!details.result && hasOwn.call(details, "expected")) { + if (details.negative) { + expected = "NOT " + QUnit.dump.parse(details.expected); + } else { + expected = QUnit.dump.parse(details.expected); + } + + actual = QUnit.dump.parse(details.actual); + message += ""; + + if (actual !== expected) { + + message += ""; + + if (typeof details.actual === "number" && typeof details.expected === "number") { + if (!isNaN(details.actual) && !isNaN(details.expected)) { + showDiff = true; + diff = details.actual - details.expected; + diff = (diff > 0 ? "+" : "") + diff; + } + } else if (typeof details.actual !== "boolean" && typeof details.expected !== "boolean") { + diff = QUnit.diff(expected, actual); + + // don't show diff if there is zero overlap + showDiff = stripHtml(diff).length !== stripHtml(expected).length + stripHtml(actual).length; + } + + if (showDiff) { + message += ""; + } + } else if (expected.indexOf("[object Array]") !== -1 || expected.indexOf("[object Object]") !== -1) { + message += ""; + } else { + message += ""; + } + + if (details.source) { + message += ""; + } + + message += "
        Expected:
        " + escapeText(expected) + "
        Result:
        " + escapeText(actual) + "
        Diff:
        " + diff + "
        Message: " + "Diff suppressed as the depth of object is more than current max depth (" + QUnit.config.maxDepth + ").

        Hint: Use QUnit.dump.maxDepth to " + " run with a higher max depth or " + "Rerun without max depth.

        Message: " + "Diff suppressed as the expected and actual results have an equivalent" + " serialization
        Source:
        " + escapeText(details.source) + "
        "; + + // This occurs when pushFailure is set and we have an extracted stack trace + } else if (!details.result && details.source) { + message += "" + "" + "
        Source:
        " + escapeText(details.source) + "
        "; + } + + assertList = testItem.getElementsByTagName("ol")[0]; + + assertLi = document.createElement("li"); + assertLi.className = details.result ? "pass" : "fail"; + assertLi.innerHTML = message; + assertList.appendChild(assertLi); + }); + + QUnit.testDone(function (details) { + var testTitle, + time, + testItem, + assertList, + status, + good, + bad, + testCounts, + skipped, + sourceName, + tests = id("qunit-tests"); + + if (!tests) { + return; + } + + testItem = id("qunit-test-output-" + details.testId); + + removeClass(testItem, "running"); + + if (details.failed > 0) { + status = "failed"; + } else if (details.todo) { + status = "todo"; + } else { + status = details.skipped ? "skipped" : "passed"; + } + + assertList = testItem.getElementsByTagName("ol")[0]; + + good = details.passed; + bad = details.failed; + + // This test passed if it has no unexpected failed assertions + var testPassed = details.failed > 0 ? details.todo : !details.todo; + + if (testPassed) { + + // Collapse the passing tests + addClass(assertList, "qunit-collapsed"); + } else if (config.collapse) { + if (!collapseNext) { + + // Skip collapsing the first failing test + collapseNext = true; + } else { + + // Collapse remaining tests + addClass(assertList, "qunit-collapsed"); + } + } + + // The testItem.firstChild is the test name + testTitle = testItem.firstChild; + + testCounts = bad ? "" + bad + ", " + "" + good + ", " : ""; + + testTitle.innerHTML += " (" + testCounts + details.assertions.length + ")"; + + if (details.skipped) { + stats.skippedTests++; + + testItem.className = "skipped"; + skipped = document.createElement("em"); + skipped.className = "qunit-skipped-label"; + skipped.innerHTML = "skipped"; + testItem.insertBefore(skipped, testTitle); + } else { + addEvent(testTitle, "click", function () { + toggleClass(assertList, "qunit-collapsed"); + }); + + testItem.className = testPassed ? "pass" : "fail"; + + if (details.todo) { + var todoLabel = document.createElement("em"); + todoLabel.className = "qunit-todo-label"; + todoLabel.innerHTML = "todo"; + testItem.className += " todo"; + testItem.insertBefore(todoLabel, testTitle); + } + + time = document.createElement("span"); + time.className = "runtime"; + time.innerHTML = details.runtime + " ms"; + testItem.insertBefore(time, assertList); + + if (!testPassed) { + stats.failedTests++; + } else if (details.todo) { + stats.todoTests++; + } else { + stats.passedTests++; + } + } + + // Show the source of the test when showing assertions + if (details.source) { + sourceName = document.createElement("p"); + sourceName.innerHTML = "Source: " + details.source; + addClass(sourceName, "qunit-source"); + if (testPassed) { + addClass(sourceName, "qunit-collapsed"); + } + addEvent(testTitle, "click", function () { + toggleClass(sourceName, "qunit-collapsed"); + }); + testItem.appendChild(sourceName); + } + + if (config.hidepassed && status === "passed") { + + // use removeChild instead of remove because of support + hiddenTests.push(testItem); + + tests.removeChild(testItem); + } + }); + + // Avoid readyState issue with phantomjs + // Ref: #818 + var notPhantom = function (p) { + return !(p && p.version && p.version.major > 0); + }(window$1.phantom); + + if (notPhantom && document.readyState === "complete") { + QUnit.load(); + } else { + addEvent(window$1, "load", QUnit.load); + } + + // Wrap window.onerror. We will call the original window.onerror to see if + // the existing handler fully handles the error; if not, we will call the + // QUnit.onError function. + var originalWindowOnError = window$1.onerror; + + // Cover uncaught exceptions + // Returning true will suppress the default browser handler, + // returning false will let it run. + window$1.onerror = function (message, fileName, lineNumber, columnNumber, errorObj) { + var ret = false; + if (originalWindowOnError) { + for (var _len = arguments.length, args = Array(_len > 5 ? _len - 5 : 0), _key = 5; _key < _len; _key++) { + args[_key - 5] = arguments[_key]; + } + + ret = originalWindowOnError.call.apply(originalWindowOnError, [this, message, fileName, lineNumber, columnNumber, errorObj].concat(args)); + } + + // Treat return value as window.onerror itself does, + // Only do our handling if not suppressed. + if (ret !== true) { + var error = { + message: message, + fileName: fileName, + lineNumber: lineNumber + }; + + // According to + // https://blog.sentry.io/2016/01/04/client-javascript-reporting-window-onerror, + // most modern browsers support an errorObj argument; use that to + // get a full stack trace if it's available. + if (errorObj && errorObj.stack) { + error.stacktrace = extractStacktrace(errorObj, 0); + } + + ret = QUnit.onError(error); + } + + return ret; + }; + + // Listen for unhandled rejections, and call QUnit.onUnhandledRejection + window$1.addEventListener("unhandledrejection", function (event) { + QUnit.onUnhandledRejection(event.reason); + }); + })(); + + /* + * This file is a modified version of google-diff-match-patch's JavaScript implementation + * (https://code.google.com/p/google-diff-match-patch/source/browse/trunk/javascript/diff_match_patch_uncompressed.js), + * modifications are licensed as more fully set forth in LICENSE.txt. + * + * The original source of google-diff-match-patch is attributable and licensed as follows: + * + * Copyright 2006 Google Inc. + * https://code.google.com/p/google-diff-match-patch/ + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * More Info: + * https://code.google.com/p/google-diff-match-patch/ + * + * Usage: QUnit.diff(expected, actual) + * + */ + QUnit.diff = function () { + function DiffMatchPatch() {} + + // DIFF FUNCTIONS + + /** + * The data structure representing a diff is an array of tuples: + * [[DIFF_DELETE, 'Hello'], [DIFF_INSERT, 'Goodbye'], [DIFF_EQUAL, ' world.']] + * which means: delete 'Hello', add 'Goodbye' and keep ' world.' + */ + var DIFF_DELETE = -1, + DIFF_INSERT = 1, + DIFF_EQUAL = 0; + + /** + * Find the differences between two texts. Simplifies the problem by stripping + * any common prefix or suffix off the texts before diffing. + * @param {string} text1 Old string to be diffed. + * @param {string} text2 New string to be diffed. + * @param {boolean=} optChecklines Optional speedup flag. If present and false, + * then don't run a line-level diff first to identify the changed areas. + * Defaults to true, which does a faster, slightly less optimal diff. + * @return {!Array.} Array of diff tuples. + */ + DiffMatchPatch.prototype.DiffMain = function (text1, text2, optChecklines) { + var deadline, checklines, commonlength, commonprefix, commonsuffix, diffs; + + // The diff must be complete in up to 1 second. + deadline = new Date().getTime() + 1000; + + // Check for null inputs. + if (text1 === null || text2 === null) { + throw new Error("Null input. (DiffMain)"); + } + + // Check for equality (speedup). + if (text1 === text2) { + if (text1) { + return [[DIFF_EQUAL, text1]]; + } + return []; + } + + if (typeof optChecklines === "undefined") { + optChecklines = true; + } + + checklines = optChecklines; + + // Trim off common prefix (speedup). + commonlength = this.diffCommonPrefix(text1, text2); + commonprefix = text1.substring(0, commonlength); + text1 = text1.substring(commonlength); + text2 = text2.substring(commonlength); + + // Trim off common suffix (speedup). + commonlength = this.diffCommonSuffix(text1, text2); + commonsuffix = text1.substring(text1.length - commonlength); + text1 = text1.substring(0, text1.length - commonlength); + text2 = text2.substring(0, text2.length - commonlength); + + // Compute the diff on the middle block. + diffs = this.diffCompute(text1, text2, checklines, deadline); + + // Restore the prefix and suffix. + if (commonprefix) { + diffs.unshift([DIFF_EQUAL, commonprefix]); + } + if (commonsuffix) { + diffs.push([DIFF_EQUAL, commonsuffix]); + } + this.diffCleanupMerge(diffs); + return diffs; + }; + + /** + * Reduce the number of edits by eliminating operationally trivial equalities. + * @param {!Array.} diffs Array of diff tuples. + */ + DiffMatchPatch.prototype.diffCleanupEfficiency = function (diffs) { + var changes, equalities, equalitiesLength, lastequality, pointer, preIns, preDel, postIns, postDel; + changes = false; + equalities = []; // Stack of indices where equalities are found. + equalitiesLength = 0; // Keeping our own length var is faster in JS. + /** @type {?string} */ + lastequality = null; + + // Always equal to diffs[equalities[equalitiesLength - 1]][1] + pointer = 0; // Index of current position. + + // Is there an insertion operation before the last equality. + preIns = false; + + // Is there a deletion operation before the last equality. + preDel = false; + + // Is there an insertion operation after the last equality. + postIns = false; + + // Is there a deletion operation after the last equality. + postDel = false; + while (pointer < diffs.length) { + + // Equality found. + if (diffs[pointer][0] === DIFF_EQUAL) { + if (diffs[pointer][1].length < 4 && (postIns || postDel)) { + + // Candidate found. + equalities[equalitiesLength++] = pointer; + preIns = postIns; + preDel = postDel; + lastequality = diffs[pointer][1]; + } else { + + // Not a candidate, and can never become one. + equalitiesLength = 0; + lastequality = null; + } + postIns = postDel = false; + + // An insertion or deletion. + } else { + + if (diffs[pointer][0] === DIFF_DELETE) { + postDel = true; + } else { + postIns = true; + } + + /* + * Five types to be split: + * ABXYCD + * AXCD + * ABXC + * AXCD + * ABXC + */ + if (lastequality && (preIns && preDel && postIns && postDel || lastequality.length < 2 && preIns + preDel + postIns + postDel === 3)) { + + // Duplicate record. + diffs.splice(equalities[equalitiesLength - 1], 0, [DIFF_DELETE, lastequality]); + + // Change second copy to insert. + diffs[equalities[equalitiesLength - 1] + 1][0] = DIFF_INSERT; + equalitiesLength--; // Throw away the equality we just deleted; + lastequality = null; + if (preIns && preDel) { + + // No changes made which could affect previous entry, keep going. + postIns = postDel = true; + equalitiesLength = 0; + } else { + equalitiesLength--; // Throw away the previous equality. + pointer = equalitiesLength > 0 ? equalities[equalitiesLength - 1] : -1; + postIns = postDel = false; + } + changes = true; + } + } + pointer++; + } + + if (changes) { + this.diffCleanupMerge(diffs); + } + }; + + /** + * Convert a diff array into a pretty HTML report. + * @param {!Array.} diffs Array of diff tuples. + * @param {integer} string to be beautified. + * @return {string} HTML representation. + */ + DiffMatchPatch.prototype.diffPrettyHtml = function (diffs) { + var op, + data, + x, + html = []; + for (x = 0; x < diffs.length; x++) { + op = diffs[x][0]; // Operation (insert, delete, equal) + data = diffs[x][1]; // Text of change. + switch (op) { + case DIFF_INSERT: + html[x] = "" + escapeText(data) + ""; + break; + case DIFF_DELETE: + html[x] = "" + escapeText(data) + ""; + break; + case DIFF_EQUAL: + html[x] = "" + escapeText(data) + ""; + break; + } + } + return html.join(""); + }; + + /** + * Determine the common prefix of two strings. + * @param {string} text1 First string. + * @param {string} text2 Second string. + * @return {number} The number of characters common to the start of each + * string. + */ + DiffMatchPatch.prototype.diffCommonPrefix = function (text1, text2) { + var pointermid, pointermax, pointermin, pointerstart; + + // Quick check for common null cases. + if (!text1 || !text2 || text1.charAt(0) !== text2.charAt(0)) { + return 0; + } + + // Binary search. + // Performance analysis: https://neil.fraser.name/news/2007/10/09/ + pointermin = 0; + pointermax = Math.min(text1.length, text2.length); + pointermid = pointermax; + pointerstart = 0; + while (pointermin < pointermid) { + if (text1.substring(pointerstart, pointermid) === text2.substring(pointerstart, pointermid)) { + pointermin = pointermid; + pointerstart = pointermin; + } else { + pointermax = pointermid; + } + pointermid = Math.floor((pointermax - pointermin) / 2 + pointermin); + } + return pointermid; + }; + + /** + * Determine the common suffix of two strings. + * @param {string} text1 First string. + * @param {string} text2 Second string. + * @return {number} The number of characters common to the end of each string. + */ + DiffMatchPatch.prototype.diffCommonSuffix = function (text1, text2) { + var pointermid, pointermax, pointermin, pointerend; + + // Quick check for common null cases. + if (!text1 || !text2 || text1.charAt(text1.length - 1) !== text2.charAt(text2.length - 1)) { + return 0; + } + + // Binary search. + // Performance analysis: https://neil.fraser.name/news/2007/10/09/ + pointermin = 0; + pointermax = Math.min(text1.length, text2.length); + pointermid = pointermax; + pointerend = 0; + while (pointermin < pointermid) { + if (text1.substring(text1.length - pointermid, text1.length - pointerend) === text2.substring(text2.length - pointermid, text2.length - pointerend)) { + pointermin = pointermid; + pointerend = pointermin; + } else { + pointermax = pointermid; + } + pointermid = Math.floor((pointermax - pointermin) / 2 + pointermin); + } + return pointermid; + }; + + /** + * Find the differences between two texts. Assumes that the texts do not + * have any common prefix or suffix. + * @param {string} text1 Old string to be diffed. + * @param {string} text2 New string to be diffed. + * @param {boolean} checklines Speedup flag. If false, then don't run a + * line-level diff first to identify the changed areas. + * If true, then run a faster, slightly less optimal diff. + * @param {number} deadline Time when the diff should be complete by. + * @return {!Array.} Array of diff tuples. + * @private + */ + DiffMatchPatch.prototype.diffCompute = function (text1, text2, checklines, deadline) { + var diffs, longtext, shorttext, i, hm, text1A, text2A, text1B, text2B, midCommon, diffsA, diffsB; + + if (!text1) { + + // Just add some text (speedup). + return [[DIFF_INSERT, text2]]; + } + + if (!text2) { + + // Just delete some text (speedup). + return [[DIFF_DELETE, text1]]; + } + + longtext = text1.length > text2.length ? text1 : text2; + shorttext = text1.length > text2.length ? text2 : text1; + i = longtext.indexOf(shorttext); + if (i !== -1) { + + // Shorter text is inside the longer text (speedup). + diffs = [[DIFF_INSERT, longtext.substring(0, i)], [DIFF_EQUAL, shorttext], [DIFF_INSERT, longtext.substring(i + shorttext.length)]]; + + // Swap insertions for deletions if diff is reversed. + if (text1.length > text2.length) { + diffs[0][0] = diffs[2][0] = DIFF_DELETE; + } + return diffs; + } + + if (shorttext.length === 1) { + + // Single character string. + // After the previous speedup, the character can't be an equality. + return [[DIFF_DELETE, text1], [DIFF_INSERT, text2]]; + } + + // Check to see if the problem can be split in two. + hm = this.diffHalfMatch(text1, text2); + if (hm) { + + // A half-match was found, sort out the return data. + text1A = hm[0]; + text1B = hm[1]; + text2A = hm[2]; + text2B = hm[3]; + midCommon = hm[4]; + + // Send both pairs off for separate processing. + diffsA = this.DiffMain(text1A, text2A, checklines, deadline); + diffsB = this.DiffMain(text1B, text2B, checklines, deadline); + + // Merge the results. + return diffsA.concat([[DIFF_EQUAL, midCommon]], diffsB); + } + + if (checklines && text1.length > 100 && text2.length > 100) { + return this.diffLineMode(text1, text2, deadline); + } + + return this.diffBisect(text1, text2, deadline); + }; + + /** + * Do the two texts share a substring which is at least half the length of the + * longer text? + * This speedup can produce non-minimal diffs. + * @param {string} text1 First string. + * @param {string} text2 Second string. + * @return {Array.} Five element Array, containing the prefix of + * text1, the suffix of text1, the prefix of text2, the suffix of + * text2 and the common middle. Or null if there was no match. + * @private + */ + DiffMatchPatch.prototype.diffHalfMatch = function (text1, text2) { + var longtext, shorttext, dmp, text1A, text2B, text2A, text1B, midCommon, hm1, hm2, hm; + + longtext = text1.length > text2.length ? text1 : text2; + shorttext = text1.length > text2.length ? text2 : text1; + if (longtext.length < 4 || shorttext.length * 2 < longtext.length) { + return null; // Pointless. + } + dmp = this; // 'this' becomes 'window' in a closure. + + /** + * Does a substring of shorttext exist within longtext such that the substring + * is at least half the length of longtext? + * Closure, but does not reference any external variables. + * @param {string} longtext Longer string. + * @param {string} shorttext Shorter string. + * @param {number} i Start index of quarter length substring within longtext. + * @return {Array.} Five element Array, containing the prefix of + * longtext, the suffix of longtext, the prefix of shorttext, the suffix + * of shorttext and the common middle. Or null if there was no match. + * @private + */ + function diffHalfMatchI(longtext, shorttext, i) { + var seed, j, bestCommon, prefixLength, suffixLength, bestLongtextA, bestLongtextB, bestShorttextA, bestShorttextB; + + // Start with a 1/4 length substring at position i as a seed. + seed = longtext.substring(i, i + Math.floor(longtext.length / 4)); + j = -1; + bestCommon = ""; + while ((j = shorttext.indexOf(seed, j + 1)) !== -1) { + prefixLength = dmp.diffCommonPrefix(longtext.substring(i), shorttext.substring(j)); + suffixLength = dmp.diffCommonSuffix(longtext.substring(0, i), shorttext.substring(0, j)); + if (bestCommon.length < suffixLength + prefixLength) { + bestCommon = shorttext.substring(j - suffixLength, j) + shorttext.substring(j, j + prefixLength); + bestLongtextA = longtext.substring(0, i - suffixLength); + bestLongtextB = longtext.substring(i + prefixLength); + bestShorttextA = shorttext.substring(0, j - suffixLength); + bestShorttextB = shorttext.substring(j + prefixLength); + } + } + if (bestCommon.length * 2 >= longtext.length) { + return [bestLongtextA, bestLongtextB, bestShorttextA, bestShorttextB, bestCommon]; + } else { + return null; + } + } + + // First check if the second quarter is the seed for a half-match. + hm1 = diffHalfMatchI(longtext, shorttext, Math.ceil(longtext.length / 4)); + + // Check again based on the third quarter. + hm2 = diffHalfMatchI(longtext, shorttext, Math.ceil(longtext.length / 2)); + if (!hm1 && !hm2) { + return null; + } else if (!hm2) { + hm = hm1; + } else if (!hm1) { + hm = hm2; + } else { + + // Both matched. Select the longest. + hm = hm1[4].length > hm2[4].length ? hm1 : hm2; + } + + // A half-match was found, sort out the return data. + if (text1.length > text2.length) { + text1A = hm[0]; + text1B = hm[1]; + text2A = hm[2]; + text2B = hm[3]; + } else { + text2A = hm[0]; + text2B = hm[1]; + text1A = hm[2]; + text1B = hm[3]; + } + midCommon = hm[4]; + return [text1A, text1B, text2A, text2B, midCommon]; + }; + + /** + * Do a quick line-level diff on both strings, then rediff the parts for + * greater accuracy. + * This speedup can produce non-minimal diffs. + * @param {string} text1 Old string to be diffed. + * @param {string} text2 New string to be diffed. + * @param {number} deadline Time when the diff should be complete by. + * @return {!Array.} Array of diff tuples. + * @private + */ + DiffMatchPatch.prototype.diffLineMode = function (text1, text2, deadline) { + var a, diffs, linearray, pointer, countInsert, countDelete, textInsert, textDelete, j; + + // Scan the text on a line-by-line basis first. + a = this.diffLinesToChars(text1, text2); + text1 = a.chars1; + text2 = a.chars2; + linearray = a.lineArray; + + diffs = this.DiffMain(text1, text2, false, deadline); + + // Convert the diff back to original text. + this.diffCharsToLines(diffs, linearray); + + // Eliminate freak matches (e.g. blank lines) + this.diffCleanupSemantic(diffs); + + // Rediff any replacement blocks, this time character-by-character. + // Add a dummy entry at the end. + diffs.push([DIFF_EQUAL, ""]); + pointer = 0; + countDelete = 0; + countInsert = 0; + textDelete = ""; + textInsert = ""; + while (pointer < diffs.length) { + switch (diffs[pointer][0]) { + case DIFF_INSERT: + countInsert++; + textInsert += diffs[pointer][1]; + break; + case DIFF_DELETE: + countDelete++; + textDelete += diffs[pointer][1]; + break; + case DIFF_EQUAL: + + // Upon reaching an equality, check for prior redundancies. + if (countDelete >= 1 && countInsert >= 1) { + + // Delete the offending records and add the merged ones. + diffs.splice(pointer - countDelete - countInsert, countDelete + countInsert); + pointer = pointer - countDelete - countInsert; + a = this.DiffMain(textDelete, textInsert, false, deadline); + for (j = a.length - 1; j >= 0; j--) { + diffs.splice(pointer, 0, a[j]); + } + pointer = pointer + a.length; + } + countInsert = 0; + countDelete = 0; + textDelete = ""; + textInsert = ""; + break; + } + pointer++; + } + diffs.pop(); // Remove the dummy entry at the end. + + return diffs; + }; + + /** + * Find the 'middle snake' of a diff, split the problem in two + * and return the recursively constructed diff. + * See Myers 1986 paper: An O(ND) Difference Algorithm and Its Variations. + * @param {string} text1 Old string to be diffed. + * @param {string} text2 New string to be diffed. + * @param {number} deadline Time at which to bail if not yet complete. + * @return {!Array.} Array of diff tuples. + * @private + */ + DiffMatchPatch.prototype.diffBisect = function (text1, text2, deadline) { + var text1Length, text2Length, maxD, vOffset, vLength, v1, v2, x, delta, front, k1start, k1end, k2start, k2end, k2Offset, k1Offset, x1, x2, y1, y2, d, k1, k2; + + // Cache the text lengths to prevent multiple calls. + text1Length = text1.length; + text2Length = text2.length; + maxD = Math.ceil((text1Length + text2Length) / 2); + vOffset = maxD; + vLength = 2 * maxD; + v1 = new Array(vLength); + v2 = new Array(vLength); + + // Setting all elements to -1 is faster in Chrome & Firefox than mixing + // integers and undefined. + for (x = 0; x < vLength; x++) { + v1[x] = -1; + v2[x] = -1; + } + v1[vOffset + 1] = 0; + v2[vOffset + 1] = 0; + delta = text1Length - text2Length; + + // If the total number of characters is odd, then the front path will collide + // with the reverse path. + front = delta % 2 !== 0; + + // Offsets for start and end of k loop. + // Prevents mapping of space beyond the grid. + k1start = 0; + k1end = 0; + k2start = 0; + k2end = 0; + for (d = 0; d < maxD; d++) { + + // Bail out if deadline is reached. + if (new Date().getTime() > deadline) { + break; + } + + // Walk the front path one step. + for (k1 = -d + k1start; k1 <= d - k1end; k1 += 2) { + k1Offset = vOffset + k1; + if (k1 === -d || k1 !== d && v1[k1Offset - 1] < v1[k1Offset + 1]) { + x1 = v1[k1Offset + 1]; + } else { + x1 = v1[k1Offset - 1] + 1; + } + y1 = x1 - k1; + while (x1 < text1Length && y1 < text2Length && text1.charAt(x1) === text2.charAt(y1)) { + x1++; + y1++; + } + v1[k1Offset] = x1; + if (x1 > text1Length) { + + // Ran off the right of the graph. + k1end += 2; + } else if (y1 > text2Length) { + + // Ran off the bottom of the graph. + k1start += 2; + } else if (front) { + k2Offset = vOffset + delta - k1; + if (k2Offset >= 0 && k2Offset < vLength && v2[k2Offset] !== -1) { + + // Mirror x2 onto top-left coordinate system. + x2 = text1Length - v2[k2Offset]; + if (x1 >= x2) { + + // Overlap detected. + return this.diffBisectSplit(text1, text2, x1, y1, deadline); + } + } + } + } + + // Walk the reverse path one step. + for (k2 = -d + k2start; k2 <= d - k2end; k2 += 2) { + k2Offset = vOffset + k2; + if (k2 === -d || k2 !== d && v2[k2Offset - 1] < v2[k2Offset + 1]) { + x2 = v2[k2Offset + 1]; + } else { + x2 = v2[k2Offset - 1] + 1; + } + y2 = x2 - k2; + while (x2 < text1Length && y2 < text2Length && text1.charAt(text1Length - x2 - 1) === text2.charAt(text2Length - y2 - 1)) { + x2++; + y2++; + } + v2[k2Offset] = x2; + if (x2 > text1Length) { + + // Ran off the left of the graph. + k2end += 2; + } else if (y2 > text2Length) { + + // Ran off the top of the graph. + k2start += 2; + } else if (!front) { + k1Offset = vOffset + delta - k2; + if (k1Offset >= 0 && k1Offset < vLength && v1[k1Offset] !== -1) { + x1 = v1[k1Offset]; + y1 = vOffset + x1 - k1Offset; + + // Mirror x2 onto top-left coordinate system. + x2 = text1Length - x2; + if (x1 >= x2) { + + // Overlap detected. + return this.diffBisectSplit(text1, text2, x1, y1, deadline); + } + } + } + } + } + + // Diff took too long and hit the deadline or + // number of diffs equals number of characters, no commonality at all. + return [[DIFF_DELETE, text1], [DIFF_INSERT, text2]]; + }; + + /** + * Given the location of the 'middle snake', split the diff in two parts + * and recurse. + * @param {string} text1 Old string to be diffed. + * @param {string} text2 New string to be diffed. + * @param {number} x Index of split point in text1. + * @param {number} y Index of split point in text2. + * @param {number} deadline Time at which to bail if not yet complete. + * @return {!Array.} Array of diff tuples. + * @private + */ + DiffMatchPatch.prototype.diffBisectSplit = function (text1, text2, x, y, deadline) { + var text1a, text1b, text2a, text2b, diffs, diffsb; + text1a = text1.substring(0, x); + text2a = text2.substring(0, y); + text1b = text1.substring(x); + text2b = text2.substring(y); + + // Compute both diffs serially. + diffs = this.DiffMain(text1a, text2a, false, deadline); + diffsb = this.DiffMain(text1b, text2b, false, deadline); + + return diffs.concat(diffsb); + }; + + /** + * Reduce the number of edits by eliminating semantically trivial equalities. + * @param {!Array.} diffs Array of diff tuples. + */ + DiffMatchPatch.prototype.diffCleanupSemantic = function (diffs) { + var changes, equalities, equalitiesLength, lastequality, pointer, lengthInsertions2, lengthDeletions2, lengthInsertions1, lengthDeletions1, deletion, insertion, overlapLength1, overlapLength2; + changes = false; + equalities = []; // Stack of indices where equalities are found. + equalitiesLength = 0; // Keeping our own length var is faster in JS. + /** @type {?string} */ + lastequality = null; + + // Always equal to diffs[equalities[equalitiesLength - 1]][1] + pointer = 0; // Index of current position. + + // Number of characters that changed prior to the equality. + lengthInsertions1 = 0; + lengthDeletions1 = 0; + + // Number of characters that changed after the equality. + lengthInsertions2 = 0; + lengthDeletions2 = 0; + while (pointer < diffs.length) { + if (diffs[pointer][0] === DIFF_EQUAL) { + // Equality found. + equalities[equalitiesLength++] = pointer; + lengthInsertions1 = lengthInsertions2; + lengthDeletions1 = lengthDeletions2; + lengthInsertions2 = 0; + lengthDeletions2 = 0; + lastequality = diffs[pointer][1]; + } else { + // An insertion or deletion. + if (diffs[pointer][0] === DIFF_INSERT) { + lengthInsertions2 += diffs[pointer][1].length; + } else { + lengthDeletions2 += diffs[pointer][1].length; + } + + // Eliminate an equality that is smaller or equal to the edits on both + // sides of it. + if (lastequality && lastequality.length <= Math.max(lengthInsertions1, lengthDeletions1) && lastequality.length <= Math.max(lengthInsertions2, lengthDeletions2)) { + + // Duplicate record. + diffs.splice(equalities[equalitiesLength - 1], 0, [DIFF_DELETE, lastequality]); + + // Change second copy to insert. + diffs[equalities[equalitiesLength - 1] + 1][0] = DIFF_INSERT; + + // Throw away the equality we just deleted. + equalitiesLength--; + + // Throw away the previous equality (it needs to be reevaluated). + equalitiesLength--; + pointer = equalitiesLength > 0 ? equalities[equalitiesLength - 1] : -1; + + // Reset the counters. + lengthInsertions1 = 0; + lengthDeletions1 = 0; + lengthInsertions2 = 0; + lengthDeletions2 = 0; + lastequality = null; + changes = true; + } + } + pointer++; + } + + // Normalize the diff. + if (changes) { + this.diffCleanupMerge(diffs); + } + + // Find any overlaps between deletions and insertions. + // e.g: abcxxxxxxdef + // -> abcxxxdef + // e.g: xxxabcdefxxx + // -> defxxxabc + // Only extract an overlap if it is as big as the edit ahead or behind it. + pointer = 1; + while (pointer < diffs.length) { + if (diffs[pointer - 1][0] === DIFF_DELETE && diffs[pointer][0] === DIFF_INSERT) { + deletion = diffs[pointer - 1][1]; + insertion = diffs[pointer][1]; + overlapLength1 = this.diffCommonOverlap(deletion, insertion); + overlapLength2 = this.diffCommonOverlap(insertion, deletion); + if (overlapLength1 >= overlapLength2) { + if (overlapLength1 >= deletion.length / 2 || overlapLength1 >= insertion.length / 2) { + + // Overlap found. Insert an equality and trim the surrounding edits. + diffs.splice(pointer, 0, [DIFF_EQUAL, insertion.substring(0, overlapLength1)]); + diffs[pointer - 1][1] = deletion.substring(0, deletion.length - overlapLength1); + diffs[pointer + 1][1] = insertion.substring(overlapLength1); + pointer++; + } + } else { + if (overlapLength2 >= deletion.length / 2 || overlapLength2 >= insertion.length / 2) { + + // Reverse overlap found. + // Insert an equality and swap and trim the surrounding edits. + diffs.splice(pointer, 0, [DIFF_EQUAL, deletion.substring(0, overlapLength2)]); + + diffs[pointer - 1][0] = DIFF_INSERT; + diffs[pointer - 1][1] = insertion.substring(0, insertion.length - overlapLength2); + diffs[pointer + 1][0] = DIFF_DELETE; + diffs[pointer + 1][1] = deletion.substring(overlapLength2); + pointer++; + } + } + pointer++; + } + pointer++; + } + }; + + /** + * Determine if the suffix of one string is the prefix of another. + * @param {string} text1 First string. + * @param {string} text2 Second string. + * @return {number} The number of characters common to the end of the first + * string and the start of the second string. + * @private + */ + DiffMatchPatch.prototype.diffCommonOverlap = function (text1, text2) { + var text1Length, text2Length, textLength, best, length, pattern, found; + + // Cache the text lengths to prevent multiple calls. + text1Length = text1.length; + text2Length = text2.length; + + // Eliminate the null case. + if (text1Length === 0 || text2Length === 0) { + return 0; + } + + // Truncate the longer string. + if (text1Length > text2Length) { + text1 = text1.substring(text1Length - text2Length); + } else if (text1Length < text2Length) { + text2 = text2.substring(0, text1Length); + } + textLength = Math.min(text1Length, text2Length); + + // Quick check for the worst case. + if (text1 === text2) { + return textLength; + } + + // Start by looking for a single character match + // and increase length until no match is found. + // Performance analysis: https://neil.fraser.name/news/2010/11/04/ + best = 0; + length = 1; + while (true) { + pattern = text1.substring(textLength - length); + found = text2.indexOf(pattern); + if (found === -1) { + return best; + } + length += found; + if (found === 0 || text1.substring(textLength - length) === text2.substring(0, length)) { + best = length; + length++; + } + } + }; + + /** + * Split two texts into an array of strings. Reduce the texts to a string of + * hashes where each Unicode character represents one line. + * @param {string} text1 First string. + * @param {string} text2 Second string. + * @return {{chars1: string, chars2: string, lineArray: !Array.}} + * An object containing the encoded text1, the encoded text2 and + * the array of unique strings. + * The zeroth element of the array of unique strings is intentionally blank. + * @private + */ + DiffMatchPatch.prototype.diffLinesToChars = function (text1, text2) { + var lineArray, lineHash, chars1, chars2; + lineArray = []; // E.g. lineArray[4] === 'Hello\n' + lineHash = {}; // E.g. lineHash['Hello\n'] === 4 + + // '\x00' is a valid character, but various debuggers don't like it. + // So we'll insert a junk entry to avoid generating a null character. + lineArray[0] = ""; + + /** + * Split a text into an array of strings. Reduce the texts to a string of + * hashes where each Unicode character represents one line. + * Modifies linearray and linehash through being a closure. + * @param {string} text String to encode. + * @return {string} Encoded string. + * @private + */ + function diffLinesToCharsMunge(text) { + var chars, lineStart, lineEnd, lineArrayLength, line; + chars = ""; + + // Walk the text, pulling out a substring for each line. + // text.split('\n') would would temporarily double our memory footprint. + // Modifying text would create many large strings to garbage collect. + lineStart = 0; + lineEnd = -1; + + // Keeping our own length variable is faster than looking it up. + lineArrayLength = lineArray.length; + while (lineEnd < text.length - 1) { + lineEnd = text.indexOf("\n", lineStart); + if (lineEnd === -1) { + lineEnd = text.length - 1; + } + line = text.substring(lineStart, lineEnd + 1); + lineStart = lineEnd + 1; + + var lineHashExists = lineHash.hasOwnProperty ? lineHash.hasOwnProperty(line) : lineHash[line] !== undefined; + + if (lineHashExists) { + chars += String.fromCharCode(lineHash[line]); + } else { + chars += String.fromCharCode(lineArrayLength); + lineHash[line] = lineArrayLength; + lineArray[lineArrayLength++] = line; + } + } + return chars; + } + + chars1 = diffLinesToCharsMunge(text1); + chars2 = diffLinesToCharsMunge(text2); + return { + chars1: chars1, + chars2: chars2, + lineArray: lineArray + }; + }; + + /** + * Rehydrate the text in a diff from a string of line hashes to real lines of + * text. + * @param {!Array.} diffs Array of diff tuples. + * @param {!Array.} lineArray Array of unique strings. + * @private + */ + DiffMatchPatch.prototype.diffCharsToLines = function (diffs, lineArray) { + var x, chars, text, y; + for (x = 0; x < diffs.length; x++) { + chars = diffs[x][1]; + text = []; + for (y = 0; y < chars.length; y++) { + text[y] = lineArray[chars.charCodeAt(y)]; + } + diffs[x][1] = text.join(""); + } + }; + + /** + * Reorder and merge like edit sections. Merge equalities. + * Any edit section can move as long as it doesn't cross an equality. + * @param {!Array.} diffs Array of diff tuples. + */ + DiffMatchPatch.prototype.diffCleanupMerge = function (diffs) { + var pointer, countDelete, countInsert, textInsert, textDelete, commonlength, changes, diffPointer, position; + diffs.push([DIFF_EQUAL, ""]); // Add a dummy entry at the end. + pointer = 0; + countDelete = 0; + countInsert = 0; + textDelete = ""; + textInsert = ""; + + while (pointer < diffs.length) { + switch (diffs[pointer][0]) { + case DIFF_INSERT: + countInsert++; + textInsert += diffs[pointer][1]; + pointer++; + break; + case DIFF_DELETE: + countDelete++; + textDelete += diffs[pointer][1]; + pointer++; + break; + case DIFF_EQUAL: + + // Upon reaching an equality, check for prior redundancies. + if (countDelete + countInsert > 1) { + if (countDelete !== 0 && countInsert !== 0) { + + // Factor out any common prefixes. + commonlength = this.diffCommonPrefix(textInsert, textDelete); + if (commonlength !== 0) { + if (pointer - countDelete - countInsert > 0 && diffs[pointer - countDelete - countInsert - 1][0] === DIFF_EQUAL) { + diffs[pointer - countDelete - countInsert - 1][1] += textInsert.substring(0, commonlength); + } else { + diffs.splice(0, 0, [DIFF_EQUAL, textInsert.substring(0, commonlength)]); + pointer++; + } + textInsert = textInsert.substring(commonlength); + textDelete = textDelete.substring(commonlength); + } + + // Factor out any common suffixies. + commonlength = this.diffCommonSuffix(textInsert, textDelete); + if (commonlength !== 0) { + diffs[pointer][1] = textInsert.substring(textInsert.length - commonlength) + diffs[pointer][1]; + textInsert = textInsert.substring(0, textInsert.length - commonlength); + textDelete = textDelete.substring(0, textDelete.length - commonlength); + } + } + + // Delete the offending records and add the merged ones. + if (countDelete === 0) { + diffs.splice(pointer - countInsert, countDelete + countInsert, [DIFF_INSERT, textInsert]); + } else if (countInsert === 0) { + diffs.splice(pointer - countDelete, countDelete + countInsert, [DIFF_DELETE, textDelete]); + } else { + diffs.splice(pointer - countDelete - countInsert, countDelete + countInsert, [DIFF_DELETE, textDelete], [DIFF_INSERT, textInsert]); + } + pointer = pointer - countDelete - countInsert + (countDelete ? 1 : 0) + (countInsert ? 1 : 0) + 1; + } else if (pointer !== 0 && diffs[pointer - 1][0] === DIFF_EQUAL) { + + // Merge this equality with the previous one. + diffs[pointer - 1][1] += diffs[pointer][1]; + diffs.splice(pointer, 1); + } else { + pointer++; + } + countInsert = 0; + countDelete = 0; + textDelete = ""; + textInsert = ""; + break; + } + } + if (diffs[diffs.length - 1][1] === "") { + diffs.pop(); // Remove the dummy entry at the end. + } + + // Second pass: look for single edits surrounded on both sides by equalities + // which can be shifted sideways to eliminate an equality. + // e.g: ABAC -> ABAC + changes = false; + pointer = 1; + + // Intentionally ignore the first and last element (don't need checking). + while (pointer < diffs.length - 1) { + if (diffs[pointer - 1][0] === DIFF_EQUAL && diffs[pointer + 1][0] === DIFF_EQUAL) { + + diffPointer = diffs[pointer][1]; + position = diffPointer.substring(diffPointer.length - diffs[pointer - 1][1].length); + + // This is a single edit surrounded by equalities. + if (position === diffs[pointer - 1][1]) { + + // Shift the edit over the previous equality. + diffs[pointer][1] = diffs[pointer - 1][1] + diffs[pointer][1].substring(0, diffs[pointer][1].length - diffs[pointer - 1][1].length); + diffs[pointer + 1][1] = diffs[pointer - 1][1] + diffs[pointer + 1][1]; + diffs.splice(pointer - 1, 1); + changes = true; + } else if (diffPointer.substring(0, diffs[pointer + 1][1].length) === diffs[pointer + 1][1]) { + + // Shift the edit over the next equality. + diffs[pointer - 1][1] += diffs[pointer + 1][1]; + diffs[pointer][1] = diffs[pointer][1].substring(diffs[pointer + 1][1].length) + diffs[pointer + 1][1]; + diffs.splice(pointer + 1, 1); + changes = true; + } + } + pointer++; + } + + // If shifts were made, the diff needs reordering and another shift sweep. + if (changes) { + this.diffCleanupMerge(diffs); + } + }; + + return function (o, n) { + var diff, output, text; + diff = new DiffMatchPatch(); + output = diff.DiffMain(o, n); + diff.diffCleanupEfficiency(output); + text = diff.diffPrettyHtml(output); + + return text; + }; + }(); + +}((function() { return this; }()))); diff --git a/test/tests-extensions.js b/test/tests-extensions.js index f0e5994..5a0b8ea 100644 --- a/test/tests-extensions.js +++ b/test/tests-extensions.js @@ -1,10 +1,12 @@ /*jshint -W024 */ /*jshint -W117 */ -module("extensions"); +QUnit.module("extensions"); -test("String.resolve basic (one dimension) test", 1, function () +QUnit.test("String.resolve basic (one dimension) test", function ( assert ) { + assert.expect( 1 ); + // given var values = { first: "test", @@ -16,11 +18,13 @@ test("String.resolve basic (one dimension) test", 1, function () var result = stringToResolve.resolve(values); // then - equal(result, "test case", "Valid string"); + assert.equal(result, "test case", "Valid string"); }); -test("String.resolve advanced (n dimension) test", 1, function () +QUnit.test("String.resolve advanced (n dimension) test", function ( assert ) { + assert.expect( 1 ); + // given var values = { first: { @@ -41,5 +45,5 @@ test("String.resolve advanced (n dimension) test", 1, function () var result = stringToResolve.resolve(values); // then - equal(result, "this is a more advanced test case", "Valid string"); + assert.equal(result, "this is a more advanced test case", "Valid string"); }); \ No newline at end of file diff --git a/test/tests-internal.js b/test/tests-internal.js index dea9eb6..5f88b30 100644 --- a/test/tests-internal.js +++ b/test/tests-internal.js @@ -1,19 +1,21 @@ /*jshint -W024 */ /*jshint -W117 */ -module("internal functions", { - setup: function () +QUnit.module("internal functions", { + beforeEach: function () { $("#qunit-fixture").html("
        "); }, - teardown: function () + afterEach: function () { $("#qunit-fixture").empty(); } }); -test("findFooterAndHeaderItems test", 1, function () +QUnit.test("findFooterAndHeaderItems test", function ( assert ) { + assert.expect( 1 ); + // given var instance = { footer: $("#test > tfoot"), @@ -25,11 +27,13 @@ test("findFooterAndHeaderItems test", 1, function () var result = findFooterAndHeaderItems.call(instance, selector); // then - equal(result.length, 2, "Found two elements as expected"); + assert.equal(result.length, 2, "Found two elements as expected"); }); -test("findFooterAndHeaderItems test (footer is null)", 1, function () +QUnit.test("findFooterAndHeaderItems test (footer is null)", function ( assert ) { + assert.expect( 1 ); + // given var instance = { footer: null, @@ -41,11 +45,13 @@ test("findFooterAndHeaderItems test (footer is null)", 1, function () var result = findFooterAndHeaderItems.call(instance, selector); // then - equal(result.length, 1, "Found one element as expected"); + assert.equal(result.length, 1, "Found one element as expected"); }); -test("findFooterAndHeaderItems test (header is null)", 1, function () +QUnit.test("findFooterAndHeaderItems test (header is null)", function ( assert ) { + assert.expect( 1 ); + // given var instance = { footer: $("#test > tfoot"), @@ -57,11 +63,13 @@ test("findFooterAndHeaderItems test (header is null)", 1, function () var result = findFooterAndHeaderItems.call(instance, selector); // then - equal(result.length, 1, "Found one element as expected"); + assert.equal(result.length, 1, "Found one element as expected"); }); -test("findFooterAndHeaderItems test (footer and header is string empty)", 2, function () +QUnit.test("findFooterAndHeaderItems test (footer and header is string empty)", function ( assert ) { + assert.expect( 2 ); + // given var instance = { footer: "", @@ -73,12 +81,14 @@ test("findFooterAndHeaderItems test (footer and header is string empty)", 2, fun var result = findFooterAndHeaderItems.call(instance, selector); // then - equal(result.length, 0, "Foundd one element as expecte"); - ok(result.find, "Got an empty jQuery array as expected"); + assert.equal(result.length, 0, "Foundd one element as expecte"); + assert.ok(result.find, "Got an empty jQuery array as expected"); }); -test("findFooterAndHeaderItems test (footer and header is null)", 2, function () +QUnit.test("findFooterAndHeaderItems test (footer and header is null)", function ( assert ) { + assert.expect( 2 ); + // given var instance = { footer: null, @@ -90,12 +100,14 @@ test("findFooterAndHeaderItems test (footer and header is null)", 2, function () var result = findFooterAndHeaderItems.call(instance, selector); // then - equal(result.length, 0, "Found no elements as expected"); - ok(result.find, "Got an empty jQuery array as expected"); + assert.equal(result.length, 0, "Found no elements as expected"); + assert.ok(result.find, "Got an empty jQuery array as expected"); }); -test("getRequest post function test", 1, function () +QUnit.test("getRequest post function test", function ( assert ) { + assert.expect( 1 ); + // given var instance = { options: { @@ -124,10 +136,12 @@ test("getRequest post function test", 1, function () var result = getRequest.call(instance); // then - propEqual(result, expected, "Valid request object"); + assert.propEqual(result, expected, "Valid request object"); }); -test("getRequest post object test", 1, function() { +QUnit.test("getRequest post object test", function( assert ) { + assert.expect( 1 ); + // given var instance = { options: { @@ -153,11 +167,13 @@ test("getRequest post object test", 1, function() { var result = getRequest.call(instance); // then - propEqual(result, expected, "Valid request object"); + assert.propEqual(result, expected, "Valid request object"); }); -test("getCssSelector test", 1, function () +QUnit.test("getCssSelector test", function ( assert ) { + assert.expect( 1 ); + // given var classNames = " itallic bold normal "; @@ -165,11 +181,13 @@ test("getCssSelector test", 1, function () var result = getCssSelector(classNames); // then - equal(result, ".itallic.bold.normal", "Valid css selector"); + assert.equal(result, ".itallic.bold.normal", "Valid css selector"); }); -test("getUrl function test", 1, function () +QUnit.test("getUrl function test", function ( assert ) { + assert.expect( 1 ); + // given var instance = { options: { @@ -184,11 +202,13 @@ test("getUrl function test", 1, function () var result = getUrl.call(instance); // then - equal(result, "url/test/1", "Valid URL"); + assert.equal(result, "url/test/1", "Valid URL"); }); -test("getUrl string test", 1, function () +QUnit.test("getUrl string test", function ( assert ) { + assert.expect( 1 ); + // given var instance = { options: { @@ -200,5 +220,5 @@ test("getUrl string test", 1, function () var result = getUrl.call(instance); // then - equal(result, "url/test/1", "Valid URL"); + assert.equal(result, "url/test/1", "Valid URL"); }); diff --git a/test/tests-rendering.js b/test/tests-rendering.js index 3fa6682..632e603 100644 --- a/test/tests-rendering.js +++ b/test/tests-rendering.js @@ -1,18 +1,18 @@ /*jshint -W024 */ /*jshint -W117 */ -module("render functions", { - setup: function () +QUnit.module("render functions", { + beforeEach: function () { $("#qunit-fixture").html("
        "); }, - teardown: function () + afterEach: function () { $("#qunit-fixture").empty(); } }); -function renderInfosTest(expected, message, current, rowCount, total) +function renderInfosTest(expected, message, current, rowCount, total, assert) { // given var instance = { @@ -41,15 +41,19 @@ function renderInfosTest(expected, message, current, rowCount, total) // then var infos = instance.header.find(".infos").text(); - equal(infos, expected, message); + assert.equal(infos, expected, message); } -test("renderInfos all test", 1, function () +QUnit.test("renderInfos all test", function ( assert ) { - renderInfosTest("11010", "Valid infos", 1, -1, 10); + assert.expect( 1 ); + + renderInfosTest("11010", "Valid infos", 1, -1, 10, assert); }); -test("renderInfos paged test", 1, function () +QUnit.test("renderInfos paged test", function ( assert ) { - renderInfosTest("1510", "Valid infos", 1, 5, 10); + assert.expect( 1 ); + + renderInfosTest("1510", "Valid infos", 1, 5, 10, assert); }); \ No newline at end of file From 566fb2d6911096317c3481d5e115e351d527996b Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 13:12:00 +0100 Subject: [PATCH 10/17] Use newer version of grunt-contrib-qunit --- package.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 5572922..bdb2676 100644 --- a/package.json +++ b/package.json @@ -36,8 +36,8 @@ ], "license": "MIT", "dependencies": { - "jquery": ">=1.9", - "bootstrap": "3.1.1 - 3.3" + "bootstrap": "3.1.1 - 3.3", + "jquery": ">=1.9" }, "devDependencies": { "grunt": "^0.4.5", @@ -49,7 +49,7 @@ "grunt-contrib-cssmin": "~0.10.0", "grunt-contrib-jshint": "~0.10.0", "grunt-contrib-less": "~0.11.0", - "grunt-contrib-qunit": "~0.4.0", + "grunt-contrib-qunit": "~3.0.1", "grunt-contrib-uglify": "~0.9.2", "grunt-contrib-yuidoc": "~0.5.2", "grunt-exec": "~0.4.5", From 57bfbafdb4f09e4d797ed7acda0a38d91d57981c Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 13:25:21 +0100 Subject: [PATCH 11/17] Document build process in README.md --- README.md | 34 +++++++++++++++++++++++++++++++++- 1 file changed, 33 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 5c0a613..44a933d 100644 --- a/README.md +++ b/README.md @@ -53,9 +53,41 @@ Instructions will follow soon! I'm always happy to help answer your questions. The best way to get quick answers is to go to [stackoverflow.com](http://stackoverflow.com) and tag your questions always with **jquery-bootgrid**. +## Building + +### Environment + +jquery-bootgrid uses npm to install its own dependencies, and mono to build a NuGet package. + +You should install both using your package manager. On macOS, this works as follows: +``` +brew install npm +brew install mono +``` + +### Build dependencies + +From the folder that jquery-bootgrid resides in, run: +``` +npm install . +sudo npm install -g grunt +``` + +### Build jquery-bootgrid itself +``` +grunt +``` + +If you are done with your modifications, you should increase the version number in boewer.json and package.json, +update the changelog, and then run: +``` +grunt release +``` + + ## Contributing -Instructions will follow soon! +No instructions yet. ## License From f91d5b02a972fc6fba0b1da097607ed6142074bd Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 13:26:30 +0100 Subject: [PATCH 12/17] Release 1.3.2 --- CHANGELOG.md | 8 + bower.json | 2 +- dist/jQuery.Bootgrid.1.3.1.nupkg | Bin 27538 -> 0 bytes dist/jQuery.Bootgrid.1.3.2.nupkg | Bin 0 -> 26227 bytes dist/jquery.bootgrid-1.3.1.zip | Bin 29488 -> 0 bytes dist/jquery.bootgrid-1.3.2.zip | Bin 0 -> 28898 bytes dist/jquery.bootgrid.css | 6 +- dist/jquery.bootgrid.fa.js | 20 +- dist/jquery.bootgrid.fa.min.js | 6 +- dist/jquery.bootgrid.js | 3278 +++++++++++++++--------------- dist/jquery.bootgrid.min.css | 6 +- dist/jquery.bootgrid.min.js | 8 +- package.json | 2 +- 13 files changed, 1672 insertions(+), 1664 deletions(-) delete mode 100644 dist/jQuery.Bootgrid.1.3.1.nupkg create mode 100644 dist/jQuery.Bootgrid.1.3.2.nupkg delete mode 100644 dist/jquery.bootgrid-1.3.1.zip create mode 100644 dist/jquery.bootgrid-1.3.2.zip diff --git a/CHANGELOG.md b/CHANGELOG.md index a3c5e48..fddc843 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,13 @@ # Changelog +## 1.3.2 +### Bug fixes +- Fix jQuery and Bootstrap versions in package.json, fix license info in package.json + +### Enhancements & Features +- Use left and right single guillemets in pagination. +- Migrated test suite to Qunit 2.8.0. + ## 1.3.1 ### Enhancements & Features diff --git a/bower.json b/bower.json index 856e916..08853e5 100644 --- a/bower.json +++ b/bower.json @@ -13,7 +13,7 @@ "accessibility", "bootstrap" ], - "version": "1.3.1", + "version": "1.3.2", "authors": [ { "name": "Rafael Staib", diff --git a/dist/jQuery.Bootgrid.1.3.1.nupkg b/dist/jQuery.Bootgrid.1.3.1.nupkg deleted file mode 100644 index c430a89692eed3a2e7d5fb7b5e5ecfbb6e8445ba..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27538 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 A4gdfE diff --git a/dist/jQuery.Bootgrid.1.3.2.nupkg b/dist/jQuery.Bootgrid.1.3.2.nupkg new file mode 100644 index 0000000000000000000000000000000000000000..aa1a599d518ba5d60f167b7d447616e238d7f89e GIT binary patch literal 26227 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≻54{i51ivl7?F&(5>(`*m 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 wm!|(S$6u)Y&m#W4)&C&uzk=NAe tr").first(), + sorted = false; - function highlightAppendedRows(rows) + /*jshint -W018*/ + firstHeadRow.children().each(function () { - if (this.options.highlightRows) + var $this = $(this), + data = $this.data(), + column = { + id: data.columnId, + identifier: that.identifier == null && data.identifier || false, + converter: that.options.converters[data.converter || data.type] || that.options.converters["string"], + text: $this.text(), + align: data.align || "left", + headerAlign: data.headerAlign || "left", + cssClass: data.cssClass || "", + headerCssClass: data.headerCssClass || "", + formatter: that.options.formatters[data.formatter] || null, + order: (!sorted && (data.order === "asc" || data.order === "desc")) ? data.order : null, + searchable: !(data.searchable === false), // default: true + sortable: !(data.sortable === false), // default: true + visible: !(data.visible === false), // default: true + visibleInSelection: !(data.visibleInSelection === false), // default: true + width: ($.isNumeric(data.width)) ? data.width + "px" : + (typeof(data.width) === "string") ? data.width : null + }; + that.columns.push(column); + if (column.order != null) { - // todo: implement + that.sortDictionary[column.id] = column.order; } - } - - function isVisible(column) - { - return column.visible; - } - function loadColumns() - { - var that = this, - firstHeadRow = this.element.find("thead > tr").first(), - sorted = false; + // Prevents multiple identifiers + if (column.identifier) + { + that.identifier = column.id; + that.converter = column.converter; + } - /*jshint -W018*/ - firstHeadRow.children().each(function () + // ensures that only the first order will be applied in case of multi sorting is disabled + if (!that.options.multiSort && column.order !== null) { - var $this = $(this), - data = $this.data(), - column = { - id: data.columnId, - identifier: that.identifier == null && data.identifier || false, - converter: that.options.converters[data.converter || data.type] || that.options.converters["string"], - text: $this.text(), - align: data.align || "left", - headerAlign: data.headerAlign || "left", - cssClass: data.cssClass || "", - headerCssClass: data.headerCssClass || "", - formatter: that.options.formatters[data.formatter] || null, - order: (!sorted && (data.order === "asc" || data.order === "desc")) ? data.order : null, - searchable: !(data.searchable === false), // default: true - sortable: !(data.sortable === false), // default: true - visible: !(data.visible === false), // default: true - visibleInSelection: !(data.visibleInSelection === false), // default: true - width: ($.isNumeric(data.width)) ? data.width + "px" : - (typeof(data.width) === "string") ? data.width : null - }; - that.columns.push(column); - if (column.order != null) - { - that.sortDictionary[column.id] = column.order; - } + sorted = true; + } + }); + /*jshint +W018*/ +} + +/* +response = { + current: 1, + rowCount: 10, + rows: [{}, {}], + sort: [{ "columnId": "asc" }], + total: 101 +} +*/ + +function loadData() +{ + var that = this; - // Prevents multiple identifiers - if (column.identifier) - { - that.identifier = column.id; - that.converter = column.converter; - } + this.element._bgBusyAria(true).trigger("load" + namespace); + showLoading.call(this); + + function containsPhrase(row) + { + var column, + searchPattern = new RegExp(that.searchPhrase, (that.options.caseSensitive) ? "g" : "gi"); - // ensures that only the first order will be applied in case of multi sorting is disabled - if (!that.options.multiSort && column.order !== null) + for (var i = 0; i < that.columns.length; i++) + { + column = that.columns[i]; + if (column.searchable && column.visible && + column.converter.to(row[column.id]).search(searchPattern) > -1) { - sorted = true; + return true; } - }); - /*jshint +W018*/ - } + } - /* - response = { - current: 1, - rowCount: 10, - rows: [{}, {}], - sort: [{ "columnId": "asc" }], - total: 101 + return false; } - */ - function loadData() + function update(rows, total) { - var that = this; + that.currentRows = rows; + setTotals.call(that, total); - this.element._bgBusyAria(true).trigger("load" + namespace); - showLoading.call(this); - - function containsPhrase(row) + if (!that.options.keepSelection) { - var column, - searchPattern = new RegExp(that.searchPhrase, (that.options.caseSensitive) ? "g" : "gi"); - - for (var i = 0; i < that.columns.length; i++) - { - column = that.columns[i]; - if (column.searchable && column.visible && - column.converter.to(row[column.id]).search(searchPattern) > -1) - { - return true; - } - } - - return false; + that.selectedRows = []; } - function update(rows, total) - { - that.currentRows = rows; - setTotals.call(that, total); + renderRows.call(that, rows); + renderInfos.call(that); + renderPagination.call(that); - if (!that.options.keepSelection) - { - that.selectedRows = []; - } + that.element._bgBusyAria(false).trigger("loaded" + namespace); + } - renderRows.call(that, rows); - renderInfos.call(that); - renderPagination.call(that); + if (this.options.ajax) + { + var request = getRequest.call(this), + url = getUrl.call(this); - that.element._bgBusyAria(false).trigger("loaded" + namespace); + if (url == null || typeof url !== "string" || url.length === 0) + { + throw new Error("Url setting must be a none empty string or a function that returns one."); } - if (this.options.ajax) + // aborts the previous ajax request if not already finished or failed + if (this.xqr) { - var request = getRequest.call(this), - url = getUrl.call(this); - - if (url == null || typeof url !== "string" || url.length === 0) - { - throw new Error("Url setting must be a none empty string or a function that returns one."); - } + this.xqr.abort(); + } - // aborts the previous ajax request if not already finished or failed - if (this.xqr) + var settings = { + url: url, + data: request, + success: function(response) { - this.xqr.abort(); - } + that.xqr = null; - var settings = { - url: url, - data: request, - success: function(response) + if (typeof (response) === "string") { - that.xqr = null; + response = $.parseJSON(response); + } - if (typeof (response) === "string") - { - response = $.parseJSON(response); - } + response = that.options.responseHandler(response); - response = that.options.responseHandler(response); + that.current = response.current; + update(response.rows, response.total); + }, + error: function (jqXHR, textStatus, errorThrown) + { + that.xqr = null; - that.current = response.current; - update(response.rows, response.total); - }, - error: function (jqXHR, textStatus, errorThrown) + if (textStatus !== "abort") { - that.xqr = null; - - if (textStatus !== "abort") - { - renderNoResultsRow.call(that); // overrides loading mask - that.element._bgBusyAria(false).trigger("loaded" + namespace); - } + renderNoResultsRow.call(that); // overrides loading mask + that.element._bgBusyAria(false).trigger("loaded" + namespace); } - }; - settings = $.extend(this.options.ajaxSettings, settings); - - this.xqr = $.ajax(settings); - } - else - { - var rows = (this.searchPhrase.length > 0) ? this.rows.where(containsPhrase) : this.rows, - total = rows.length; - if (this.rowCount !== -1) - { - rows = rows.page(this.current, this.rowCount); } + }; + settings = $.extend(this.options.ajaxSettings, settings); - // todo: improve the following comment - // setTimeout decouples the initialization so that adding event handlers happens before - window.setTimeout(function () { update(rows, total); }, 10); + this.xqr = $.ajax(settings); + } + else + { + var rows = (this.searchPhrase.length > 0) ? this.rows.where(containsPhrase) : this.rows, + total = rows.length; + if (this.rowCount !== -1) + { + rows = rows.page(this.current, this.rowCount); } + + // todo: improve the following comment + // setTimeout decouples the initialization so that adding event handlers happens before + window.setTimeout(function () { update(rows, total); }, 10); } +} - function loadRows() +function loadRows() +{ + if (!this.options.ajax) { - if (!this.options.ajax) + var that = this, + rows = this.element.find("tbody > tr"); + + rows.each(function () { - var that = this, - rows = this.element.find("tbody > tr"); + var $this = $(this), + cells = $this.children("td"), + row = {}; - rows.each(function () + $.each(that.columns, function (i, column) { - var $this = $(this), - cells = $this.children("td"), - row = {}; + row[column.id] = column.converter.from(cells.eq(i).text()); + }); - $.each(that.columns, function (i, column) - { - row[column.id] = column.converter.from(cells.eq(i).text()); - }); + appendRow.call(that, row); + }); - appendRow.call(that, row); - }); + setTotals.call(this, this.rows.length); + sortRows.call(this); + } +} - setTotals.call(this, this.rows.length); - sortRows.call(this); - } +function setTotals(total) +{ + this.total = total; + this.totalPages = (this.rowCount === -1) ? 1 : + Math.ceil(this.total / this.rowCount); +} + +function prepareTable() +{ + var tpl = this.options.templates, + wrapper = (this.element.parent().hasClass(this.options.css.responsiveTable)) ? + this.element.parent() : this.element; + + this.element.addClass(this.options.css.table); + + // checks whether there is an tbody element; otherwise creates one + if (this.element.children("tbody").length === 0) + { + this.element.append(tpl.body); } - function setTotals(total) + if (this.options.navigation & 1) { - this.total = total; - this.totalPages = (this.rowCount === -1) ? 1 : - Math.ceil(this.total / this.rowCount); + this.header = $(tpl.header.resolve(getParams.call(this, { id: this.element._bgId() + "-header" }))); + wrapper.before(this.header); } - function prepareTable() + if (this.options.navigation & 2) { - var tpl = this.options.templates, - wrapper = (this.element.parent().hasClass(this.options.css.responsiveTable)) ? - this.element.parent() : this.element; + this.footer = $(tpl.footer.resolve(getParams.call(this, { id: this.element._bgId() + "-footer" }))); + wrapper.after(this.footer); + } +} - this.element.addClass(this.options.css.table); +function renderActions() +{ + if (this.options.navigation !== 0) + { + var css = this.options.css, + selector = getCssSelector(css.actions), + actionItems = findFooterAndHeaderItems.call(this, selector); - // checks whether there is an tbody element; otherwise creates one - if (this.element.children("tbody").length === 0) + if (actionItems.length > 0) { - this.element.append(tpl.body); - } + var that = this, + tpl = this.options.templates, + actions = $(tpl.actions.resolve(getParams.call(this))); - if (this.options.navigation & 1) - { - this.header = $(tpl.header.resolve(getParams.call(this, { id: this.element._bgId() + "-header" }))); - wrapper.before(this.header); - } + // Refresh Button + if (this.options.ajax) + { + var refreshIcon = tpl.icon.resolve(getParams.call(this, { iconCss: css.iconRefresh })), + refresh = $(tpl.actionButton.resolve(getParams.call(this, + { content: refreshIcon, text: this.options.labels.refresh }))) + .on("click" + namespace, function (e) + { + // todo: prevent multiple fast clicks (fast click detection) + e.stopPropagation(); + that.current = 1; + loadData.call(that); + }); + actions.append(refresh); + } - if (this.options.navigation & 2) - { - this.footer = $(tpl.footer.resolve(getParams.call(this, { id: this.element._bgId() + "-footer" }))); - wrapper.after(this.footer); + // Row count selection + renderRowCountSelection.call(this, actions); + + // Column selection + renderColumnSelection.call(this, actions); + + replacePlaceHolder.call(this, actionItems, actions); } } +} - function renderActions() +function renderColumnSelection(actions) +{ + if (this.options.columnSelection && this.columns.length > 1) { - if (this.options.navigation !== 0) - { - var css = this.options.css, - selector = getCssSelector(css.actions), - actionItems = findFooterAndHeaderItems.call(this, selector); + var that = this, + css = this.options.css, + tpl = this.options.templates, + icon = tpl.icon.resolve(getParams.call(this, { iconCss: css.iconColumns })), + dropDown = $(tpl.actionDropDown.resolve(getParams.call(this, { content: icon }))), + selector = getCssSelector(css.dropDownItem), + checkboxSelector = getCssSelector(css.dropDownItemCheckbox), + itemsSelector = getCssSelector(css.dropDownMenuItems); - if (actionItems.length > 0) + $.each(this.columns, function (i, column) + { + if (column.visibleInSelection) { - var that = this, - tpl = this.options.templates, - actions = $(tpl.actions.resolve(getParams.call(this))); - - // Refresh Button - if (this.options.ajax) - { - var refreshIcon = tpl.icon.resolve(getParams.call(this, { iconCss: css.iconRefresh })), - refresh = $(tpl.actionButton.resolve(getParams.call(this, - { content: refreshIcon, text: this.options.labels.refresh }))) - .on("click" + namespace, function (e) + var item = $(tpl.actionDropDownCheckboxItem.resolve(getParams.call(that, + { name: column.id, label: column.text, checked: column.visible }))) + .on("click" + namespace, selector, function (e) + { + e.stopPropagation(); + + var $this = $(this), + checkbox = $this.find(checkboxSelector); + if (!checkbox.prop("disabled")) { - // todo: prevent multiple fast clicks (fast click detection) - e.stopPropagation(); - that.current = 1; + column.visible = checkbox.prop("checked"); + var enable = that.columns.where(isVisible).length > 1; + $this.parents(itemsSelector).find(selector + ":has(" + checkboxSelector + ":checked)") + ._bgEnableAria(enable).find(checkboxSelector)._bgEnableField(enable); + + that.element.find("tbody").empty(); // Fixes an column visualization bug + renderTableHeader.call(that); loadData.call(that); - }); - actions.append(refresh); - } - - // Row count selection - renderRowCountSelection.call(this, actions); - - // Column selection - renderColumnSelection.call(this, actions); - - replacePlaceHolder.call(this, actionItems, actions); + } + }); + dropDown.find(getCssSelector(css.dropDownMenuItems)).append(item); } - } + }); + actions.append(dropDown); } +} - function renderColumnSelection(actions) +function renderInfos() +{ + if (this.options.navigation !== 0) { - if (this.options.columnSelection && this.columns.length > 1) - { - var that = this, - css = this.options.css, - tpl = this.options.templates, - icon = tpl.icon.resolve(getParams.call(this, { iconCss: css.iconColumns })), - dropDown = $(tpl.actionDropDown.resolve(getParams.call(this, { content: icon }))), - selector = getCssSelector(css.dropDownItem), - checkboxSelector = getCssSelector(css.dropDownItemCheckbox), - itemsSelector = getCssSelector(css.dropDownMenuItems); + var selector = getCssSelector(this.options.css.infos), + infoItems = findFooterAndHeaderItems.call(this, selector); - $.each(this.columns, function (i, column) - { - if (column.visibleInSelection) - { - var item = $(tpl.actionDropDownCheckboxItem.resolve(getParams.call(that, - { name: column.id, label: column.text, checked: column.visible }))) - .on("click" + namespace, selector, function (e) - { - e.stopPropagation(); - - var $this = $(this), - checkbox = $this.find(checkboxSelector); - if (!checkbox.prop("disabled")) - { - column.visible = checkbox.prop("checked"); - var enable = that.columns.where(isVisible).length > 1; - $this.parents(itemsSelector).find(selector + ":has(" + checkboxSelector + ":checked)") - ._bgEnableAria(enable).find(checkboxSelector)._bgEnableField(enable); - - that.element.find("tbody").empty(); // Fixes an column visualization bug - renderTableHeader.call(that); - loadData.call(that); - } - }); - dropDown.find(getCssSelector(css.dropDownMenuItems)).append(item); - } - }); - actions.append(dropDown); + if (infoItems.length > 0) + { + var end = (this.current * this.rowCount), + infos = $(this.options.templates.infos.resolve(getParams.call(this, { + end: (this.total === 0 || end === -1 || end > this.total) ? this.total : end, + start: (this.total === 0) ? 0 : (end - this.rowCount + 1), + total: this.total + }))); + + replacePlaceHolder.call(this, infoItems, infos); } } +} + +function renderNoResultsRow() +{ + var tbody = this.element.children("tbody").first(), + tpl = this.options.templates, + count = this.columns.where(isVisible).length; + + if (this.selection) + { + count = count + 1; + } + tbody.html(tpl.noResults.resolve(getParams.call(this, { columns: count }))); +} - function renderInfos() +function renderPagination() +{ + if (this.options.navigation !== 0) { - if (this.options.navigation !== 0) + var selector = getCssSelector(this.options.css.pagination), + paginationItems = findFooterAndHeaderItems.call(this, selector)._bgShowAria(this.rowCount !== -1); + + if (this.rowCount !== -1 && paginationItems.length > 0) { - var selector = getCssSelector(this.options.css.infos), - infoItems = findFooterAndHeaderItems.call(this, selector); + var tpl = this.options.templates, + current = this.current, + totalPages = this.totalPages, + pagination = $(tpl.pagination.resolve(getParams.call(this))), + offsetRight = totalPages - current, + offsetLeft = (this.options.padding - current) * -1, + startWith = ((offsetRight >= this.options.padding) ? + Math.max(offsetLeft, 1) : + Math.max((offsetLeft - this.options.padding + offsetRight), 1)), + maxCount = this.options.padding * 2 + 1, + count = (totalPages >= maxCount) ? maxCount : totalPages; + + renderPaginationItem.call(this, pagination, "first", "«", "first") + ._bgEnableAria(current > 1); + renderPaginationItem.call(this, pagination, "prev", "‹", "prev") + ._bgEnableAria(current > 1); + + for (var i = 0; i < count; i++) + { + var pos = i + startWith; + renderPaginationItem.call(this, pagination, pos, pos, "page-" + pos) + ._bgEnableAria()._bgSelectAria(pos === current); + } - if (infoItems.length > 0) + if (count === 0) { - var end = (this.current * this.rowCount), - infos = $(this.options.templates.infos.resolve(getParams.call(this, { - end: (this.total === 0 || end === -1 || end > this.total) ? this.total : end, - start: (this.total === 0) ? 0 : (end - this.rowCount + 1), - total: this.total - }))); - - replacePlaceHolder.call(this, infoItems, infos); + renderPaginationItem.call(this, pagination, 1, 1, "page-" + 1) + ._bgEnableAria(false)._bgSelectAria(); } - } - } - function renderNoResultsRow() - { - var tbody = this.element.children("tbody").first(), - tpl = this.options.templates, - count = this.columns.where(isVisible).length; + renderPaginationItem.call(this, pagination, "next", "›", "next") + ._bgEnableAria(totalPages > current); + renderPaginationItem.call(this, pagination, "last", "»", "last") + ._bgEnableAria(totalPages > current); - if (this.selection) - { - count = count + 1; + replacePlaceHolder.call(this, paginationItems, pagination); } - tbody.html(tpl.noResults.resolve(getParams.call(this, { columns: count }))); } +} - function renderPagination() - { - if (this.options.navigation !== 0) - { - var selector = getCssSelector(this.options.css.pagination), - paginationItems = findFooterAndHeaderItems.call(this, selector)._bgShowAria(this.rowCount !== -1); - - if (this.rowCount !== -1 && paginationItems.length > 0) +function renderPaginationItem(list, page, text, markerCss) +{ + var that = this, + tpl = this.options.templates, + css = this.options.css, + values = getParams.call(this, { css: markerCss, text: text, page: page }), + item = $(tpl.paginationItem.resolve(values)) + .on("click" + namespace, getCssSelector(css.paginationButton), function (e) { - var tpl = this.options.templates, - current = this.current, - totalPages = this.totalPages, - pagination = $(tpl.pagination.resolve(getParams.call(this))), - offsetRight = totalPages - current, - offsetLeft = (this.options.padding - current) * -1, - startWith = ((offsetRight >= this.options.padding) ? - Math.max(offsetLeft, 1) : - Math.max((offsetLeft - this.options.padding + offsetRight), 1)), - maxCount = this.options.padding * 2 + 1, - count = (totalPages >= maxCount) ? maxCount : totalPages; - - renderPaginationItem.call(this, pagination, "first", "«", "first") - ._bgEnableAria(current > 1); - renderPaginationItem.call(this, pagination, "prev", "<", "prev") - ._bgEnableAria(current > 1); - - for (var i = 0; i < count; i++) - { - var pos = i + startWith; - renderPaginationItem.call(this, pagination, pos, pos, "page-" + pos) - ._bgEnableAria()._bgSelectAria(pos === current); - } + e.stopPropagation(); + e.preventDefault(); - if (count === 0) + var $this = $(this), + parent = $this.parent(); + if (!parent.hasClass("active") && !parent.hasClass("disabled")) { - renderPaginationItem.call(this, pagination, 1, 1, "page-" + 1) - ._bgEnableAria(false)._bgSelectAria(); + var commandList = { + first: 1, + prev: that.current - 1, + next: that.current + 1, + last: that.totalPages + }; + var command = $this.data("page"); + that.current = commandList[command] || command; + loadData.call(that); } + $this.trigger("blur"); + }); - renderPaginationItem.call(this, pagination, "next", ">", "next") - ._bgEnableAria(totalPages > current); - renderPaginationItem.call(this, pagination, "last", "»", "last") - ._bgEnableAria(totalPages > current); + list.append(item); + return item; +} - replacePlaceHolder.call(this, paginationItems, pagination); - } - } - } +function renderRowCountSelection(actions) +{ + var that = this, + rowCountList = this.options.rowCount; - function renderPaginationItem(list, page, text, markerCss) + function getText(value) { - var that = this, - tpl = this.options.templates, - css = this.options.css, - values = getParams.call(this, { css: markerCss, text: text, page: page }), - item = $(tpl.paginationItem.resolve(values)) - .on("click" + namespace, getCssSelector(css.paginationButton), function (e) - { - e.stopPropagation(); - e.preventDefault(); - - var $this = $(this), - parent = $this.parent(); - if (!parent.hasClass("active") && !parent.hasClass("disabled")) - { - var commandList = { - first: 1, - prev: that.current - 1, - next: that.current + 1, - last: that.totalPages - }; - var command = $this.data("page"); - that.current = commandList[command] || command; - loadData.call(that); - } - $this.trigger("blur"); - }); - - list.append(item); - return item; + return (value === -1) ? that.options.labels.all : value; } - function renderRowCountSelection(actions) + if ($.isArray(rowCountList)) { - var that = this, - rowCountList = this.options.rowCount; - - function getText(value) - { - return (value === -1) ? that.options.labels.all : value; - } + var css = this.options.css, + tpl = this.options.templates, + dropDown = $(tpl.actionDropDown.resolve(getParams.call(this, { content: getText(this.rowCount) }))), + menuSelector = getCssSelector(css.dropDownMenu), + menuTextSelector = getCssSelector(css.dropDownMenuText), + menuItemsSelector = getCssSelector(css.dropDownMenuItems), + menuItemSelector = getCssSelector(css.dropDownItemButton); - if ($.isArray(rowCountList)) + $.each(rowCountList, function (index, value) { - var css = this.options.css, - tpl = this.options.templates, - dropDown = $(tpl.actionDropDown.resolve(getParams.call(this, { content: getText(this.rowCount) }))), - menuSelector = getCssSelector(css.dropDownMenu), - menuTextSelector = getCssSelector(css.dropDownMenuText), - menuItemsSelector = getCssSelector(css.dropDownMenuItems), - menuItemSelector = getCssSelector(css.dropDownItemButton); + var item = $(tpl.actionDropDownItem.resolve(getParams.call(that, + { text: getText(value), action: value }))) + ._bgSelectAria(value === that.rowCount) + .on("click" + namespace, menuItemSelector, function (e) + { + e.preventDefault(); - $.each(rowCountList, function (index, value) - { - var item = $(tpl.actionDropDownItem.resolve(getParams.call(that, - { text: getText(value), action: value }))) - ._bgSelectAria(value === that.rowCount) - .on("click" + namespace, menuItemSelector, function (e) + var $this = $(this), + newRowCount = $this.data("action"); + if (newRowCount !== that.rowCount) { - e.preventDefault(); - - var $this = $(this), - newRowCount = $this.data("action"); - if (newRowCount !== that.rowCount) + // todo: sophisticated solution needed for calculating which page is selected + that.current = 1; // that.rowCount === -1 ---> All + that.rowCount = newRowCount; + $this.parents(menuItemsSelector).children().each(function () { - // todo: sophisticated solution needed for calculating which page is selected - that.current = 1; // that.rowCount === -1 ---> All - that.rowCount = newRowCount; - $this.parents(menuItemsSelector).children().each(function () - { - var $item = $(this), - currentRowCount = $item.find(menuItemSelector).data("action"); - $item._bgSelectAria(currentRowCount === newRowCount); - }); - $this.parents(menuSelector).find(menuTextSelector).text(getText(newRowCount)); - loadData.call(that); - } - }); - dropDown.find(menuItemsSelector).append(item); - }); - actions.append(dropDown); - } + var $item = $(this), + currentRowCount = $item.find(menuItemSelector).data("action"); + $item._bgSelectAria(currentRowCount === newRowCount); + }); + $this.parents(menuSelector).find(menuTextSelector).text(getText(newRowCount)); + loadData.call(that); + } + }); + dropDown.find(menuItemsSelector).append(item); + }); + actions.append(dropDown); } +} - function renderRows(rows) +function renderRows(rows) +{ + if (rows.length > 0) { - if (rows.length > 0) + var that = this, + css = this.options.css, + tpl = this.options.templates, + tbody = this.element.children("tbody").first(), + allRowsSelected = true, + html = ""; + + $.each(rows, function (index, row) { - var that = this, - css = this.options.css, - tpl = this.options.templates, - tbody = this.element.children("tbody").first(), - allRowsSelected = true, - html = ""; + var cells = "", + rowAttr = " data-row-id=\"" + ((that.identifier == null) ? index : row[that.identifier]) + "\"", + rowCss = ""; - $.each(rows, function (index, row) + if (that.selection) { - var cells = "", - rowAttr = " data-row-id=\"" + ((that.identifier == null) ? index : row[that.identifier]) + "\"", - rowCss = ""; - - if (that.selection) - { - var selected = ($.inArray(row[that.identifier], that.selectedRows) !== -1), - selectBox = tpl.select.resolve(getParams.call(that, - { type: "checkbox", value: row[that.identifier], checked: selected })); - cells += tpl.cell.resolve(getParams.call(that, { content: selectBox, css: css.selectCell })); - allRowsSelected = (allRowsSelected && selected); - if (selected) - { - rowCss += css.selected; - rowAttr += " aria-selected=\"true\""; - } - } - - var status = row.status != null && that.options.statusMapping[row.status]; - if (status) + var selected = ($.inArray(row[that.identifier], that.selectedRows) !== -1), + selectBox = tpl.select.resolve(getParams.call(that, + { type: "checkbox", value: row[that.identifier], checked: selected })); + cells += tpl.cell.resolve(getParams.call(that, { content: selectBox, css: css.selectCell })); + allRowsSelected = (allRowsSelected && selected); + if (selected) { - rowCss += status; + rowCss += css.selected; + rowAttr += " aria-selected=\"true\""; } + } - $.each(that.columns, function (j, column) - { - if (column.visible) - { - var value = ($.isFunction(column.formatter)) ? - column.formatter.call(that, column, row) : - column.converter.to(row[column.id]), - cssClass = (column.cssClass.length > 0) ? " " + column.cssClass : ""; - cells += tpl.cell.resolve(getParams.call(that, { - content: (value == null || value === "") ? " " : value, - css: ((column.align === "right") ? css.right : (column.align === "center") ? - css.center : css.left) + cssClass, - style: (column.width == null) ? "" : "width:" + column.width + ";" })); - } - }); + var status = row.status != null && that.options.statusMapping[row.status]; + if (status) + { + rowCss += status; + } - if (rowCss.length > 0) + $.each(that.columns, function (j, column) + { + if (column.visible) { - rowAttr += " class=\"" + rowCss + "\""; + var value = ($.isFunction(column.formatter)) ? + column.formatter.call(that, column, row) : + column.converter.to(row[column.id]), + cssClass = (column.cssClass.length > 0) ? " " + column.cssClass : ""; + cells += tpl.cell.resolve(getParams.call(that, { + content: (value == null || value === "") ? " " : value, + css: ((column.align === "right") ? css.right : (column.align === "center") ? + css.center : css.left) + cssClass, + style: (column.width == null) ? "" : "width:" + column.width + ";" })); } - html += tpl.row.resolve(getParams.call(that, { attr: rowAttr, cells: cells })); }); - // sets or clears multi selectbox state - that.element.find("thead " + getCssSelector(that.options.css.selectBox)) - .prop("checked", allRowsSelected); + if (rowCss.length > 0) + { + rowAttr += " class=\"" + rowCss + "\""; + } + html += tpl.row.resolve(getParams.call(that, { attr: rowAttr, cells: cells })); + }); + + // sets or clears multi selectbox state + that.element.find("thead " + getCssSelector(that.options.css.selectBox)) + .prop("checked", allRowsSelected); - tbody.html(html); + tbody.html(html); - registerRowEvents.call(this, tbody); - } - else - { - renderNoResultsRow.call(this); - } + registerRowEvents.call(this, tbody); } - - function registerRowEvents(tbody) + else { - var that = this, - selectBoxSelector = getCssSelector(this.options.css.selectBox); - - if (this.selection) - { - tbody.off("click" + namespace, selectBoxSelector) - .on("click" + namespace, selectBoxSelector, function(e) - { - e.stopPropagation(); - - var $this = $(this), - id = that.converter.from($this.val()); + renderNoResultsRow.call(this); + } +} - if ($this.prop("checked")) - { - that.select([id]); - } - else - { - that.deselect([id]); - } - }); - } +function registerRowEvents(tbody) +{ + var that = this, + selectBoxSelector = getCssSelector(this.options.css.selectBox); - tbody.off("click" + namespace, "> tr") - .on("click" + namespace, "> tr", function(e) + if (this.selection && !this.options.rowSelect) + { + tbody.off("click" + namespace, selectBoxSelector) + .on("click" + namespace, selectBoxSelector, function(e) { e.stopPropagation(); var $this = $(this), - id = (that.identifier == null) ? $this.data("row-id") : - that.converter.from($this.data("row-id") + ""), - row = (that.identifier == null) ? that.currentRows[id] : - that.currentRows.first(function (item) { return item[that.identifier] === id; }); + id = that.converter.from($this.val()); - if (that.selection && that.options.rowSelect) + if ($this.prop("checked")) { - if ($this.hasClass(that.options.css.selected)) - { - that.deselect([id]); - } - else - { - that.select([id]); - } + that.select([id]); + } + else + { + that.deselect([id]); } - - that.element.trigger("click" + namespace, [that.columns, row]); }); } - function renderSearchField() - { - if (this.options.navigation !== 0) + tbody.off("click" + namespace, "> tr") + .on("click" + namespace, "> tr", function(e) { - var css = this.options.css, - selector = getCssSelector(css.search), - searchItems = findFooterAndHeaderItems.call(this, selector); + e.stopPropagation(); + + var $this = $(this), + id = (that.identifier == null) ? $this.data("row-id") : + that.converter.from($this.data("row-id") + ""), + row = (that.identifier == null) ? that.currentRows[id] : + that.currentRows.first(function (item) { return item[that.identifier] === id; }); + + if (that.selection && that.options.rowSelect) + { + if ($this.hasClass(that.options.css.selected)) + { + that.deselect([id]); + } + else + { + that.select([id]); + } + } + + that.element.trigger("click" + namespace, [that.columns, row]); + }); +} + +function renderSearchField() +{ + if (this.options.navigation !== 0) + { + var css = this.options.css, + selector = getCssSelector(css.search), + searchItems = findFooterAndHeaderItems.call(this, selector); - if (searchItems.length > 0) + if (searchItems.length > 0) + { + var that = this, + tpl = this.options.templates, + timer = null, // fast keyup detection + currentValue = "", + searchFieldSelector = getCssSelector(css.searchField), + search = $(tpl.search.resolve(getParams.call(this))), + searchField = (search.is(searchFieldSelector)) ? search : + search.find(searchFieldSelector); + + searchField.on("keyup" + namespace, function (e) { - var that = this, - tpl = this.options.templates, - timer = null, // fast keyup detection - currentValue = "", - searchFieldSelector = getCssSelector(css.searchField), - search = $(tpl.search.resolve(getParams.call(this))), - searchField = (search.is(searchFieldSelector)) ? search : - search.find(searchFieldSelector); - - searchField.on("keyup" + namespace, function (e) + e.stopPropagation(); + var newValue = $(this).val(); + if (currentValue !== newValue || (e.which === 13 && newValue !== "")) { - e.stopPropagation(); - var newValue = $(this).val(); - if (currentValue !== newValue || (e.which === 13 && newValue !== "")) + currentValue = newValue; + if (e.which === 13 || newValue.length === 0 || newValue.length >= that.options.searchSettings.characters) { - currentValue = newValue; - if (e.which === 13 || newValue.length === 0 || newValue.length >= that.options.searchSettings.characters) + window.clearTimeout(timer); + timer = window.setTimeout(function () { - window.clearTimeout(timer); - timer = window.setTimeout(function () - { - executeSearch.call(that, newValue); - }, that.options.searchSettings.delay); - } + executeSearch.call(that, newValue); + }, that.options.searchSettings.delay); } - }); + } + }); - replacePlaceHolder.call(this, searchItems, search); - } + replacePlaceHolder.call(this, searchItems, search); } } +} - function executeSearch(phrase) +function executeSearch(phrase) +{ + if (this.searchPhrase !== phrase) { - if (this.searchPhrase !== phrase) - { - this.current = 1; - this.searchPhrase = phrase; - loadData.call(this); - } + this.current = 1; + this.searchPhrase = phrase; + loadData.call(this); } +} - function renderTableHeader() +function renderTableHeader() +{ + var that = this, + headerRow = this.element.find("thead > tr"), + css = this.options.css, + tpl = this.options.templates, + html = "", + sorting = this.options.sorting; + + if (this.selection) { - var that = this, - headerRow = this.element.find("thead > tr"), - css = this.options.css, - tpl = this.options.templates, - html = "", - sorting = this.options.sorting; + var selectBox = (this.options.multiSelect) ? + tpl.select.resolve(getParams.call(that, { type: "checkbox", value: "all" })) : ""; + html += tpl.rawHeaderCell.resolve(getParams.call(that, { content: selectBox, + css: css.selectCell })); + } - if (this.selection) + $.each(this.columns, function (index, column) + { + if (column.visible) { - var selectBox = (this.options.multiSelect) ? - tpl.select.resolve(getParams.call(that, { type: "checkbox", value: "all" })) : ""; - html += tpl.rawHeaderCell.resolve(getParams.call(that, { content: selectBox, - css: css.selectCell })); + var sortOrder = that.sortDictionary[column.id], + iconCss = ((sorting && sortOrder && sortOrder === "asc") ? css.iconUp : + (sorting && sortOrder && sortOrder === "desc") ? css.iconDown : ""), + icon = tpl.icon.resolve(getParams.call(that, { iconCss: iconCss })), + align = column.headerAlign, + cssClass = (column.headerCssClass.length > 0) ? " " + column.headerCssClass : ""; + html += tpl.headerCell.resolve(getParams.call(that, { + column: column, icon: icon, sortable: sorting && column.sortable && css.sortable || "", + css: ((align === "right") ? css.right : (align === "center") ? + css.center : css.left) + cssClass, + style: (column.width == null) ? "" : "width:" + column.width + ";" })); } + }); - $.each(this.columns, function (index, column) - { - if (column.visible) - { - var sortOrder = that.sortDictionary[column.id], - iconCss = ((sorting && sortOrder && sortOrder === "asc") ? css.iconUp : - (sorting && sortOrder && sortOrder === "desc") ? css.iconDown : ""), - icon = tpl.icon.resolve(getParams.call(that, { iconCss: iconCss })), - align = column.headerAlign, - cssClass = (column.headerCssClass.length > 0) ? " " + column.headerCssClass : ""; - html += tpl.headerCell.resolve(getParams.call(that, { - column: column, icon: icon, sortable: sorting && column.sortable && css.sortable || "", - css: ((align === "right") ? css.right : (align === "center") ? - css.center : css.left) + cssClass, - style: (column.width == null) ? "" : "width:" + column.width + ";" })); - } - }); + headerRow.html(html); - headerRow.html(html); + if (sorting) + { + var sortingSelector = getCssSelector(css.sortable); + headerRow.off("click" + namespace, sortingSelector) + .on("click" + namespace, sortingSelector, function (e) + { + e.preventDefault(); - if (sorting) - { - var sortingSelector = getCssSelector(css.sortable); - headerRow.off("click" + namespace, sortingSelector) - .on("click" + namespace, sortingSelector, function (e) - { - e.preventDefault(); + setTableHeaderSortDirection.call(that, $(this)); + sortRows.call(that); + loadData.call(that); + }); + } - setTableHeaderSortDirection.call(that, $(this)); - sortRows.call(that); - loadData.call(that); - }); - } + // todo: create a own function for that piece of code + if (this.selection && this.options.multiSelect) + { + var selectBoxSelector = getCssSelector(css.selectBox); + headerRow.off("click" + namespace, selectBoxSelector) + .on("click" + namespace, selectBoxSelector, function(e) + { + e.stopPropagation(); - // todo: create a own function for that piece of code - if (this.selection && this.options.multiSelect) - { - var selectBoxSelector = getCssSelector(css.selectBox); - headerRow.off("click" + namespace, selectBoxSelector) - .on("click" + namespace, selectBoxSelector, function(e) + if ($(this).prop("checked")) { - e.stopPropagation(); - - if ($(this).prop("checked")) - { - that.select(); - } - else - { - that.deselect(); - } - }); - } + that.select(); + } + else + { + that.deselect(); + } + }); } +} - function setTableHeaderSortDirection(element) - { - var css = this.options.css, - iconSelector = getCssSelector(css.icon), - columnId = element.data("column-id") || element.parents("th").first().data("column-id"), - sortOrder = this.sortDictionary[columnId], - icon = element.find(iconSelector); +function setTableHeaderSortDirection(element) +{ + var css = this.options.css, + iconSelector = getCssSelector(css.icon), + columnId = element.data("column-id") || element.parents("th").first().data("column-id"), + sortOrder = this.sortDictionary[columnId], + icon = element.find(iconSelector); - if (!this.options.multiSort) - { - element.parents("tr").first().find(iconSelector).removeClass(css.iconDown + " " + css.iconUp); - this.sortDictionary = {}; - } + if (!this.options.multiSort) + { + element.parents("tr").first().find(iconSelector).removeClass(css.iconDown + " " + css.iconUp); + this.sortDictionary = {}; + } - if (sortOrder && sortOrder === "asc") - { - this.sortDictionary[columnId] = "desc"; - icon.removeClass(css.iconUp).addClass(css.iconDown); - } - else if (sortOrder && sortOrder === "desc") + if (sortOrder && sortOrder === "asc") + { + this.sortDictionary[columnId] = "desc"; + icon.removeClass(css.iconUp).addClass(css.iconDown); + } + else if (sortOrder && sortOrder === "desc") + { + if (this.options.multiSort) { - if (this.options.multiSort) + var newSort = {}; + for (var key in this.sortDictionary) { - var newSort = {}; - for (var key in this.sortDictionary) + if (key !== columnId) { - if (key !== columnId) - { - newSort[key] = this.sortDictionary[key]; - } + newSort[key] = this.sortDictionary[key]; } - this.sortDictionary = newSort; - icon.removeClass(css.iconDown); - } - else - { - this.sortDictionary[columnId] = "asc"; - icon.removeClass(css.iconDown).addClass(css.iconUp); } + this.sortDictionary = newSort; + icon.removeClass(css.iconDown); } else { this.sortDictionary[columnId] = "asc"; - icon.addClass(css.iconUp); + icon.removeClass(css.iconDown).addClass(css.iconUp); } } - - function replacePlaceHolder(placeholder, element) + else { - placeholder.each(function (index, item) - { - // todo: check how append is implemented. Perhaps cloning here is superfluous. - $(item).before(element.clone(true)).remove(); - }); + this.sortDictionary[columnId] = "asc"; + icon.addClass(css.iconUp); } +} - function showLoading() +function replacePlaceHolder(placeholder, element) +{ + placeholder.each(function (index, item) { - var that = this; + // todo: check how append is implemented. Perhaps cloning here is superfluous. + $(item).before(element.clone(true)).remove(); + }); +} - window.setTimeout(function() +function showLoading() +{ + var that = this; + + window.setTimeout(function() + { + if (that.element._bgAria("busy") === "true") { - if (that.element._bgAria("busy") === "true") + var tpl = that.options.templates, + thead = that.element.children("thead").first(), + tbody = that.element.children("tbody").first(), + firstCell = tbody.find("tr > td").first(), + padding = (that.element.height() - thead.height()) - (firstCell.height() + 20), + count = that.columns.where(isVisible).length; + + if (that.selection) { - var tpl = that.options.templates, - thead = that.element.children("thead").first(), - tbody = that.element.children("tbody").first(), - firstCell = tbody.find("tr > td").first(), - padding = (that.element.height() - thead.height()) - (firstCell.height() + 20), - count = that.columns.where(isVisible).length; - - if (that.selection) - { - count = count + 1; - } - tbody.html(tpl.loading.resolve(getParams.call(that, { columns: count }))); - if (that.rowCount !== -1 && padding > 0) - { - tbody.find("tr > td").css("padding", "20px 0 " + padding + "px"); - } + count = count + 1; } - }, 250); - } + tbody.html(tpl.loading.resolve(getParams.call(that, { columns: count }))); + if (that.rowCount !== -1 && padding > 0) + { + tbody.find("tr > td").css("padding", "20px 0 " + padding + "px"); + } + } + }, 250); +} + +function sortRows() +{ + var sortArray = []; - function sortRows() + function sort(x, y, current) { - var sortArray = []; + current = current || 0; + var next = current + 1, + item = sortArray[current]; - function sort(x, y, current) + function sortOrder(value) { - current = current || 0; - var next = current + 1, - item = sortArray[current]; + return (item.order === "asc") ? value : value * -1; + } - function sortOrder(value) - { - return (item.order === "asc") ? value : value * -1; - } + return (x[item.id] > y[item.id]) ? sortOrder(1) : + (x[item.id] < y[item.id]) ? sortOrder(-1) : + (sortArray.length > next) ? sort(x, y, next) : 0; + } - return (x[item.id] > y[item.id]) ? sortOrder(1) : - (x[item.id] < y[item.id]) ? sortOrder(-1) : - (sortArray.length > next) ? sort(x, y, next) : 0; - } + if (!this.options.ajax) + { + var that = this; - if (!this.options.ajax) + for (var key in this.sortDictionary) { - var that = this; - - for (var key in this.sortDictionary) + if (this.options.multiSort || sortArray.length === 0) { - if (this.options.multiSort || sortArray.length === 0) - { - sortArray.push({ - id: key, - order: this.sortDictionary[key] - }); - } + sortArray.push({ + id: key, + order: this.sortDictionary[key] + }); } + } - if (sortArray.length > 0) - { - this.rows.sort(sort); - } + if (sortArray.length > 0) + { + this.rows.sort(sort); } } + } - // GRID PUBLIC CLASS DEFINITION - // ==================== +// GRID PUBLIC CLASS DEFINITION +// ==================== + +/** + * Represents the jQuery Bootgrid plugin. + * + * @class Grid + * @constructor + * @param element {Object} The corresponding DOM element. + * @param options {Object} The options to override default settings. + * @chainable + **/ +var Grid = function(element, options) +{ + this.element = $(element); + this.origin = this.element.clone(); + this.options = $.extend(true, {}, Grid.defaults, this.element.data(), options); + // overrides rowCount explicitly because deep copy ($.extend) leads to strange behaviour + var rowCount = this.options.rowCount = this.element.data().rowCount || options.rowCount || this.options.rowCount; + this.columns = []; + this.current = 1; + this.currentRows = []; + this.identifier = null; // The first column ID that is marked as identifier + this.selection = false; + this.converter = null; // The converter for the column that is marked as identifier + this.rowCount = ($.isArray(rowCount)) ? rowCount[0] : rowCount; + this.rows = []; + this.searchPhrase = ""; + this.selectedRows = []; + this.sortDictionary = {}; + this.total = 0; + this.totalPages = 0; + this.cachedParams = { + lbl: this.options.labels, + css: this.options.css, + ctx: {} + }; + this.header = null; + this.footer = null; + this.xqr = null; + + // todo: implement cache +}; + +/** + * An object that represents the default settings. + * + * @static + * @class defaults + * @for Grid + * @example + * // Global approach + * $.bootgrid.defaults.selection = true; + * @example + * // Initialization approach + * $("#bootgrid").bootgrid({ selection = true }); + **/ +Grid.defaults = { + navigation: 3, // it's a flag: 0 = none, 1 = top, 2 = bottom, 3 = both (top and bottom) + padding: 2, // page padding (pagination) + columnSelection: true, + rowCount: [10, 25, 50, -1], // rows per page int or array of int (-1 represents "All") /** - * Represents the jQuery Bootgrid plugin. + * Enables row selection (to enable multi selection see also `multiSelect`). Default value is `false`. * - * @class Grid - * @constructor - * @param element {Object} The corresponding DOM element. - * @param options {Object} The options to override default settings. - * @chainable + * @property selection + * @type Boolean + * @default false + * @for defaults + * @since 1.0.0 **/ - var Grid = function(element, options) - { - this.element = $(element); - this.origin = this.element.clone(); - this.options = $.extend(true, {}, Grid.defaults, this.element.data(), options); - // overrides rowCount explicitly because deep copy ($.extend) leads to strange behaviour - var rowCount = this.options.rowCount = this.element.data().rowCount || options.rowCount || this.options.rowCount; - this.columns = []; - this.current = 1; - this.currentRows = []; - this.identifier = null; // The first column ID that is marked as identifier - this.selection = false; - this.converter = null; // The converter for the column that is marked as identifier - this.rowCount = ($.isArray(rowCount)) ? rowCount[0] : rowCount; - this.rows = []; - this.searchPhrase = ""; - this.selectedRows = []; - this.sortDictionary = {}; - this.total = 0; - this.totalPages = 0; - this.cachedParams = { - lbl: this.options.labels, - css: this.options.css, - ctx: {} - }; - this.header = null; - this.footer = null; - this.xqr = null; + selection: false, - // todo: implement cache - }; + /** + * Enables multi selection (`selection` must be set to `true` as well). Default value is `false`. + * + * @property multiSelect + * @type Boolean + * @default false + * @for defaults + * @since 1.0.0 + **/ + multiSelect: false, /** - * An object that represents the default settings. + * Enables entire row click selection (`selection` must be set to `true` as well). Default value is `false`. * - * @static - * @class defaults - * @for Grid - * @example - * // Global approach - * $.bootgrid.defaults.selection = true; - * @example - * // Initialization approach - * $("#bootgrid").bootgrid({ selection = true }); + * @property rowSelect + * @type Boolean + * @default false + * @for defaults + * @since 1.1.0 **/ - Grid.defaults = { - navigation: 3, // it's a flag: 0 = none, 1 = top, 2 = bottom, 3 = both (top and bottom) - padding: 2, // page padding (pagination) - columnSelection: true, - rowCount: [10, 25, 50, -1], // rows per page int or array of int (-1 represents "All") + rowSelect: false, - /** - * Enables row selection (to enable multi selection see also `multiSelect`). Default value is `false`. - * - * @property selection - * @type Boolean - * @default false - * @for defaults - * @since 1.0.0 - **/ - selection: false, + /** + * Defines whether the row selection is saved internally on filtering, paging and sorting + * (even if the selected rows are not visible). + * + * @property keepSelection + * @type Boolean + * @default false + * @for defaults + * @since 1.1.0 + **/ + keepSelection: false, - /** - * Enables multi selection (`selection` must be set to `true` as well). Default value is `false`. - * - * @property multiSelect - * @type Boolean - * @default false - * @for defaults - * @since 1.0.0 - **/ - multiSelect: false, + highlightRows: false, // highlights new rows (find the page of the first new row) + sorting: true, + multiSort: false, + /** + * General search settings to configure the search field behaviour. + * + * @property searchSettings + * @type Object + * @for defaults + * @since 1.2.0 + **/ + searchSettings: { /** - * Enables entire row click selection (`selection` must be set to `true` as well). Default value is `false`. + * The time in milliseconds to wait before search gets executed. * - * @property rowSelect - * @type Boolean - * @default false - * @for defaults - * @since 1.1.0 + * @property delay + * @type Number + * @default 250 + * @for searchSettings **/ - rowSelect: false, - + delay: 250, + /** - * Defines whether the row selection is saved internally on filtering, paging and sorting - * (even if the selected rows are not visible). + * The characters to type before the search gets executed. * - * @property keepSelection - * @type Boolean - * @default false - * @for defaults - * @since 1.1.0 + * @property characters + * @type Number + * @default 1 + * @for searchSettings **/ - keepSelection: false, + characters: 1 + }, - highlightRows: false, // highlights new rows (find the page of the first new row) - sorting: true, - multiSort: false, + /** + * Defines whether the data shall be loaded via an asynchronous HTTP (Ajax) request. + * + * @property ajax + * @type Boolean + * @default false + * @for defaults + **/ + ajax: false, + /** + * Ajax request settings that shall be used for server-side communication. + * All setting except data, error, success and url can be overridden. + * For the full list of settings go to http://api.jquery.com/jQuery.ajax/. + * + * @property ajaxSettings + * @type Object + * @for defaults + * @since 1.2.0 + **/ + ajaxSettings: { /** - * General search settings to configure the search field behaviour. + * Specifies the HTTP method which shall be used when sending data to the server. + * Go to http://api.jquery.com/jQuery.ajax/ for more details. + * This setting is overriden for backward compatibility. * - * @property searchSettings - * @type Object - * @for defaults - * @since 1.2.0 + * @property method + * @type String + * @default "POST" + * @for ajaxSettings **/ - searchSettings: { - /** - * The time in milliseconds to wait before search gets executed. - * - * @property delay - * @type Number - * @default 250 - * @for searchSettings - **/ - delay: 250, - - /** - * The characters to type before the search gets executed. - * - * @property characters - * @type Number - * @default 1 - * @for searchSettings - **/ - characters: 1 - }, + method: "POST" + }, - /** - * Defines whether the data shall be loaded via an asynchronous HTTP (Ajax) request. - * - * @property ajax - * @type Boolean - * @default false - * @for defaults - **/ - ajax: false, + /** + * Enriches the request object with additional properties. Either a `PlainObject` or a `Function` + * that returns a `PlainObject` can be passed. Default value is `{}`. + * + * @property post + * @type Object|Function + * @default function (request) { return request; } + * @for defaults + * @deprecated Use instead `requestHandler` + **/ + post: {}, // or use function () { return {}; } (reserved properties are "current", "rowCount", "sort" and "searchPhrase") - /** - * Ajax request settings that shall be used for server-side communication. - * All setting except data, error, success and url can be overridden. - * For the full list of settings go to http://api.jquery.com/jQuery.ajax/. - * - * @property ajaxSettings - * @type Object - * @for defaults - * @since 1.2.0 - **/ - ajaxSettings: { - /** - * Specifies the HTTP method which shall be used when sending data to the server. - * Go to http://api.jquery.com/jQuery.ajax/ for more details. - * This setting is overriden for backward compatibility. - * - * @property method - * @type String - * @default "POST" - * @for ajaxSettings - **/ - method: "POST" - }, + /** + * Sets the data URL to a data service (e.g. a REST service). Either a `String` or a `Function` + * that returns a `String` can be passed. Default value is `""`. + * + * @property url + * @type String|Function + * @default "" + * @for defaults + **/ + url: "", // or use function () { return ""; } - /** - * Enriches the request object with additional properties. Either a `PlainObject` or a `Function` - * that returns a `PlainObject` can be passed. Default value is `{}`. - * - * @property post - * @type Object|Function - * @default function (request) { return request; } - * @for defaults - * @deprecated Use instead `requestHandler` - **/ - post: {}, // or use function () { return {}; } (reserved properties are "current", "rowCount", "sort" and "searchPhrase") + /** + * Defines whether the search is case sensitive or insensitive. + * + * @property caseSensitive + * @type Boolean + * @default true + * @for defaults + * @since 1.1.0 + **/ + caseSensitive: true, - /** - * Sets the data URL to a data service (e.g. a REST service). Either a `String` or a `Function` - * that returns a `String` can be passed. Default value is `""`. - * - * @property url - * @type String|Function - * @default "" - * @for defaults - **/ - url: "", // or use function () { return ""; } + // note: The following properties should not be used via data-api attributes - /** - * Defines whether the search is case sensitive or insensitive. - * - * @property caseSensitive - * @type Boolean - * @default true - * @for defaults - * @since 1.1.0 - **/ - caseSensitive: true, + /** + * Transforms the JSON request object in what ever is needed on the server-side implementation. + * + * @property requestHandler + * @type Function + * @default function (request) { return request; } + * @for defaults + * @since 1.1.0 + **/ + requestHandler: function (request) { return request; }, + + /** + * Transforms the response object into the expected JSON response object. + * + * @property responseHandler + * @type Function + * @default function (response) { return response; } + * @for defaults + * @since 1.1.0 + **/ + responseHandler: function (response) { return response; }, + + /** + * A list of converters. + * + * @property converters + * @type Object + * @for defaults + * @since 1.0.0 + **/ + converters: { + numeric: { + from: function (value) { return +value; }, // converts from string to numeric + to: function (value) { return value + ""; } // converts from numeric to string + }, + string: { + // default converter + from: function (value) { return value; }, + to: function (value) { return value; } + } + }, - // note: The following properties should not be used via data-api attributes + /** + * Contains all css classes. + * + * @property css + * @type Object + * @for defaults + **/ + css: { + actions: "actions btn-group", // must be a unique class name or constellation of class names within the header and footer + center: "text-center", + columnHeaderAnchor: "column-header-anchor", // must be a unique class name or constellation of class names within the column header cell + columnHeaderText: "text", + dropDownItem: "dropdown-item", // must be a unique class name or constellation of class names within the actionDropDown, + dropDownItemButton: "dropdown-item-button", // must be a unique class name or constellation of class names within the actionDropDown + dropDownItemCheckbox: "dropdown-item-checkbox", // must be a unique class name or constellation of class names within the actionDropDown + dropDownMenu: "dropdown btn-group", // must be a unique class name or constellation of class names within the actionDropDown + dropDownMenuItems: "dropdown-menu pull-right", // must be a unique class name or constellation of class names within the actionDropDown + dropDownMenuText: "dropdown-text", // must be a unique class name or constellation of class names within the actionDropDown + footer: "bootgrid-footer container-fluid", + header: "bootgrid-header container-fluid", + icon: "icon glyphicon", + iconColumns: "glyphicon-th-list", + iconDown: "glyphicon-chevron-down", + iconRefresh: "glyphicon-refresh", + iconSearch: "glyphicon-search", + iconUp: "glyphicon-chevron-up", + infos: "infos", // must be a unique class name or constellation of class names within the header and footer, + left: "text-left", + pagination: "pagination", // must be a unique class name or constellation of class names within the header and footer + paginationButton: "button", // must be a unique class name or constellation of class names within the pagination /** - * Transforms the JSON request object in what ever is needed on the server-side implementation. + * CSS class to select the parent div which activates responsive mode. * - * @property requestHandler - * @type Function - * @default function (request) { return request; } - * @for defaults + * @property responsiveTable + * @type String + * @default "table-responsive" + * @for css * @since 1.1.0 **/ - requestHandler: function (request) { return request; }, + responsiveTable: "table-responsive", + + right: "text-right", + search: "search form-group", // must be a unique class name or constellation of class names within the header and footer + searchField: "search-field form-control", + selectBox: "select-box", // must be a unique class name or constellation of class names within the entire table + selectCell: "select-cell", // must be a unique class name or constellation of class names within the entire table /** - * Transforms the response object into the expected JSON response object. + * CSS class to highlight selected rows. * - * @property responseHandler - * @type Function - * @default function (response) { return response; } - * @for defaults + * @property selected + * @type String + * @default "active" + * @for css * @since 1.1.0 **/ - responseHandler: function (response) { return response; }, + selected: "active", - /** - * A list of converters. - * - * @property converters - * @type Object - * @for defaults - * @since 1.0.0 - **/ - converters: { - numeric: { - from: function (value) { return +value; }, // converts from string to numeric - to: function (value) { return value + ""; } // converts from numeric to string - }, - string: { - // default converter - from: function (value) { return value; }, - to: function (value) { return value; } - } - }, + sortable: "sortable", + table: "bootgrid-table table" + }, - /** - * Contains all css classes. - * - * @property css - * @type Object - * @for defaults - **/ - css: { - actions: "actions btn-group", // must be a unique class name or constellation of class names within the header and footer - center: "text-center", - columnHeaderAnchor: "column-header-anchor", // must be a unique class name or constellation of class names within the column header cell - columnHeaderText: "text", - dropDownItem: "dropdown-item", // must be a unique class name or constellation of class names within the actionDropDown, - dropDownItemButton: "dropdown-item-button", // must be a unique class name or constellation of class names within the actionDropDown - dropDownItemCheckbox: "dropdown-item-checkbox", // must be a unique class name or constellation of class names within the actionDropDown - dropDownMenu: "dropdown btn-group", // must be a unique class name or constellation of class names within the actionDropDown - dropDownMenuItems: "dropdown-menu pull-right", // must be a unique class name or constellation of class names within the actionDropDown - dropDownMenuText: "dropdown-text", // must be a unique class name or constellation of class names within the actionDropDown - footer: "bootgrid-footer container-fluid", - header: "bootgrid-header container-fluid", - icon: "icon glyphicon", - iconColumns: "glyphicon-th-list", - iconDown: "glyphicon-chevron-down", - iconRefresh: "glyphicon-refresh", - iconSearch: "glyphicon-search", - iconUp: "glyphicon-chevron-up", - infos: "infos", // must be a unique class name or constellation of class names within the header and footer, - left: "text-left", - pagination: "pagination", // must be a unique class name or constellation of class names within the header and footer - paginationButton: "button", // must be a unique class name or constellation of class names within the pagination - - /** - * CSS class to select the parent div which activates responsive mode. - * - * @property responsiveTable - * @type String - * @default "table-responsive" - * @for css - * @since 1.1.0 - **/ - responsiveTable: "table-responsive", - - right: "text-right", - search: "search form-group", // must be a unique class name or constellation of class names within the header and footer - searchField: "search-field form-control", - selectBox: "select-box", // must be a unique class name or constellation of class names within the entire table - selectCell: "select-cell", // must be a unique class name or constellation of class names within the entire table - - /** - * CSS class to highlight selected rows. - * - * @property selected - * @type String - * @default "active" - * @for css - * @since 1.1.0 - **/ - selected: "active", - - sortable: "sortable", - table: "bootgrid-table table" - }, + /** + * A dictionary of formatters. + * + * @property formatters + * @type Object + * @for defaults + * @since 1.0.0 + **/ + formatters: {}, + + /** + * Contains all labels. + * + * @property labels + * @type Object + * @for defaults + **/ + labels: { + all: "All", + infos: "Showing {{ctx.start}} to {{ctx.end}} of {{ctx.total}} entries", + loading: "Loading...", + noResults: "No results found!", + refresh: "Refresh", + search: "Search" + }, + /** + * Specifies the mapping between status and contextual classes to color rows. + * + * @property statusMapping + * @type Object + * @for defaults + * @since 1.2.0 + **/ + statusMapping: { /** - * A dictionary of formatters. + * Specifies a successful or positive action. * - * @property formatters - * @type Object - * @for defaults - * @since 1.0.0 + * @property 0 + * @type String + * @for statusMapping **/ - formatters: {}, + 0: "success", /** - * Contains all labels. + * Specifies a neutral informative change or action. * - * @property labels - * @type Object - * @for defaults + * @property 1 + * @type String + * @for statusMapping **/ - labels: { - all: "All", - infos: "Showing {{ctx.start}} to {{ctx.end}} of {{ctx.total}} entries", - loading: "Loading...", - noResults: "No results found!", - refresh: "Refresh", - search: "Search" - }, + 1: "info", /** - * Specifies the mapping between status and contextual classes to color rows. + * Specifies a warning that might need attention. * - * @property statusMapping - * @type Object - * @for defaults - * @since 1.2.0 + * @property 2 + * @type String + * @for statusMapping **/ - statusMapping: { - /** - * Specifies a successful or positive action. - * - * @property 0 - * @type String - * @for statusMapping - **/ - 0: "success", - - /** - * Specifies a neutral informative change or action. - * - * @property 1 - * @type String - * @for statusMapping - **/ - 1: "info", - - /** - * Specifies a warning that might need attention. - * - * @property 2 - * @type String - * @for statusMapping - **/ - 2: "warning", - - /** - * Specifies a dangerous or potentially negative action. - * - * @property 3 - * @type String - * @for statusMapping - **/ - 3: "danger" - }, - + 2: "warning", + /** - * Contains all templates. + * Specifies a dangerous or potentially negative action. * - * @property templates - * @type Object - * @for defaults + * @property 3 + * @type String + * @for statusMapping **/ - templates: { - actionButton: "", - actionDropDown: "
          ", - actionDropDownItem: "
        • {{ctx.text}}
        • ", - actionDropDownCheckboxItem: "
        • ", - actions: "
          ", - body: "", - cell: "{{ctx.content}}", - footer: "

          ", - header: "

          ", - headerCell: "{{ctx.column.text}}{{ctx.icon}}", - icon: "", - infos: "
          {{lbl.infos}}
          ", - loading: "{{lbl.loading}}", - noResults: "{{lbl.noResults}}", - pagination: "
            ", - paginationItem: "
          • {{ctx.text}}
          • ", - rawHeaderCell: "{{ctx.content}}", // Used for the multi select box - row: "{{ctx.cells}}", - search: "
            ", - select: "" - } - }; + 3: "danger" + }, /** - * Appends rows. + * Contains all templates. * - * @method append - * @param rows {Array} An array of rows to append - * @chainable + * @property templates + * @type Object + * @for defaults **/ - Grid.prototype.append = function(rows) + templates: { + actionButton: "", + actionDropDown: "
              ", + actionDropDownItem: "
            • {{ctx.text}}
            • ", + actionDropDownCheckboxItem: "
            • ", + actions: "
              ", + body: "", + cell: "{{ctx.content}}", + footer: "

              ", + header: "

              ", + headerCell: "{{ctx.column.text}}{{ctx.icon}}", + icon: "", + infos: "
              {{lbl.infos}}
              ", + loading: "{{lbl.loading}}", + noResults: "{{lbl.noResults}}", + pagination: "
                ", + paginationItem: "
              • {{ctx.text}}
              • ", + rawHeaderCell: "{{ctx.content}}", // Used for the multi select box + row: "{{ctx.cells}}", + search: "
                ", + select: "" + } +}; + +/** + * Appends rows. + * + * @method append + * @param rows {Array} An array of rows to append + * @chainable + **/ +Grid.prototype.append = function(rows) +{ + if (this.options.ajax) { - if (this.options.ajax) + // todo: implement ajax PUT + } + else + { + var appendedRows = []; + for (var i = 0; i < rows.length; i++) { - // todo: implement ajax PUT - } - else - { - var appendedRows = []; - for (var i = 0; i < rows.length; i++) + if (appendRow.call(this, rows[i])) { - if (appendRow.call(this, rows[i])) - { - appendedRows.push(rows[i]); - } + appendedRows.push(rows[i]); } - sortRows.call(this); - highlightAppendedRows.call(this, appendedRows); - loadData.call(this); - this.element.trigger("appended" + namespace, [appendedRows]); - } - - return this; - }; - - /** - * Removes all rows. - * - * @method clear - * @chainable - **/ - Grid.prototype.clear = function() - { - if (this.options.ajax) - { - // todo: implement ajax POST - } - else - { - var removedRows = $.extend([], this.rows); - this.rows = []; - this.current = 1; - this.total = 0; - loadData.call(this); - this.element.trigger("cleared" + namespace, [removedRows]); } + sortRows.call(this); + highlightAppendedRows.call(this, appendedRows); + loadData.call(this); + this.element.trigger("appended" + namespace, [appendedRows]); + } - return this; - }; + return this; +}; - /** - * Removes the control functionality completely and transforms the current state to the initial HTML structure. - * - * @method destroy - * @chainable - **/ - Grid.prototype.destroy = function() +/** + * Removes all rows. + * + * @method clear + * @chainable + **/ +Grid.prototype.clear = function() +{ + if (this.options.ajax) { - // todo: this method has to be optimized (the complete initial state must be restored) - $(window).off(namespace); - if (this.options.navigation & 1) - { - this.header.remove(); - } - if (this.options.navigation & 2) - { - this.footer.remove(); - } - this.element.before(this.origin).remove(); - - return this; - }; - - /** - * Resets the state and reloads rows. - * - * @method reload - * @chainable - **/ - Grid.prototype.reload = function() + // todo: implement ajax POST + } + else { - this.current = 1; // reset + var removedRows = $.extend([], this.rows); + this.rows = []; + this.current = 1; + this.total = 0; loadData.call(this); + this.element.trigger("cleared" + namespace, [removedRows]); + } - return this; - }; + return this; +}; - /** - * Removes rows by ids. Removes selected rows if no ids are provided. - * - * @method remove - * @param [rowsIds] {Array} An array of rows ids to remove - * @chainable - **/ - Grid.prototype.remove = function(rowIds) +/** + * Removes the control functionality completely and transforms the current state to the initial HTML structure. + * + * @method destroy + * @chainable + **/ +Grid.prototype.destroy = function() +{ + // todo: this method has to be optimized (the complete initial state must be restored) + $(window).off(namespace); + if (this.options.navigation & 1) + { + this.header.remove(); + } + if (this.options.navigation & 2) + { + this.footer.remove(); + } + this.element.before(this.origin).remove(); + + return this; +}; + +/** + * Resets the state and reloads rows. + * + * @method reload + * @chainable + **/ +Grid.prototype.reload = function() +{ + this.current = 1; // reset + loadData.call(this); + + return this; +}; + +/** + * Removes rows by ids. Removes selected rows if no ids are provided. + * + * @method remove + * @param [rowsIds] {Array} An array of rows ids to remove + * @chainable + **/ +Grid.prototype.remove = function(rowIds) +{ + if (this.identifier != null) { - if (this.identifier != null) + var that = this; + + if (this.options.ajax) + { + // todo: implement ajax DELETE + } + else { - var that = this; + rowIds = rowIds || this.selectedRows; + var id, + removedRows = []; - if (this.options.ajax) + for (var i = 0; i < rowIds.length; i++) { - // todo: implement ajax DELETE - } - else - { - rowIds = rowIds || this.selectedRows; - var id, - removedRows = []; + id = rowIds[i]; - for (var i = 0; i < rowIds.length; i++) + for (var j = 0; j < this.rows.length; j++) { - id = rowIds[i]; - - for (var j = 0; j < this.rows.length; j++) + if (this.rows[j][this.identifier] === id) { - if (this.rows[j][this.identifier] === id) - { - removedRows.push(this.rows[j]); - this.rows.splice(j, 1); - break; - } + removedRows.push(this.rows[j]); + this.rows.splice(j, 1); + break; } } - - this.current = 1; // reset - loadData.call(this); - this.element.trigger("removed" + namespace, [removedRows]); } + + this.current = 1; // reset + loadData.call(this); + this.element.trigger("removed" + namespace, [removedRows]); } + } - return this; - }; + return this; +}; + +/** + * Searches in all rows for a specific phrase (but only in visible cells). + * The search filter will be reseted, if no argument is provided. + * + * @method search + * @param [phrase] {String} The phrase to search for + * @chainable + **/ +Grid.prototype.search = function(phrase) +{ + phrase = phrase || ""; - /** - * Searches in all rows for a specific phrase (but only in visible cells). - * The search filter will be reseted, if no argument is provided. - * - * @method search - * @param [phrase] {String} The phrase to search for - * @chainable - **/ - Grid.prototype.search = function(phrase) + if (this.searchPhrase !== phrase) { - phrase = phrase || ""; - - if (this.searchPhrase !== phrase) - { - var selector = getCssSelector(this.options.css.searchField), - searchFields = findFooterAndHeaderItems.call(this, selector); - searchFields.val(phrase); - } + var selector = getCssSelector(this.options.css.searchField), + searchFields = findFooterAndHeaderItems.call(this, selector); + searchFields.val(phrase); + } - executeSearch.call(this, phrase); + executeSearch.call(this, phrase); - return this; - }; + return this; +}; - /** - * Selects rows by ids. Selects all visible rows if no ids are provided. - * In server-side scenarios only visible rows are selectable. - * - * @method select - * @param [rowsIds] {Array} An array of rows ids to select - * @chainable - **/ - Grid.prototype.select = function(rowIds) +/** + * Selects rows by ids. Selects all visible rows if no ids are provided. + * In server-side scenarios only visible rows are selectable. + * + * @method select + * @param [rowsIds] {Array} An array of rows ids to select + * @chainable + **/ +Grid.prototype.select = function(rowIds) +{ + if (this.selection) { - if (this.selection) - { - rowIds = rowIds || this.currentRows.propValues(this.identifier); + rowIds = rowIds || this.currentRows.propValues(this.identifier); - var id, i, - selectedRows = []; + var id, i, + selectedRows = []; - while (rowIds.length > 0 && !(!this.options.multiSelect && selectedRows.length === 1)) + while (rowIds.length > 0 && !(!this.options.multiSelect && selectedRows.length === 1)) + { + id = rowIds.pop(); + if ($.inArray(id, this.selectedRows) === -1) { - id = rowIds.pop(); - if ($.inArray(id, this.selectedRows) === -1) + for (i = 0; i < this.currentRows.length; i++) { - for (i = 0; i < this.currentRows.length; i++) + if (this.currentRows[i][this.identifier] === id) { - if (this.currentRows[i][this.identifier] === id) - { - selectedRows.push(this.currentRows[i]); - this.selectedRows.push(id); - break; - } + selectedRows.push(this.currentRows[i]); + this.selectedRows.push(id); + break; } } } + } - if (selectedRows.length > 0) - { - var selectBoxSelector = getCssSelector(this.options.css.selectBox), - selectMultiSelectBox = this.selectedRows.length >= this.currentRows.length; - - i = 0; - while (!this.options.keepSelection && selectMultiSelectBox && i < this.currentRows.length) - { - selectMultiSelectBox = ($.inArray(this.currentRows[i++][this.identifier], this.selectedRows) !== -1); - } - this.element.find("thead " + selectBoxSelector).prop("checked", selectMultiSelectBox); + if (selectedRows.length > 0) + { + var selectBoxSelector = getCssSelector(this.options.css.selectBox), + selectMultiSelectBox = this.selectedRows.length >= this.currentRows.length; - if (!this.options.multiSelect) - { - this.element.find("tbody > tr " + selectBoxSelector + ":checked") - .trigger("click" + namespace); - } + i = 0; + while (!this.options.keepSelection && selectMultiSelectBox && i < this.currentRows.length) + { + selectMultiSelectBox = ($.inArray(this.currentRows[i++][this.identifier], this.selectedRows) !== -1); + } + this.element.find("thead " + selectBoxSelector).prop("checked", selectMultiSelectBox); - for (i = 0; i < this.selectedRows.length; i++) - { - this.element.find("tbody > tr[data-row-id=\"" + this.selectedRows[i] + "\"]") - .addClass(this.options.css.selected)._bgAria("selected", "true") - .find(selectBoxSelector).prop("checked", true); - } + if (!this.options.multiSelect) + { + this.element.find("tbody > tr " + selectBoxSelector + ":checked") + .trigger("click" + namespace); + } - this.element.trigger("selected" + namespace, [selectedRows]); + for (i = 0; i < this.selectedRows.length; i++) + { + this.element.find("tbody > tr[data-row-id=\"" + this.selectedRows[i] + "\"]") + .addClass(this.options.css.selected)._bgAria("selected", "true") + .find(selectBoxSelector).prop("checked", true); } - } - return this; - }; + this.element.trigger("selected" + namespace, [selectedRows]); + } + } - /** - * Deselects rows by ids. Deselects all visible rows if no ids are provided. - * In server-side scenarios only visible rows are deselectable. - * - * @method deselect - * @param [rowsIds] {Array} An array of rows ids to deselect - * @chainable - **/ - Grid.prototype.deselect = function(rowIds) + return this; +}; + +/** + * Deselects rows by ids. Deselects all visible rows if no ids are provided. + * In server-side scenarios only visible rows are deselectable. + * + * @method deselect + * @param [rowsIds] {Array} An array of rows ids to deselect + * @chainable + **/ +Grid.prototype.deselect = function(rowIds) +{ + if (this.selection) { - if (this.selection) - { - rowIds = rowIds || this.currentRows.propValues(this.identifier); + rowIds = rowIds || this.currentRows.propValues(this.identifier); - var id, i, pos, - deselectedRows = []; + var id, i, pos, + deselectedRows = []; - while (rowIds.length > 0) + while (rowIds.length > 0) + { + id = rowIds.pop(); + pos = $.inArray(id, this.selectedRows); + if (pos !== -1) { - id = rowIds.pop(); - pos = $.inArray(id, this.selectedRows); - if (pos !== -1) + for (i = 0; i < this.currentRows.length; i++) { - for (i = 0; i < this.currentRows.length; i++) + if (this.currentRows[i][this.identifier] === id) { - if (this.currentRows[i][this.identifier] === id) - { - deselectedRows.push(this.currentRows[i]); - this.selectedRows.splice(pos, 1); - break; - } + deselectedRows.push(this.currentRows[i]); + this.selectedRows.splice(pos, 1); + break; } } } - - if (deselectedRows.length > 0) - { - var selectBoxSelector = getCssSelector(this.options.css.selectBox); - - this.element.find("thead " + selectBoxSelector).prop("checked", false); - for (i = 0; i < deselectedRows.length; i++) - { - this.element.find("tbody > tr[data-row-id=\"" + deselectedRows[i][this.identifier] + "\"]") - .removeClass(this.options.css.selected)._bgAria("selected", "false") - .find(selectBoxSelector).prop("checked", false); - } - - this.element.trigger("deselected" + namespace, [deselectedRows]); - } } - return this; - }; + if (deselectedRows.length > 0) + { + var selectBoxSelector = getCssSelector(this.options.css.selectBox); - /** - * Sorts the rows by a given sort descriptor dictionary. - * The sort filter will be reseted, if no argument is provided. - * - * @method sort - * @param [dictionary] {Object} A sort descriptor dictionary that contains the sort information - * @chainable - **/ - Grid.prototype.sort = function(dictionary) - { - var values = (dictionary) ? $.extend({}, dictionary) : {}; + this.element.find("thead " + selectBoxSelector).prop("checked", false); + for (i = 0; i < deselectedRows.length; i++) + { + this.element.find("tbody > tr[data-row-id=\"" + deselectedRows[i][this.identifier] + "\"]") + .removeClass(this.options.css.selected)._bgAria("selected", "false") + .find(selectBoxSelector).prop("checked", false); + } - if (values === this.sortDictionary) - { - return this; + this.element.trigger("deselected" + namespace, [deselectedRows]); } + } - this.sortDictionary = values; - renderTableHeader.call(this); - sortRows.call(this); - loadData.call(this); + return this; +}; + +/** + * Sorts the rows by a given sort descriptor dictionary. + * The sort filter will be reseted, if no argument is provided. + * + * @method sort + * @param [dictionary] {Object} A sort descriptor dictionary that contains the sort information + * @chainable + **/ +Grid.prototype.sort = function(dictionary) +{ + var values = (dictionary) ? $.extend({}, dictionary) : {}; + if (values === this.sortDictionary) + { return this; - }; + } - /** - * Gets a list of the column settings. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getColumnSettings - * @return {Array} Returns a list of the column settings. - * @since 1.2.0 - **/ - Grid.prototype.getColumnSettings = function() - { - return $.merge([], this.columns); + this.sortDictionary = values; + renderTableHeader.call(this); + sortRows.call(this); + loadData.call(this); + + return this; +}; + +/** + * Gets a list of the column settings. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getColumnSettings + * @return {Array} Returns a list of the column settings. + * @since 1.2.0 + **/ +Grid.prototype.getColumnSettings = function() +{ + return $.merge([], this.columns); +}; + +/** + * Gets the current page index. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getCurrentPage + * @return {Number} Returns the current page index. + * @since 1.2.0 + **/ +Grid.prototype.getCurrentPage = function() +{ + return this.current; +}; + +/** + * Gets the current rows. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getCurrentPage + * @return {Array} Returns the current rows. + * @since 1.2.0 + **/ +Grid.prototype.getCurrentRows = function() +{ + return $.merge([], this.currentRows); +}; + +/** + * Gets a number represents the row count per page. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getRowCount + * @return {Number} Returns the row count per page. + * @since 1.2.0 + **/ +Grid.prototype.getRowCount = function() +{ + return this.rowCount; +}; + +/** + * Gets the actual search phrase. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getSearchPhrase + * @return {String} Returns the actual search phrase. + * @since 1.2.0 + **/ +Grid.prototype.getSearchPhrase = function() +{ + return this.searchPhrase; +}; + +/** + * Gets the complete list of currently selected rows. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getSelectedRows + * @return {Array} Returns all selected rows. + * @since 1.2.0 + **/ +Grid.prototype.getSelectedRows = function() +{ + return $.merge([], this.selectedRows); +}; + +/** + * Gets the sort dictionary which represents the state of column sorting. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getSortDictionary + * @return {Object} Returns the sort dictionary. + * @since 1.2.0 + **/ +Grid.prototype.getSortDictionary = function() +{ + return $.extend({}, this.sortDictionary); +}; + +/** + * Gets a number represents the total page count. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getTotalPageCount + * @return {Number} Returns the total page count. + * @since 1.2.0 + **/ +Grid.prototype.getTotalPageCount = function() +{ + return this.totalPages; +}; + +/** + * Gets a number represents the total row count. + * This method returns only for the first grid instance a value. + * Therefore be sure that only one grid instance is catched by your selector. + * + * @method getTotalRowCount + * @return {Number} Returns the total row count. + * @since 1.2.0 + **/ +Grid.prototype.getTotalRowCount = function() +{ + return this.total; }; - /** - * Gets the current page index. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getCurrentPage - * @return {Number} Returns the current page index. - * @since 1.2.0 - **/ - Grid.prototype.getCurrentPage = function() - { - return this.current; - }; +// GRID COMMON TYPE EXTENSIONS +// ============ - /** - * Gets the current rows. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getCurrentPage - * @return {Array} Returns the current rows. - * @since 1.2.0 - **/ - Grid.prototype.getCurrentRows = function() +$.fn.extend({ + _bgAria: function (name, value) { - return $.merge([], this.currentRows); - }; + return (value) ? this.attr("aria-" + name, value) : this.attr("aria-" + name); + }, - /** - * Gets a number represents the row count per page. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getRowCount - * @return {Number} Returns the row count per page. - * @since 1.2.0 - **/ - Grid.prototype.getRowCount = function() + _bgBusyAria: function(busy) { - return this.rowCount; - }; + return (busy == null || busy) ? + this._bgAria("busy", "true") : + this._bgAria("busy", "false"); + }, - /** - * Gets the actual search phrase. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getSearchPhrase - * @return {String} Returns the actual search phrase. - * @since 1.2.0 - **/ - Grid.prototype.getSearchPhrase = function() + _bgRemoveAria: function (name) { - return this.searchPhrase; - }; + return this.removeAttr("aria-" + name); + }, - /** - * Gets the complete list of currently selected rows. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getSelectedRows - * @return {Array} Returns all selected rows. - * @since 1.2.0 - **/ - Grid.prototype.getSelectedRows = function() + _bgEnableAria: function (enable) { - return $.merge([], this.selectedRows); - }; + return (enable == null || enable) ? + this.removeClass("disabled")._bgAria("disabled", "false") : + this.addClass("disabled")._bgAria("disabled", "true"); + }, - /** - * Gets the sort dictionary which represents the state of column sorting. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getSortDictionary - * @return {Object} Returns the sort dictionary. - * @since 1.2.0 - **/ - Grid.prototype.getSortDictionary = function() + _bgEnableField: function (enable) { - return $.extend({}, this.sortDictionary); - }; + return (enable == null || enable) ? + this.removeAttr("disabled") : + this.attr("disabled", "disable"); + }, - /** - * Gets a number represents the total page count. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getTotalPageCount - * @return {Number} Returns the total page count. - * @since 1.2.0 - **/ - Grid.prototype.getTotalPageCount = function() + _bgShowAria: function (show) { - return this.totalPages; - }; + return (show == null || show) ? + this.show()._bgAria("hidden", "false") : + this.hide()._bgAria("hidden", "true"); + }, - /** - * Gets a number represents the total row count. - * This method returns only for the first grid instance a value. - * Therefore be sure that only one grid instance is catched by your selector. - * - * @method getTotalRowCount - * @return {Number} Returns the total row count. - * @since 1.2.0 - **/ - Grid.prototype.getTotalRowCount = function() + _bgSelectAria: function (select) { - return this.total; - }; - - // GRID COMMON TYPE EXTENSIONS - // ============ - - $.fn.extend({ - _bgAria: function (name, value) - { - return (value) ? this.attr("aria-" + name, value) : this.attr("aria-" + name); - }, - - _bgBusyAria: function(busy) - { - return (busy == null || busy) ? - this._bgAria("busy", "true") : - this._bgAria("busy", "false"); - }, - - _bgRemoveAria: function (name) - { - return this.removeAttr("aria-" + name); - }, - - _bgEnableAria: function (enable) - { - return (enable == null || enable) ? - this.removeClass("disabled")._bgAria("disabled", "false") : - this.addClass("disabled")._bgAria("disabled", "true"); - }, + return (select == null || select) ? + this.addClass("active")._bgAria("selected", "true") : + this.removeClass("active")._bgAria("selected", "false"); + }, - _bgEnableField: function (enable) - { - return (enable == null || enable) ? - this.removeAttr("disabled") : - this.attr("disabled", "disable"); - }, - - _bgShowAria: function (show) - { - return (show == null || show) ? - this.show()._bgAria("hidden", "false") : - this.hide()._bgAria("hidden", "true"); - }, - - _bgSelectAria: function (select) - { - return (select == null || select) ? - this.addClass("active")._bgAria("selected", "true") : - this.removeClass("active")._bgAria("selected", "false"); - }, + _bgId: function (id) + { + return (id) ? this.attr("id", id) : this.attr("id"); + } +}); - _bgId: function (id) +if (!String.prototype.resolve) +{ + var formatter = { + "checked": function(value) { - return (id) ? this.attr("id", id) : this.attr("id"); + if (typeof value === "boolean") + { + return (value) ? "checked=\"checked\"" : ""; + } + return value; } - }); + }; - if (!String.prototype.resolve) + String.prototype.resolve = function (substitutes, prefixes) { - var formatter = { - "checked": function(value) + var result = this; + $.each(substitutes, function (key, value) + { + if (value != null && typeof value !== "function") { - if (typeof value === "boolean") + if (typeof value === "object") { - return (value) ? "checked=\"checked\"" : ""; + var keys = (prefixes) ? $.extend([], prefixes) : []; + keys.push(key); + result = result.resolve(value, keys) + ""; } - return value; - } - }; - - String.prototype.resolve = function (substitutes, prefixes) - { - var result = this; - $.each(substitutes, function (key, value) - { - if (value != null && typeof value !== "function") + else { - if (typeof value === "object") + if (formatter && formatter[key] && typeof formatter[key] === "function") { - var keys = (prefixes) ? $.extend([], prefixes) : []; - keys.push(key); - result = result.resolve(value, keys) + ""; - } - else - { - if (formatter && formatter[key] && typeof formatter[key] === "function") - { - value = formatter[key](value); - } - key = (prefixes) ? prefixes.join(".") + "." + key : key; - var pattern = new RegExp("\\{\\{" + key + "\\}\\}", "gm"); - result = result.replace(pattern, (value.replace) ? value.replace(/\$/gi, "$") : value); + value = formatter[key](value); } + key = (prefixes) ? prefixes.join(".") + "." + key : key; + var pattern = new RegExp("\\{\\{" + key + "\\}\\}", "gm"); + result = result.replace(pattern, (value.replace) ? value.replace(/\$/gi, "$") : value); } - }); - return result; - }; - } + } + }); + return result; + }; +} - if (!Array.prototype.first) +if (!Array.prototype.first) +{ + Array.prototype.first = function (condition) { - Array.prototype.first = function (condition) + for (var i = 0; i < this.length; i++) { - for (var i = 0; i < this.length; i++) + var item = this[i]; + if (condition(item)) { - var item = this[i]; - if (condition(item)) - { - return item; - } + return item; } - return null; - }; - } + } + return null; + }; +} - if (!Array.prototype.contains) +if (!Array.prototype.contains) +{ + Array.prototype.contains = function (condition) { - Array.prototype.contains = function (condition) + for (var i = 0; i < this.length; i++) { - for (var i = 0; i < this.length; i++) + var item = this[i]; + if (condition(item)) { - var item = this[i]; - if (condition(item)) - { - return true; - } + return true; } - return false; - }; - } + } + return false; + }; +} - if (!Array.prototype.page) +if (!Array.prototype.page) +{ + Array.prototype.page = function (page, size) { - Array.prototype.page = function (page, size) - { - var skip = (page - 1) * size, - end = skip + size; - return (this.length > skip) ? - (this.length > end) ? this.slice(skip, end) : - this.slice(skip) : []; - }; - } + var skip = (page - 1) * size, + end = skip + size; + return (this.length > skip) ? + (this.length > end) ? this.slice(skip, end) : + this.slice(skip) : []; + }; +} - if (!Array.prototype.where) +if (!Array.prototype.where) +{ + Array.prototype.where = function (condition) { - Array.prototype.where = function (condition) + var result = []; + for (var i = 0; i < this.length; i++) { - var result = []; - for (var i = 0; i < this.length; i++) + var item = this[i]; + if (condition(item)) { - var item = this[i]; - if (condition(item)) - { - result.push(item); - } + result.push(item); } - return result; - }; - } + } + return result; + }; +} - if (!Array.prototype.propValues) +if (!Array.prototype.propValues) +{ + Array.prototype.propValues = function (propName) { - Array.prototype.propValues = function (propName) + var result = []; + for (var i = 0; i < this.length; i++) { - var result = []; - for (var i = 0; i < this.length; i++) - { - result.push(this[i][propName]); - } - return result; - }; + result.push(this[i][propName]); + } + return result; + }; } - // GRID PLUGIN DEFINITION - // ===================== +// GRID PLUGIN DEFINITION +// ===================== - var old = $.fn.bootgrid; +var old = $.fn.bootgrid; - $.fn.bootgrid = function (option) - { - var args = Array.prototype.slice.call(arguments, 1), - returnValue = null, - elements = this.each(function (index) - { - var $this = $(this), - instance = $this.data(namespace), - options = typeof option === "object" && option; +$.fn.bootgrid = function (option) +{ + var args = Array.prototype.slice.call(arguments, 1), + returnValue = null, + elements = this.each(function (index) + { + var $this = $(this), + instance = $this.data(namespace), + options = typeof option === "object" && option; - if (!instance && option === "destroy") - { - return; - } - if (!instance) + if (!instance && option === "destroy") + { + return; + } + if (!instance) + { + $this.data(namespace, (instance = new Grid(this, options))); + init.call(instance); + } + if (typeof option === "string") + { + if (option.indexOf("get") === 0 && index === 0) { - $this.data(namespace, (instance = new Grid(this, options))); - init.call(instance); + returnValue = instance[option].apply(instance, args); } - if (typeof option === "string") + else if (option.indexOf("get") !== 0) { - if (option.indexOf("get") === 0 && index === 0) - { - returnValue = instance[option].apply(instance, args); - } - else if (option.indexOf("get") !== 0) - { - return instance[option].apply(instance, args); - } + return instance[option].apply(instance, args); } - }); - return (typeof option === "string" && option.indexOf("get") === 0) ? returnValue : elements; - }; + } + }); + return (typeof option === "string" && option.indexOf("get") === 0) ? returnValue : elements; +}; - $.fn.bootgrid.Constructor = Grid; +$.fn.bootgrid.Constructor = Grid; - // GRID NO CONFLICT - // =============== +// GRID NO CONFLICT +// =============== - $.fn.bootgrid.noConflict = function () - { - $.fn.bootgrid = old; - return this; - }; +$.fn.bootgrid.noConflict = function () +{ + $.fn.bootgrid = old; + return this; +}; - // GRID DATA-API - // ============ +// GRID DATA-API +// ============ $("[data-toggle=\"bootgrid\"]").bootgrid(); })(jQuery, window); \ No newline at end of file diff --git a/dist/jquery.bootgrid.min.css b/dist/jquery.bootgrid.min.css index 358c1ac..a928e13 100644 --- a/dist/jquery.bootgrid.min.css +++ b/dist/jquery.bootgrid.min.css @@ -1,5 +1,5 @@ /*! - * jQuery Bootgrid v1.3.1 - 09/11/2015 - * Copyright (c) 2014-2015 Rafael Staib (http://www.jquery-bootgrid.com) - * Licensed under MIT http://www.opensource.org/licenses/MIT + * jQuery Bootgrid v1.3.2 - 11/06/2018 + * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) + * Licensed under the MIT license. See LICENSE.txt for more details. */.bootgrid-footer,.bootgrid-header{margin:15px 0}.bootgrid-footer a,.bootgrid-header a{outline:0}.bootgrid-footer .search,.bootgrid-header .search{display:inline-block;margin:0 20px 0 0;vertical-align:middle;width:180px}.bootgrid-footer .search .glyphicon,.bootgrid-header .search .glyphicon{top:0}.bootgrid-footer .search .fa,.bootgrid-header .search .fa{display:table-cell}.bootgrid-footer .search .search-field::-ms-clear,.bootgrid-footer .search.search-field::-ms-clear,.bootgrid-header .search .search-field::-ms-clear,.bootgrid-header .search.search-field::-ms-clear{display:none}.bootgrid-footer .pagination,.bootgrid-header .pagination{margin:0!important}.bootgrid-footer .infoBar,.bootgrid-header .actionBar{text-align:right}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu{text-align:left}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item{cursor:pointer;display:block;margin:0;padding:3px 20px;white-space:nowrap}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item:focus,.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item:hover,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item:focus,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item:hover{color:#262626;text-decoration:none;background-color:#f5f5f5}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item .dropdown-item-checkbox,.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item.dropdown-item-checkbox,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item .dropdown-item-checkbox,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item.dropdown-item-checkbox{margin:0 2px 4px 0;vertical-align:middle}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item.disabled,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item.disabled{cursor:not-allowed}.bootgrid-table{table-layout:fixed}.bootgrid-table a{outline:0}.bootgrid-table th>.column-header-anchor{color:#333;cursor:not-allowed;display:block;position:relative;text-decoration:none}.bootgrid-table th>.column-header-anchor.sortable{cursor:pointer}.bootgrid-table th>.column-header-anchor>.text{display:block;margin:0 16px 0 0;overflow:hidden;-ms-text-overflow:ellipsis;-o-text-overflow:ellipsis;text-overflow:ellipsis;white-space:nowrap}.bootgrid-table th>.column-header-anchor>.icon{display:block;position:absolute;right:0;top:2px}.bootgrid-table th:active,.bootgrid-table th:hover{background:#fafafa}.bootgrid-table td{overflow:hidden;-ms-text-overflow:ellipsis;-o-text-overflow:ellipsis;text-overflow:ellipsis;white-space:nowrap}.bootgrid-table td.loading,.bootgrid-table td.no-results{background:#fff;text-align:center}.bootgrid-table td.select-cell,.bootgrid-table th.select-cell{text-align:center;width:30px}.bootgrid-table td.select-cell .select-box,.bootgrid-table th.select-cell .select-box{margin:0;outline:0}.table-responsive .bootgrid-table{table-layout:inherit!important}.table-responsive .bootgrid-table td,.table-responsive .bootgrid-table th>.column-header-anchor>.text{overflow:inherit!important;-ms-text-overflow:inherit!important;-o-text-overflow:inherit!important;text-overflow:inherit!important;white-space:inherit!important} \ No newline at end of file diff --git a/dist/jquery.bootgrid.min.js b/dist/jquery.bootgrid.min.js index b84a714..59069d6 100644 --- a/dist/jquery.bootgrid.min.js +++ b/dist/jquery.bootgrid.min.js @@ -1,6 +1,6 @@ /*! - * jQuery Bootgrid v1.3.1 - 09/11/2015 - * Copyright (c) 2014-2015 Rafael Staib (http://www.jquery-bootgrid.com) - * Licensed under MIT http://www.opensource.org/licenses/MIT + * jQuery Bootgrid v1.3.2 - 11/06/2018 + * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) + * Licensed under the MIT license. See LICENSE.txt for more details. */ -!function(a,b,c){"use strict";function d(a){function b(b){return c.identifier&&b[c.identifier]===a[c.identifier]}var c=this;return this.rows.contains(b)?!1:(this.rows.push(a),!0)}function e(b){var c=this.footer?this.footer.find(b):a(),d=this.header?this.header.find(b):a();return a.merge(c,d)}function f(b){return b?a.extend({},this.cachedParams,{ctx:b}):this.cachedParams}function g(){var b={current:this.current,rowCount:this.rowCount,sort:this.sortDictionary,searchPhrase:this.searchPhrase},c=this.options.post;return c=a.isFunction(c)?c():c,this.options.requestHandler(a.extend(!0,b,c))}function h(b){return"."+a.trim(b).replace(/\s+/gm,".")}function i(){var b=this.options.url;return a.isFunction(b)?b():b}function j(){this.element.trigger("initialize"+H),m.call(this),this.selection=this.options.selection&&null!=this.identifier,o.call(this),q.call(this),C.call(this),A.call(this),r.call(this),n.call(this),this.element.trigger("initialized"+H)}function k(a){this.options.highlightRows}function l(a){return a.visible}function m(){var b=this,c=this.element.find("thead > tr").first(),d=!1;c.children().each(function(){var c=a(this),e=c.data(),f={id:e.columnId,identifier:null==b.identifier&&e.identifier||!1,converter:b.options.converters[e.converter||e.type]||b.options.converters.string,text:c.text(),align:e.align||"left",headerAlign:e.headerAlign||"left",cssClass:e.cssClass||"",headerCssClass:e.headerCssClass||"",formatter:b.options.formatters[e.formatter]||null,order:d||"asc"!==e.order&&"desc"!==e.order?null:e.order,searchable:!(e.searchable===!1),sortable:!(e.sortable===!1),visible:!(e.visible===!1),visibleInSelection:!(e.visibleInSelection===!1),width:a.isNumeric(e.width)?e.width+"px":"string"==typeof e.width?e.width:null};b.columns.push(f),null!=f.order&&(b.sortDictionary[f.id]=f.order),f.identifier&&(b.identifier=f.id,b.converter=f.converter),b.options.multiSort||null===f.order||(d=!0)})}function n(){function c(a){for(var b,c=new RegExp(e.searchPhrase,e.options.caseSensitive?"g":"gi"),d=0;d-1)return!0;return!1}function d(a,b){e.currentRows=a,p.call(e,b),e.options.keepSelection||(e.selectedRows=[]),y.call(e,a),t.call(e),v.call(e),e.element._bgBusyAria(!1).trigger("loaded"+H)}var e=this;if(this.element._bgBusyAria(!0).trigger("load"+H),F.call(this),this.options.ajax){var f=g.call(this),h=i.call(this);if(null==h||"string"!=typeof h||0===h.length)throw new Error("Url setting must be a none empty string or a function that returns one.");this.xqr&&this.xqr.abort();var j={url:h,data:f,success:function(b){e.xqr=null,"string"==typeof b&&(b=a.parseJSON(b)),b=e.options.responseHandler(b),e.current=b.current,d(b.rows,b.total)},error:function(a,b,c){e.xqr=null,"abort"!==b&&(u.call(e),e.element._bgBusyAria(!1).trigger("loaded"+H))}};j=a.extend(this.options.ajaxSettings,j),this.xqr=a.ajax(j)}else{var k=this.searchPhrase.length>0?this.rows.where(c):this.rows,l=k.length;-1!==this.rowCount&&(k=k.page(this.current,this.rowCount)),b.setTimeout(function(){d(k,l)},10)}}function o(){if(!this.options.ajax){var b=this,c=this.element.find("tbody > tr");c.each(function(){var c=a(this),e=c.children("td"),f={};a.each(b.columns,function(a,b){f[b.id]=b.converter.from(e.eq(a).text())}),d.call(b,f)}),p.call(this,this.rows.length),G.call(this)}}function p(a){this.total=a,this.totalPages=-1===this.rowCount?1:Math.ceil(this.total/this.rowCount)}function q(){var b=this.options.templates,c=this.element.parent().hasClass(this.options.css.responsiveTable)?this.element.parent():this.element;this.element.addClass(this.options.css.table),0===this.element.children("tbody").length&&this.element.append(b.body),1&this.options.navigation&&(this.header=a(b.header.resolve(f.call(this,{id:this.element._bgId()+"-header"}))),c.before(this.header)),2&this.options.navigation&&(this.footer=a(b.footer.resolve(f.call(this,{id:this.element._bgId()+"-footer"}))),c.after(this.footer))}function r(){if(0!==this.options.navigation){var b=this.options.css,c=h(b.actions),d=e.call(this,c);if(d.length>0){var g=this,i=this.options.templates,j=a(i.actions.resolve(f.call(this)));if(this.options.ajax){var k=i.icon.resolve(f.call(this,{iconCss:b.iconRefresh})),l=a(i.actionButton.resolve(f.call(this,{content:k,text:this.options.labels.refresh}))).on("click"+H,function(a){a.stopPropagation(),g.current=1,n.call(g)});j.append(l)}x.call(this,j),s.call(this,j),E.call(this,d,j)}}}function s(b){if(this.options.columnSelection&&this.columns.length>1){var c=this,d=this.options.css,e=this.options.templates,g=e.icon.resolve(f.call(this,{iconCss:d.iconColumns})),i=a(e.actionDropDown.resolve(f.call(this,{content:g}))),j=h(d.dropDownItem),k=h(d.dropDownItemCheckbox),m=h(d.dropDownMenuItems);a.each(this.columns,function(b,g){if(g.visibleInSelection){var o=a(e.actionDropDownCheckboxItem.resolve(f.call(c,{name:g.id,label:g.text,checked:g.visible}))).on("click"+H,j,function(b){b.stopPropagation();var d=a(this),e=d.find(k);if(!e.prop("disabled")){g.visible=e.prop("checked");var f=c.columns.where(l).length>1;d.parents(m).find(j+":has("+k+":checked)")._bgEnableAria(f).find(k)._bgEnableField(f),c.element.find("tbody").empty(),C.call(c),n.call(c)}});i.find(h(d.dropDownMenuItems)).append(o)}}),b.append(i)}}function t(){if(0!==this.options.navigation){var b=h(this.options.css.infos),c=e.call(this,b);if(c.length>0){var d=this.current*this.rowCount,g=a(this.options.templates.infos.resolve(f.call(this,{end:0===this.total||-1===d||d>this.total?this.total:d,start:0===this.total?0:d-this.rowCount+1,total:this.total})));E.call(this,c,g)}}}function u(){var a=this.element.children("tbody").first(),b=this.options.templates,c=this.columns.where(l).length;this.selection&&(c+=1),a.html(b.noResults.resolve(f.call(this,{columns:c})))}function v(){if(0!==this.options.navigation){var b=h(this.options.css.pagination),c=e.call(this,b)._bgShowAria(-1!==this.rowCount);if(-1!==this.rowCount&&c.length>0){var d=this.options.templates,g=this.current,i=this.totalPages,j=a(d.pagination.resolve(f.call(this))),k=i-g,l=-1*(this.options.padding-g),m=k>=this.options.padding?Math.max(l,1):Math.max(l-this.options.padding+k,1),n=2*this.options.padding+1,o=i>=n?n:i;w.call(this,j,"first","«","first")._bgEnableAria(g>1),w.call(this,j,"prev","<","prev")._bgEnableAria(g>1);for(var p=0;o>p;p++){var q=p+m;w.call(this,j,q,q,"page-"+q)._bgEnableAria()._bgSelectAria(q===g)}0===o&&w.call(this,j,1,1,"page-1")._bgEnableAria(!1)._bgSelectAria(),w.call(this,j,"next",">","next")._bgEnableAria(i>g),w.call(this,j,"last","»","last")._bgEnableAria(i>g),E.call(this,c,j)}}}function w(b,c,d,e){var g=this,i=this.options.templates,j=this.options.css,k=f.call(this,{css:e,text:d,page:c}),l=a(i.paginationItem.resolve(k)).on("click"+H,h(j.paginationButton),function(b){b.stopPropagation(),b.preventDefault();var c=a(this),d=c.parent();if(!d.hasClass("active")&&!d.hasClass("disabled")){var e={first:1,prev:g.current-1,next:g.current+1,last:g.totalPages},f=c.data("page");g.current=e[f]||f,n.call(g)}c.trigger("blur")});return b.append(l),l}function x(b){function c(a){return-1===a?d.options.labels.all:a}var d=this,e=this.options.rowCount;if(a.isArray(e)){var g=this.options.css,i=this.options.templates,j=a(i.actionDropDown.resolve(f.call(this,{content:c(this.rowCount)}))),k=h(g.dropDownMenu),l=h(g.dropDownMenuText),m=h(g.dropDownMenuItems),o=h(g.dropDownItemButton);a.each(e,function(b,e){var g=a(i.actionDropDownItem.resolve(f.call(d,{text:c(e),action:e})))._bgSelectAria(e===d.rowCount).on("click"+H,o,function(b){b.preventDefault();var e=a(this),f=e.data("action");f!==d.rowCount&&(d.current=1,d.rowCount=f,e.parents(m).children().each(function(){var b=a(this),c=b.find(o).data("action");b._bgSelectAria(c===f)}),e.parents(k).find(l).text(c(f)),n.call(d))});j.find(m).append(g)}),b.append(j)}}function y(b){if(b.length>0){var c=this,d=this.options.css,e=this.options.templates,g=this.element.children("tbody").first(),i=!0,j="";a.each(b,function(b,g){var h="",k=' data-row-id="'+(null==c.identifier?b:g[c.identifier])+'"',l="";if(c.selection){var m=-1!==a.inArray(g[c.identifier],c.selectedRows),n=e.select.resolve(f.call(c,{type:"checkbox",value:g[c.identifier],checked:m}));h+=e.cell.resolve(f.call(c,{content:n,css:d.selectCell})),i=i&&m,m&&(l+=d.selected,k+=' aria-selected="true"')}var o=null!=g.status&&c.options.statusMapping[g.status];o&&(l+=o),a.each(c.columns,function(b,i){if(i.visible){var j=a.isFunction(i.formatter)?i.formatter.call(c,i,g):i.converter.to(g[i.id]),k=i.cssClass.length>0?" "+i.cssClass:"";h+=e.cell.resolve(f.call(c,{content:null==j||""===j?" ":j,css:("right"===i.align?d.right:"center"===i.align?d.center:d.left)+k,style:null==i.width?"":"width:"+i.width+";"}))}}),l.length>0&&(k+=' class="'+l+'"'),j+=e.row.resolve(f.call(c,{attr:k,cells:h}))}),c.element.find("thead "+h(c.options.css.selectBox)).prop("checked",i),g.html(j),z.call(this,g)}else u.call(this)}function z(b){var c=this,d=h(this.options.css.selectBox);this.selection&&b.off("click"+H,d).on("click"+H,d,function(b){b.stopPropagation();var d=a(this),e=c.converter.from(d.val());d.prop("checked")?c.select([e]):c.deselect([e])}),b.off("click"+H,"> tr").on("click"+H,"> tr",function(b){b.stopPropagation();var d=a(this),e=null==c.identifier?d.data("row-id"):c.converter.from(d.data("row-id")+""),f=null==c.identifier?c.currentRows[e]:c.currentRows.first(function(a){return a[c.identifier]===e});c.selection&&c.options.rowSelect&&(d.hasClass(c.options.css.selected)?c.deselect([e]):c.select([e])),c.element.trigger("click"+H,[c.columns,f])})}function A(){if(0!==this.options.navigation){var c=this.options.css,d=h(c.search),g=e.call(this,d);if(g.length>0){var i=this,j=this.options.templates,k=null,l="",m=h(c.searchField),n=a(j.search.resolve(f.call(this))),o=n.is(m)?n:n.find(m);o.on("keyup"+H,function(c){c.stopPropagation();var d=a(this).val();(l!==d||13===c.which&&""!==d)&&(l=d,(13===c.which||0===d.length||d.length>=i.options.searchSettings.characters)&&(b.clearTimeout(k),k=b.setTimeout(function(){B.call(i,d)},i.options.searchSettings.delay)))}),E.call(this,g,n)}}}function B(a){this.searchPhrase!==a&&(this.current=1,this.searchPhrase=a,n.call(this))}function C(){var b=this,c=this.element.find("thead > tr"),d=this.options.css,e=this.options.templates,g="",i=this.options.sorting;if(this.selection){var j=this.options.multiSelect?e.select.resolve(f.call(b,{type:"checkbox",value:"all"})):"";g+=e.rawHeaderCell.resolve(f.call(b,{content:j,css:d.selectCell}))}if(a.each(this.columns,function(a,c){if(c.visible){var h=b.sortDictionary[c.id],j=i&&h&&"asc"===h?d.iconUp:i&&h&&"desc"===h?d.iconDown:"",k=e.icon.resolve(f.call(b,{iconCss:j})),l=c.headerAlign,m=c.headerCssClass.length>0?" "+c.headerCssClass:"";g+=e.headerCell.resolve(f.call(b,{column:c,icon:k,sortable:i&&c.sortable&&d.sortable||"",css:("right"===l?d.right:"center"===l?d.center:d.left)+m,style:null==c.width?"":"width:"+c.width+";"}))}}),c.html(g),i){var k=h(d.sortable);c.off("click"+H,k).on("click"+H,k,function(c){c.preventDefault(),D.call(b,a(this)),G.call(b),n.call(b)})}if(this.selection&&this.options.multiSelect){var l=h(d.selectBox);c.off("click"+H,l).on("click"+H,l,function(c){c.stopPropagation(),a(this).prop("checked")?b.select():b.deselect()})}}function D(a){var b=this.options.css,c=h(b.icon),d=a.data("column-id")||a.parents("th").first().data("column-id"),e=this.sortDictionary[d],f=a.find(c);if(this.options.multiSort||(a.parents("tr").first().find(c).removeClass(b.iconDown+" "+b.iconUp),this.sortDictionary={}),e&&"asc"===e)this.sortDictionary[d]="desc",f.removeClass(b.iconUp).addClass(b.iconDown);else if(e&&"desc"===e)if(this.options.multiSort){var g={};for(var i in this.sortDictionary)i!==d&&(g[i]=this.sortDictionary[i]);this.sortDictionary=g,f.removeClass(b.iconDown)}else this.sortDictionary[d]="asc",f.removeClass(b.iconDown).addClass(b.iconUp);else this.sortDictionary[d]="asc",f.addClass(b.iconUp)}function E(b,c){b.each(function(b,d){a(d).before(c.clone(!0)).remove()})}function F(){var a=this;b.setTimeout(function(){if("true"===a.element._bgAria("busy")){var b=a.options.templates,c=a.element.children("thead").first(),d=a.element.children("tbody").first(),e=d.find("tr > td").first(),g=a.element.height()-c.height()-(e.height()+20),h=a.columns.where(l).length;a.selection&&(h+=1),d.html(b.loading.resolve(f.call(a,{columns:h}))),-1!==a.rowCount&&g>0&&d.find("tr > td").css("padding","20px 0 "+g+"px")}},250)}function G(){function a(c,d,e){function f(a){return"asc"===h.order?a:-1*a}e=e||0;var g=e+1,h=b[e];return c[h.id]>d[h.id]?f(1):c[h.id]g?a(c,d,g):0}var b=[];if(!this.options.ajax){for(var c in this.sortDictionary)(this.options.multiSort||0===b.length)&&b.push({id:c,order:this.sortDictionary[c]});b.length>0&&this.rows.sort(a)}}var H=".rs.jquery.bootgrid",I=function(b,c){this.element=a(b),this.origin=this.element.clone(),this.options=a.extend(!0,{},I.defaults,this.element.data(),c);var d=this.options.rowCount=this.element.data().rowCount||c.rowCount||this.options.rowCount;this.columns=[],this.current=1,this.currentRows=[],this.identifier=null,this.selection=!1,this.converter=null,this.rowCount=a.isArray(d)?d[0]:d,this.rows=[],this.searchPhrase="",this.selectedRows=[],this.sortDictionary={},this.total=0,this.totalPages=0,this.cachedParams={lbl:this.options.labels,css:this.options.css,ctx:{}},this.header=null,this.footer=null,this.xqr=null};if(I.defaults={navigation:3,padding:2,columnSelection:!0,rowCount:[10,25,50,-1],selection:!1,multiSelect:!1,rowSelect:!1,keepSelection:!1,highlightRows:!1,sorting:!0,multiSort:!1,searchSettings:{delay:250,characters:1},ajax:!1,ajaxSettings:{method:"POST"},post:{},url:"",caseSensitive:!0,requestHandler:function(a){return a},responseHandler:function(a){return a},converters:{numeric:{from:function(a){return+a},to:function(a){return a+""}},string:{from:function(a){return a},to:function(a){return a}}},css:{actions:"actions btn-group",center:"text-center",columnHeaderAnchor:"column-header-anchor",columnHeaderText:"text",dropDownItem:"dropdown-item",dropDownItemButton:"dropdown-item-button",dropDownItemCheckbox:"dropdown-item-checkbox",dropDownMenu:"dropdown btn-group",dropDownMenuItems:"dropdown-menu pull-right",dropDownMenuText:"dropdown-text",footer:"bootgrid-footer container-fluid",header:"bootgrid-header container-fluid",icon:"icon glyphicon",iconColumns:"glyphicon-th-list",iconDown:"glyphicon-chevron-down",iconRefresh:"glyphicon-refresh",iconSearch:"glyphicon-search",iconUp:"glyphicon-chevron-up",infos:"infos",left:"text-left",pagination:"pagination",paginationButton:"button",responsiveTable:"table-responsive",right:"text-right",search:"search form-group",searchField:"search-field form-control",selectBox:"select-box",selectCell:"select-cell",selected:"active",sortable:"sortable",table:"bootgrid-table table"},formatters:{},labels:{all:"All",infos:"Showing {{ctx.start}} to {{ctx.end}} of {{ctx.total}} entries",loading:"Loading...",noResults:"No results found!",refresh:"Refresh",search:"Search"},statusMapping:{0:"success",1:"info",2:"warning",3:"danger"},templates:{actionButton:'',actionDropDown:'
                ',actionDropDownItem:'
              • {{ctx.text}}
              • ',actionDropDownCheckboxItem:'
              • ',actions:'
                ',body:"",cell:'{{ctx.content}}',footer:'

                ',header:'

                ',headerCell:'{{ctx.column.text}}{{ctx.icon}}',icon:'',infos:'
                {{lbl.infos}}
                ',loading:'{{lbl.loading}}',noResults:'{{lbl.noResults}}',pagination:'
                  ',paginationItem:'
                • {{ctx.text}}
                • ',rawHeaderCell:'{{ctx.content}}',row:"{{ctx.cells}}",search:'
                  ',select:''}},I.prototype.append=function(a){if(this.options.ajax);else{for(var b=[],c=0;c0&&(this.options.multiSelect||1!==e.length);)if(c=b.pop(),-1===a.inArray(c,this.selectedRows))for(d=0;d0){var f=h(this.options.css.selectBox),g=this.selectedRows.length>=this.currentRows.length;for(d=0;!this.options.keepSelection&&g&&d tr "+f+":checked").trigger("click"+H),d=0;d tr[data-row-id="'+this.selectedRows[d]+'"]').addClass(this.options.css.selected)._bgAria("selected","true").find(f).prop("checked",!0);this.element.trigger("selected"+H,[e])}}return this},I.prototype.deselect=function(b){if(this.selection){b=b||this.currentRows.propValues(this.identifier);for(var c,d,e,f=[];b.length>0;)if(c=b.pop(),e=a.inArray(c,this.selectedRows),-1!==e)for(d=0;d0){var g=h(this.options.css.selectBox);for(this.element.find("thead "+g).prop("checked",!1),d=0;d tr[data-row-id="'+f[d][this.identifier]+'"]').removeClass(this.options.css.selected)._bgAria("selected","false").find(g).prop("checked",!1);this.element.trigger("deselected"+H,[f])}}return this},I.prototype.sort=function(b){var c=b?a.extend({},b):{};return c===this.sortDictionary?this:(this.sortDictionary=c,C.call(this),G.call(this),n.call(this),this)},I.prototype.getColumnSettings=function(){return a.merge([],this.columns)},I.prototype.getCurrentPage=function(){return this.current},I.prototype.getCurrentRows=function(){return a.merge([],this.currentRows)},I.prototype.getRowCount=function(){return this.rowCount},I.prototype.getSearchPhrase=function(){return this.searchPhrase},I.prototype.getSelectedRows=function(){return a.merge([],this.selectedRows)},I.prototype.getSortDictionary=function(){return a.extend({},this.sortDictionary)},I.prototype.getTotalPageCount=function(){return this.totalPages},I.prototype.getTotalRowCount=function(){return this.total},a.fn.extend({_bgAria:function(a,b){return b?this.attr("aria-"+a,b):this.attr("aria-"+a)},_bgBusyAria:function(a){return null==a||a?this._bgAria("busy","true"):this._bgAria("busy","false")},_bgRemoveAria:function(a){return this.removeAttr("aria-"+a)},_bgEnableAria:function(a){return null==a||a?this.removeClass("disabled")._bgAria("disabled","false"):this.addClass("disabled")._bgAria("disabled","true")},_bgEnableField:function(a){return null==a||a?this.removeAttr("disabled"):this.attr("disabled","disable")},_bgShowAria:function(a){return null==a||a?this.show()._bgAria("hidden","false"):this.hide()._bgAria("hidden","true")},_bgSelectAria:function(a){return null==a||a?this.addClass("active")._bgAria("selected","true"):this.removeClass("active")._bgAria("selected","false")},_bgId:function(a){return a?this.attr("id",a):this.attr("id")}}),!String.prototype.resolve){var J={checked:function(a){return"boolean"==typeof a?a?'checked="checked"':"":a}};String.prototype.resolve=function(b,c){var d=this;return a.each(b,function(b,e){if(null!=e&&"function"!=typeof e)if("object"==typeof e){var f=c?a.extend([],c):[];f.push(b),d=d.resolve(e,f)+""}else{J&&J[b]&&"function"==typeof J[b]&&(e=J[b](e)),b=c?c.join(".")+"."+b:b;var g=new RegExp("\\{\\{"+b+"\\}\\}","gm");d=d.replace(g,e.replace?e.replace(/\$/gi,"$"):e)}}),d}}Array.prototype.first||(Array.prototype.first=function(a){for(var b=0;bc?this.length>d?this.slice(c,d):this.slice(c):[]}),Array.prototype.where||(Array.prototype.where=function(a){for(var b=[],c=0;c tr").first(),d=!1;c.children().each(function(){var c=a(this),e=c.data(),f={id:e.columnId,identifier:null==b.identifier&&e.identifier||!1,converter:b.options.converters[e.converter||e.type]||b.options.converters.string,text:c.text(),align:e.align||"left",headerAlign:e.headerAlign||"left",cssClass:e.cssClass||"",headerCssClass:e.headerCssClass||"",formatter:b.options.formatters[e.formatter]||null,order:d||"asc"!==e.order&&"desc"!==e.order?null:e.order,searchable:!(!1===e.searchable),sortable:!(!1===e.sortable),visible:!(!1===e.visible),visibleInSelection:!(!1===e.visibleInSelection),width:a.isNumeric(e.width)?e.width+"px":"string"==typeof e.width?e.width:null};b.columns.push(f),null!=f.order&&(b.sortDictionary[f.id]=f.order),f.identifier&&(b.identifier=f.id,b.converter=f.converter),b.options.multiSort||null===f.order||(d=!0)})}function n(){function c(a){for(var b,c=new RegExp(e.searchPhrase,e.options.caseSensitive?"g":"gi"),d=0;d-1)return!0;return!1}function d(a,b){e.currentRows=a,p.call(e,b),e.options.keepSelection||(e.selectedRows=[]),y.call(e,a),t.call(e),v.call(e),e.element._bgBusyAria(!1).trigger("loaded"+H)}var e=this;if(this.element._bgBusyAria(!0).trigger("load"+H),F.call(this),this.options.ajax){var f=g.call(this),h=i.call(this);if(null==h||"string"!=typeof h||0===h.length)throw new Error("Url setting must be a none empty string or a function that returns one.");this.xqr&&this.xqr.abort();var j={url:h,data:f,success:function(b){e.xqr=null,"string"==typeof b&&(b=a.parseJSON(b)),b=e.options.responseHandler(b),e.current=b.current,d(b.rows,b.total)},error:function(a,b,c){e.xqr=null,"abort"!==b&&(u.call(e),e.element._bgBusyAria(!1).trigger("loaded"+H))}};j=a.extend(this.options.ajaxSettings,j),this.xqr=a.ajax(j)}else{var k=this.searchPhrase.length>0?this.rows.where(c):this.rows,l=k.length;-1!==this.rowCount&&(k=k.page(this.current,this.rowCount)),b.setTimeout(function(){d(k,l)},10)}}function o(){if(!this.options.ajax){var b=this;this.element.find("tbody > tr").each(function(){var c=a(this),e=c.children("td"),f={};a.each(b.columns,function(a,b){f[b.id]=b.converter.from(e.eq(a).text())}),d.call(b,f)}),p.call(this,this.rows.length),G.call(this)}}function p(a){this.total=a,this.totalPages=-1===this.rowCount?1:Math.ceil(this.total/this.rowCount)}function q(){var b=this.options.templates,c=this.element.parent().hasClass(this.options.css.responsiveTable)?this.element.parent():this.element;this.element.addClass(this.options.css.table),0===this.element.children("tbody").length&&this.element.append(b.body),1&this.options.navigation&&(this.header=a(b.header.resolve(f.call(this,{id:this.element._bgId()+"-header"}))),c.before(this.header)),2&this.options.navigation&&(this.footer=a(b.footer.resolve(f.call(this,{id:this.element._bgId()+"-footer"}))),c.after(this.footer))}function r(){if(0!==this.options.navigation){var b=this.options.css,c=h(b.actions),d=e.call(this,c);if(d.length>0){var g=this,i=this.options.templates,j=a(i.actions.resolve(f.call(this)));if(this.options.ajax){var k=i.icon.resolve(f.call(this,{iconCss:b.iconRefresh})),l=a(i.actionButton.resolve(f.call(this,{content:k,text:this.options.labels.refresh}))).on("click"+H,function(a){a.stopPropagation(),g.current=1,n.call(g)});j.append(l)}x.call(this,j),s.call(this,j),E.call(this,d,j)}}}function s(b){if(this.options.columnSelection&&this.columns.length>1){var c=this,d=this.options.css,e=this.options.templates,g=e.icon.resolve(f.call(this,{iconCss:d.iconColumns})),i=a(e.actionDropDown.resolve(f.call(this,{content:g}))),j=h(d.dropDownItem),k=h(d.dropDownItemCheckbox),m=h(d.dropDownMenuItems);a.each(this.columns,function(b,g){if(g.visibleInSelection){var o=a(e.actionDropDownCheckboxItem.resolve(f.call(c,{name:g.id,label:g.text,checked:g.visible}))).on("click"+H,j,function(b){b.stopPropagation();var d=a(this),e=d.find(k);if(!e.prop("disabled")){g.visible=e.prop("checked");var f=c.columns.where(l).length>1;d.parents(m).find(j+":has("+k+":checked)")._bgEnableAria(f).find(k)._bgEnableField(f),c.element.find("tbody").empty(),C.call(c),n.call(c)}});i.find(h(d.dropDownMenuItems)).append(o)}}),b.append(i)}}function t(){if(0!==this.options.navigation){var b=h(this.options.css.infos),c=e.call(this,b);if(c.length>0){var d=this.current*this.rowCount,g=a(this.options.templates.infos.resolve(f.call(this,{end:0===this.total||-1===d||d>this.total?this.total:d,start:0===this.total?0:d-this.rowCount+1,total:this.total})));E.call(this,c,g)}}}function u(){var a=this.element.children("tbody").first(),b=this.options.templates,c=this.columns.where(l).length;this.selection&&(c+=1),a.html(b.noResults.resolve(f.call(this,{columns:c})))}function v(){if(0!==this.options.navigation){var b=h(this.options.css.pagination),c=e.call(this,b)._bgShowAria(-1!==this.rowCount);if(-1!==this.rowCount&&c.length>0){var d=this.options.templates,g=this.current,i=this.totalPages,j=a(d.pagination.resolve(f.call(this))),k=i-g,l=-1*(this.options.padding-g),m=k>=this.options.padding?Math.max(l,1):Math.max(l-this.options.padding+k,1),n=2*this.options.padding+1,o=i>=n?n:i;w.call(this,j,"first","«","first")._bgEnableAria(g>1),w.call(this,j,"prev","‹","prev")._bgEnableAria(g>1);for(var p=0;pg),w.call(this,j,"last","»","last")._bgEnableAria(i>g),E.call(this,c,j)}}}function w(b,c,d,e){var g=this,i=this.options.templates,j=this.options.css,k=f.call(this,{css:e,text:d,page:c}),l=a(i.paginationItem.resolve(k)).on("click"+H,h(j.paginationButton),function(b){b.stopPropagation(),b.preventDefault();var c=a(this),d=c.parent();if(!d.hasClass("active")&&!d.hasClass("disabled")){var e={first:1,prev:g.current-1,next:g.current+1,last:g.totalPages},f=c.data("page");g.current=e[f]||f,n.call(g)}c.trigger("blur")});return b.append(l),l}function x(b){function c(a){return-1===a?d.options.labels.all:a}var d=this,e=this.options.rowCount;if(a.isArray(e)){var g=this.options.css,i=this.options.templates,j=a(i.actionDropDown.resolve(f.call(this,{content:c(this.rowCount)}))),k=h(g.dropDownMenu),l=h(g.dropDownMenuText),m=h(g.dropDownMenuItems),o=h(g.dropDownItemButton);a.each(e,function(b,e){var g=a(i.actionDropDownItem.resolve(f.call(d,{text:c(e),action:e})))._bgSelectAria(e===d.rowCount).on("click"+H,o,function(b){b.preventDefault();var e=a(this),f=e.data("action");f!==d.rowCount&&(d.current=1,d.rowCount=f,e.parents(m).children().each(function(){var b=a(this),c=b.find(o).data("action");b._bgSelectAria(c===f)}),e.parents(k).find(l).text(c(f)),n.call(d))});j.find(m).append(g)}),b.append(j)}}function y(b){if(b.length>0){var c=this,d=this.options.css,e=this.options.templates,g=this.element.children("tbody").first(),i=!0,j="";a.each(b,function(b,g){var h="",k=' data-row-id="'+(null==c.identifier?b:g[c.identifier])+'"',l="";if(c.selection){var m=-1!==a.inArray(g[c.identifier],c.selectedRows),n=e.select.resolve(f.call(c,{type:"checkbox",value:g[c.identifier],checked:m}));h+=e.cell.resolve(f.call(c,{content:n,css:d.selectCell})),i=i&&m,m&&(l+=d.selected,k+=' aria-selected="true"')}var o=null!=g.status&&c.options.statusMapping[g.status];o&&(l+=o),a.each(c.columns,function(b,i){if(i.visible){var j=a.isFunction(i.formatter)?i.formatter.call(c,i,g):i.converter.to(g[i.id]),k=i.cssClass.length>0?" "+i.cssClass:"";h+=e.cell.resolve(f.call(c,{content:null==j||""===j?" ":j,css:("right"===i.align?d.right:"center"===i.align?d.center:d.left)+k,style:null==i.width?"":"width:"+i.width+";"}))}}),l.length>0&&(k+=' class="'+l+'"'),j+=e.row.resolve(f.call(c,{attr:k,cells:h}))}),c.element.find("thead "+h(c.options.css.selectBox)).prop("checked",i),g.html(j),z.call(this,g)}else u.call(this)}function z(b){var c=this,d=h(this.options.css.selectBox);this.selection&&!this.options.rowSelect&&b.off("click"+H,d).on("click"+H,d,function(b){b.stopPropagation();var d=a(this),e=c.converter.from(d.val());d.prop("checked")?c.select([e]):c.deselect([e])}),b.off("click"+H,"> tr").on("click"+H,"> tr",function(b){b.stopPropagation();var d=a(this),e=null==c.identifier?d.data("row-id"):c.converter.from(d.data("row-id")+""),f=null==c.identifier?c.currentRows[e]:c.currentRows.first(function(a){return a[c.identifier]===e});c.selection&&c.options.rowSelect&&(d.hasClass(c.options.css.selected)?c.deselect([e]):c.select([e])),c.element.trigger("click"+H,[c.columns,f])})}function A(){if(0!==this.options.navigation){var c=this.options.css,d=h(c.search),g=e.call(this,d);if(g.length>0){var i=this,j=this.options.templates,k=null,l="",m=h(c.searchField),n=a(j.search.resolve(f.call(this)));(n.is(m)?n:n.find(m)).on("keyup"+H,function(c){c.stopPropagation();var d=a(this).val();(l!==d||13===c.which&&""!==d)&&(l=d,(13===c.which||0===d.length||d.length>=i.options.searchSettings.characters)&&(b.clearTimeout(k),k=b.setTimeout(function(){B.call(i,d)},i.options.searchSettings.delay)))}),E.call(this,g,n)}}}function B(a){this.searchPhrase!==a&&(this.current=1,this.searchPhrase=a,n.call(this))}function C(){var b=this,c=this.element.find("thead > tr"),d=this.options.css,e=this.options.templates,g="",i=this.options.sorting;if(this.selection){var j=this.options.multiSelect?e.select.resolve(f.call(b,{type:"checkbox",value:"all"})):"";g+=e.rawHeaderCell.resolve(f.call(b,{content:j,css:d.selectCell}))}if(a.each(this.columns,function(a,c){if(c.visible){var h=b.sortDictionary[c.id],j=i&&h&&"asc"===h?d.iconUp:i&&h&&"desc"===h?d.iconDown:"",k=e.icon.resolve(f.call(b,{iconCss:j})),l=c.headerAlign,m=c.headerCssClass.length>0?" "+c.headerCssClass:"";g+=e.headerCell.resolve(f.call(b,{column:c,icon:k,sortable:i&&c.sortable&&d.sortable||"",css:("right"===l?d.right:"center"===l?d.center:d.left)+m,style:null==c.width?"":"width:"+c.width+";"}))}}),c.html(g),i){var k=h(d.sortable);c.off("click"+H,k).on("click"+H,k,function(c){c.preventDefault(),D.call(b,a(this)),G.call(b),n.call(b)})}if(this.selection&&this.options.multiSelect){var l=h(d.selectBox);c.off("click"+H,l).on("click"+H,l,function(c){c.stopPropagation(),a(this).prop("checked")?b.select():b.deselect()})}}function D(a){var b=this.options.css,c=h(b.icon),d=a.data("column-id")||a.parents("th").first().data("column-id"),e=this.sortDictionary[d],f=a.find(c);if(this.options.multiSort||(a.parents("tr").first().find(c).removeClass(b.iconDown+" "+b.iconUp),this.sortDictionary={}),e&&"asc"===e)this.sortDictionary[d]="desc",f.removeClass(b.iconUp).addClass(b.iconDown);else if(e&&"desc"===e)if(this.options.multiSort){var g={};for(var i in this.sortDictionary)i!==d&&(g[i]=this.sortDictionary[i]);this.sortDictionary=g,f.removeClass(b.iconDown)}else this.sortDictionary[d]="asc",f.removeClass(b.iconDown).addClass(b.iconUp);else this.sortDictionary[d]="asc",f.addClass(b.iconUp)}function E(b,c){b.each(function(b,d){a(d).before(c.clone(!0)).remove()})}function F(){var a=this;b.setTimeout(function(){if("true"===a.element._bgAria("busy")){var b=a.options.templates,c=a.element.children("thead").first(),d=a.element.children("tbody").first(),e=d.find("tr > td").first(),g=a.element.height()-c.height()-(e.height()+20),h=a.columns.where(l).length;a.selection&&(h+=1),d.html(b.loading.resolve(f.call(a,{columns:h}))),-1!==a.rowCount&&g>0&&d.find("tr > td").css("padding","20px 0 "+g+"px")}},250)}function G(){function a(c,d,e){function f(a){return"asc"===h.order?a:-1*a}e=e||0;var g=e+1,h=b[e];return c[h.id]>d[h.id]?f(1):c[h.id]g?a(c,d,g):0}var b=[];if(!this.options.ajax){for(var c in this.sortDictionary)(this.options.multiSort||0===b.length)&&b.push({id:c,order:this.sortDictionary[c]});b.length>0&&this.rows.sort(a)}}var H=".rs.jquery.bootgrid",I=function(b,c){this.element=a(b),this.origin=this.element.clone(),this.options=a.extend(!0,{},I.defaults,this.element.data(),c);var d=this.options.rowCount=this.element.data().rowCount||c.rowCount||this.options.rowCount;this.columns=[],this.current=1,this.currentRows=[],this.identifier=null,this.selection=!1,this.converter=null,this.rowCount=a.isArray(d)?d[0]:d,this.rows=[],this.searchPhrase="",this.selectedRows=[],this.sortDictionary={},this.total=0,this.totalPages=0,this.cachedParams={lbl:this.options.labels,css:this.options.css,ctx:{}},this.header=null,this.footer=null,this.xqr=null};if(I.defaults={navigation:3,padding:2,columnSelection:!0,rowCount:[10,25,50,-1],selection:!1,multiSelect:!1,rowSelect:!1,keepSelection:!1,highlightRows:!1,sorting:!0,multiSort:!1,searchSettings:{delay:250,characters:1},ajax:!1,ajaxSettings:{method:"POST"},post:{},url:"",caseSensitive:!0,requestHandler:function(a){return a},responseHandler:function(a){return a},converters:{numeric:{from:function(a){return+a},to:function(a){return a+""}},string:{from:function(a){return a},to:function(a){return a}}},css:{actions:"actions btn-group",center:"text-center",columnHeaderAnchor:"column-header-anchor",columnHeaderText:"text",dropDownItem:"dropdown-item",dropDownItemButton:"dropdown-item-button",dropDownItemCheckbox:"dropdown-item-checkbox",dropDownMenu:"dropdown btn-group",dropDownMenuItems:"dropdown-menu pull-right",dropDownMenuText:"dropdown-text",footer:"bootgrid-footer container-fluid",header:"bootgrid-header container-fluid",icon:"icon glyphicon",iconColumns:"glyphicon-th-list",iconDown:"glyphicon-chevron-down",iconRefresh:"glyphicon-refresh",iconSearch:"glyphicon-search",iconUp:"glyphicon-chevron-up",infos:"infos",left:"text-left",pagination:"pagination",paginationButton:"button",responsiveTable:"table-responsive",right:"text-right",search:"search form-group",searchField:"search-field form-control",selectBox:"select-box",selectCell:"select-cell",selected:"active",sortable:"sortable",table:"bootgrid-table table"},formatters:{},labels:{all:"All",infos:"Showing {{ctx.start}} to {{ctx.end}} of {{ctx.total}} entries",loading:"Loading...",noResults:"No results found!",refresh:"Refresh",search:"Search"},statusMapping:{0:"success",1:"info",2:"warning",3:"danger"},templates:{actionButton:'',actionDropDown:'
                  ',actionDropDownItem:'
                • {{ctx.text}}
                • ',actionDropDownCheckboxItem:'
                • ',actions:'
                  ',body:"",cell:'{{ctx.content}}',footer:'

                  ',header:'

                  ',headerCell:'{{ctx.column.text}}{{ctx.icon}}',icon:'',infos:'
                  {{lbl.infos}}
                  ',loading:'{{lbl.loading}}',noResults:'{{lbl.noResults}}',pagination:'
                    ',paginationItem:'
                  • {{ctx.text}}
                  • ',rawHeaderCell:'{{ctx.content}}',row:"{{ctx.cells}}",search:'
                    ',select:''}},I.prototype.append=function(a){if(this.options.ajax);else{for(var b=[],c=0;c0&&(this.options.multiSelect||1!==e.length);)if(c=b.pop(),-1===a.inArray(c,this.selectedRows))for(d=0;d0){var f=h(this.options.css.selectBox),g=this.selectedRows.length>=this.currentRows.length;for(d=0;!this.options.keepSelection&&g&&d tr "+f+":checked").trigger("click"+H),d=0;d tr[data-row-id="'+this.selectedRows[d]+'"]').addClass(this.options.css.selected)._bgAria("selected","true").find(f).prop("checked",!0);this.element.trigger("selected"+H,[e])}}return this},I.prototype.deselect=function(b){if(this.selection){b=b||this.currentRows.propValues(this.identifier);for(var c,d,e,f=[];b.length>0;)if(c=b.pop(),-1!==(e=a.inArray(c,this.selectedRows)))for(d=0;d0){var g=h(this.options.css.selectBox);for(this.element.find("thead "+g).prop("checked",!1),d=0;d tr[data-row-id="'+f[d][this.identifier]+'"]').removeClass(this.options.css.selected)._bgAria("selected","false").find(g).prop("checked",!1);this.element.trigger("deselected"+H,[f])}}return this},I.prototype.sort=function(b){var c=b?a.extend({},b):{};return c===this.sortDictionary?this:(this.sortDictionary=c,C.call(this),G.call(this),n.call(this),this)},I.prototype.getColumnSettings=function(){return a.merge([],this.columns)},I.prototype.getCurrentPage=function(){return this.current},I.prototype.getCurrentRows=function(){return a.merge([],this.currentRows)},I.prototype.getRowCount=function(){return this.rowCount},I.prototype.getSearchPhrase=function(){return this.searchPhrase},I.prototype.getSelectedRows=function(){return a.merge([],this.selectedRows)},I.prototype.getSortDictionary=function(){return a.extend({},this.sortDictionary)},I.prototype.getTotalPageCount=function(){return this.totalPages},I.prototype.getTotalRowCount=function(){return this.total},a.fn.extend({_bgAria:function(a,b){return b?this.attr("aria-"+a,b):this.attr("aria-"+a)},_bgBusyAria:function(a){return null==a||a?this._bgAria("busy","true"):this._bgAria("busy","false")},_bgRemoveAria:function(a){return this.removeAttr("aria-"+a)},_bgEnableAria:function(a){return null==a||a?this.removeClass("disabled")._bgAria("disabled","false"):this.addClass("disabled")._bgAria("disabled","true")},_bgEnableField:function(a){return null==a||a?this.removeAttr("disabled"):this.attr("disabled","disable")},_bgShowAria:function(a){return null==a||a?this.show()._bgAria("hidden","false"):this.hide()._bgAria("hidden","true")},_bgSelectAria:function(a){return null==a||a?this.addClass("active")._bgAria("selected","true"):this.removeClass("active")._bgAria("selected","false")},_bgId:function(a){return a?this.attr("id",a):this.attr("id")}}),!String.prototype.resolve){var J={checked:function(a){return"boolean"==typeof a?a?'checked="checked"':"":a}};String.prototype.resolve=function(b,c){var d=this;return a.each(b,function(b,e){if(null!=e&&"function"!=typeof e)if("object"==typeof e){var f=c?a.extend([],c):[];f.push(b),d=d.resolve(e,f)+""}else{J&&J[b]&&"function"==typeof J[b]&&(e=J[b](e)),b=c?c.join(".")+"."+b:b;var g=new RegExp("\\{\\{"+b+"\\}\\}","gm");d=d.replace(g,e.replace?e.replace(/\$/gi,"$"):e)}}),d}}Array.prototype.first||(Array.prototype.first=function(a){for(var b=0;bc?this.length>d?this.slice(c,d):this.slice(c):[]}),Array.prototype.where||(Array.prototype.where=function(a){for(var b=[],c=0;c Date: Tue, 6 Nov 2018 14:38:42 +0100 Subject: [PATCH 13/17] Update dependencies to 2018 versions --- CHANGELOG.md | 4 ++++ Gruntfile.js | 3 ++- package.json | 32 ++++++++++++++++---------------- 3 files changed, 22 insertions(+), 17 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index fddc843..24c5550 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,9 @@ # Changelog +## 1.3.3 (in development) +### Enhancements & Features +- Update dependencies to 2018 versions, eliminating 72 vulnerabilities. + ## 1.3.2 ### Bug fixes - Fix jQuery and Bootstrap versions in package.json, fix license info in package.json diff --git a/Gruntfile.js b/Gruntfile.js index 3af122a..35b94fa 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -102,6 +102,7 @@ module.exports = function (grunt) { default: { options: { 'adjoining-classes': false, + 'order-alphabetical': false, 'important': false, 'outline-none': false, 'overqualified-elements': false @@ -114,7 +115,7 @@ module.exports = function (grunt) { curly: true, eqeqeq: true, immed: true, - latedef: true, + latedef: 'nofunc', newcap: true, noarg: true, sub: true, diff --git a/package.json b/package.json index 9d5a105..00c7d23 100644 --- a/package.json +++ b/package.json @@ -36,25 +36,25 @@ ], "license": "MIT", "dependencies": { - "bootstrap": "3.1.1 - 3.3", - "jquery": ">=1.9" + "bootstrap": "^3.3.7", + "jquery": "^3.2.1" }, "devDependencies": { - "grunt": "^0.4.5", - "grunt-cli": "~0.1.13", - "grunt-contrib-clean": "~0.5.0", - "grunt-contrib-compress": "~0.8.0", - "grunt-contrib-concat": "~0.4.0", - "grunt-contrib-csslint": "~0.3.1", - "grunt-contrib-cssmin": "~0.10.0", - "grunt-contrib-jshint": "~0.10.0", - "grunt-contrib-less": "~0.11.0", + "grunt": "^1.0.3", + "grunt-cli": "~1.3.2", + "grunt-contrib-clean": "~2.0.0", + "grunt-contrib-compress": "~1.4.3", + "grunt-contrib-concat": "~1.0.1", + "grunt-contrib-csslint": "~2.0.0", + "grunt-contrib-cssmin": "~3.0.0", + "grunt-contrib-jshint": "~2.0.0", + "grunt-contrib-less": "~2.0.0", "grunt-contrib-qunit": "~3.0.1", - "grunt-contrib-uglify": "~0.9.2", - "grunt-contrib-yuidoc": "~0.5.2", - "grunt-exec": "~0.4.5", - "grunt-nuget": "~0.1.4", - "grunt-regex-replace": "~0.2.6" + "grunt-contrib-uglify": "~4.0.0", + "grunt-contrib-yuidoc": "~1.0.0", + "grunt-exec": "~3.0.0", + "grunt-nuget": "~0.3.1", + "grunt-regex-replace": "~0.4.0" }, "readmeFilename": "README.md" } From 86dec85f5bb0493b9b33f584ea084939ef442972 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 14:45:22 +0100 Subject: [PATCH 14/17] Fix typo --- test/tests-internal.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/tests-internal.js b/test/tests-internal.js index 5f88b30..cf46e5d 100644 --- a/test/tests-internal.js +++ b/test/tests-internal.js @@ -81,7 +81,7 @@ QUnit.test("findFooterAndHeaderItems test (footer and header is string empty)", var result = findFooterAndHeaderItems.call(instance, selector); // then - assert.equal(result.length, 0, "Foundd one element as expecte"); + assert.equal(result.length, 0, "Found one element as expected"); assert.ok(result.find, "Got an empty jQuery array as expected"); }); From 2cb624911b5923019f7268c0eebfd0a1e2edd5d1 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 15:35:21 +0100 Subject: [PATCH 15/17] List Deciso B.V. as author, list previous author as contributor --- package.json | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 00c7d23..1fd1828 100644 --- a/package.json +++ b/package.json @@ -6,10 +6,16 @@ "description": "Nice, sleek and intuitive. A grid control especially designed for bootstrap.", "homepage": "http://www.jquery-bootgrid.com", "author": { - "name": "Rafael Staib", - "email": "me@rafaelstaib.com", - "url": "http://www.rafaelstaib.com" + "name": "Deciso B.V.", + "url": "https://www.opnsense.org" }, + "contributors": [ + { + "name": "Rafael Staib", + "email": "me@rafaelstaib.com", + "url": "http://www.rafaelstaib.com" + } + ], "bugs": "https://github.com/opnsense/jquery-bootgrid/issues", "scripts": { "test": "grunt" From e16adf815518292461a8bc50f8a1c0c7021362d0 Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 15:50:56 +0100 Subject: [PATCH 16/17] Remove unused Modernizr file --- CHANGELOG.md | 1 + demo/index.htm | 1 - demo/js/modernizr-2.8.1.js | 4 ---- demo/send.htm | 1 - 4 files changed, 1 insertion(+), 6 deletions(-) delete mode 100644 demo/js/modernizr-2.8.1.js diff --git a/CHANGELOG.md b/CHANGELOG.md index 24c5550..335cd58 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -3,6 +3,7 @@ ## 1.3.3 (in development) ### Enhancements & Features - Update dependencies to 2018 versions, eliminating 72 vulnerabilities. +- The old, unused copy of Modernizr in the demo has been removed. ## 1.3.2 ### Bug fixes diff --git a/demo/index.htm b/demo/index.htm index a4244a5..5d0c37b 100644 --- a/demo/index.htm +++ b/demo/index.htm @@ -6,7 +6,6 @@ jQuery Bootgrid Demo - "].join(""),l.id=h,(m?l:n).innerHTML+=f,n.appendChild(l),m||(n.style.background="",n.style.overflow="hidden",k=g.style.overflow,g.style.overflow="hidden",g.appendChild(n)),i=c(l,a),m?l.parentNode.removeChild(l):(n.parentNode.removeChild(n),g.style.overflow=k),!!i},y={}.hasOwnProperty,z;!C(y,"undefined")&&!C(y.call,"undefined")?z=function(a,b){return y.call(a,b)}:z=function(a,b){return b in a&&C(a.constructor.prototype[b],"undefined")},Function.prototype.bind||(Function.prototype.bind=function(b){var c=this;if(typeof c!="function")throw new TypeError;var d=v.call(arguments,1),e=function(){if(this instanceof e){var a=function(){};a.prototype=c.prototype;var f=new a,g=c.apply(f,d.concat(v.call(arguments)));return Object(g)===g?g:f}return c.apply(b,d.concat(v.call(arguments)))};return e}),r.flexbox=function(){return G("flexWrap")},r.flexboxlegacy=function(){return G("boxDirection")},r.rgba=function(){return A("background-color:rgba(150,255,150,.5)"),D(j.backgroundColor,"rgba")},r.hsla=function(){return A("background-color:hsla(120,40%,100%,.5)"),D(j.backgroundColor,"rgba")||D(j.backgroundColor,"hsla")},r.multiplebgs=function(){return A("background:url(https://),url(https://),red url(https://)"),/(url\s*\(.*?){3}/.test(j.background)},r.backgroundsize=function(){return G("backgroundSize")},r.borderimage=function(){return G("borderImage")},r.borderradius=function(){return G("borderRadius")},r.boxshadow=function(){return G("boxShadow")},r.textshadow=function(){return b.createElement("div").style.textShadow===""},r.opacity=function(){return B("opacity:.55"),/^0.55$/.test(j.opacity)},r.cssanimations=function(){return G("animationName")},r.csscolumns=function(){return G("columnCount")},r.cssgradients=function(){var a="background-image:",b="gradient(linear,left top,right bottom,from(#9f9),to(white));",c="linear-gradient(left top,#9f9, white);";return A((a+"-webkit- ".split(" ").join(b+a)+n.join(c+a)).slice(0,-a.length)),D(j.backgroundImage,"gradient")},r.cssreflections=function(){return G("boxReflect")},r.csstransforms=function(){return!!G("transform")},r.csstransforms3d=function(){var a=!!G("perspective");return a&&"webkitPerspective"in g.style&&x("@media (transform-3d),(-webkit-transform-3d){#modernizr{left:9px;position:absolute;height:3px;}}",function(b,c){a=b.offsetLeft===9&&b.offsetHeight===3}),a},r.csstransitions=function(){return G("transition")},r.fontface=function(){var a;return x('@font-face {font-family:"font";src:url("https://")}',function(c,d){var e=b.getElementById("smodernizr"),f=e.sheet||e.styleSheet,g=f?f.cssRules&&f.cssRules[0]?f.cssRules[0].cssText:f.cssText||"":"";a=/src/i.test(g)&&g.indexOf(d.split(" ")[0])===0}),a},r.generatedcontent=function(){var a;return x(["#",h,"{font:0/0 a}#",h,':after{content:"',l,'";visibility:hidden;font:3px/1 a}'].join(""),function(b){a=b.offsetHeight>=3}),a};for(var H in r)z(r,H)&&(w=H.toLowerCase(),e[w]=r[H](),u.push((e[w]?"":"no-")+w));return e.addTest=function(a,b){if(typeof a=="object")for(var d in a)z(a,d)&&e.addTest(d,a[d]);else{a=a.toLowerCase();if(e[a]!==c)return e;b=typeof b=="function"?b():b,typeof f!="undefined"&&f&&(g.className+=" "+(b?"":"no-")+a),e[a]=b}return e},A(""),i=k=null,function(a,b){function l(a,b){var c=a.createElement("p"),d=a.getElementsByTagName("head")[0]||a.documentElement;return c.innerHTML="x",d.insertBefore(c.lastChild,d.firstChild)}function m(){var a=s.elements;return typeof a=="string"?a.split(" "):a}function n(a){var b=j[a[h]];return b||(b={},i++,a[h]=i,j[i]=b),b}function o(a,c,d){c||(c=b);if(k)return c.createElement(a);d||(d=n(c));var g;return d.cache[a]?g=d.cache[a].cloneNode():f.test(a)?g=(d.cache[a]=d.createElem(a)).cloneNode():g=d.createElem(a),g.canHaveChildren&&!e.test(a)&&!g.tagUrn?d.frag.appendChild(g):g}function p(a,c){a||(a=b);if(k)return a.createDocumentFragment();c=c||n(a);var d=c.frag.cloneNode(),e=0,f=m(),g=f.length;for(;e",g="hidden"in a,k=a.childNodes.length==1||function(){b.createElement("a");var a=b.createDocumentFragment();return typeof a.cloneNode=="undefined"||typeof a.createDocumentFragment=="undefined"||typeof a.createElement=="undefined"}()}catch(c){g=!0,k=!0}})();var s={elements:d.elements||"abbr article aside audio bdi canvas data datalist details dialog figcaption figure footer header hgroup main mark meter nav output progress section summary template time video",version:c,shivCSS:d.shivCSS!==!1,supportsUnknownElements:k,shivMethods:d.shivMethods!==!1,type:"default",shivDocument:r,createElement:o,createDocumentFragment:p};a.html5=s,r(b)}(this,b),e._version=d,e._prefixes=n,e._domPrefixes=q,e._cssomPrefixes=p,e.testProp=function(a){return E([a])},e.testAllProps=G,e.testStyles=x,g.className=g.className.replace(/(^|\s)no-js(\s|$)/,"$1$2")+(f?" js "+u.join(" "):""),e}(this,this.document),function(a,b,c){function d(a){return"[object Function]"==o.call(a)}function e(a){return"string"==typeof a}function f(){}function g(a){return!a||"loaded"==a||"complete"==a||"uninitialized"==a}function h(){var a=p.shift();q=1,a?a.t?m(function(){("c"==a.t?B.injectCss:B.injectJs)(a.s,0,a.a,a.x,a.e,1)},0):(a(),h()):q=0}function i(a,c,d,e,f,i,j){function k(b){if(!o&&g(l.readyState)&&(u.r=o=1,!q&&h(),l.onload=l.onreadystatechange=null,b)){"img"!=a&&m(function(){t.removeChild(l)},50);for(var d in y[c])y[c].hasOwnProperty(d)&&y[c][d].onload()}}var j=j||B.errorTimeout,l=b.createElement(a),o=0,r=0,u={t:d,s:c,e:f,a:i,x:j};1===y[c]&&(r=1,y[c]=[]),"object"==a?l.data=c:(l.src=c,l.type=a),l.width=l.height="0",l.onerror=l.onload=l.onreadystatechange=function(){k.call(this,r)},p.splice(e,0,u),"img"!=a&&(r||2===y[c]?(t.insertBefore(l,s?null:n),m(k,j)):y[c].push(l))}function j(a,b,c,d,f){return q=0,b=b||"j",e(a)?i("c"==b?v:u,a,b,this.i++,c,d,f):(p.splice(this.i++,0,a),1==p.length&&h()),this}function k(){var a=B;return a.loader={load:j,i:0},a}var l=b.documentElement,m=a.setTimeout,n=b.getElementsByTagName("script")[0],o={}.toString,p=[],q=0,r="MozAppearance"in l.style,s=r&&!!b.createRange().compareNode,t=s?l:n.parentNode,l=a.opera&&"[object Opera]"==o.call(a.opera),l=!!b.attachEvent&&!l,u=r?"object":l?"script":"img",v=l?"script":u,w=Array.isArray||function(a){return"[object Array]"==o.call(a)},x=[],y={},z={timeout:function(a,b){return b.length&&(a.timeout=b[0]),a}},A,B;B=function(a){function b(a){var a=a.split("!"),b=x.length,c=a.pop(),d=a.length,c={url:c,origUrl:c,prefixes:a},e,f,g;for(f=0;fjQuery Bootgrid Demo - @-webkit-viewport { width: device-width; From 351ad0a9896088062429bbe05c46ca305afe163c Mon Sep 17 00:00:00 2001 From: Michael Steenbeek Date: Tue, 6 Nov 2018 17:29:47 +0100 Subject: [PATCH 17/17] Release 1.3.3 --- CHANGELOG.md | 2 +- bower.json | 2 +- dist/jQuery.Bootgrid.1.3.2.nupkg | Bin 26227 -> 0 bytes dist/jQuery.Bootgrid.1.3.3.nupkg | Bin 0 -> 25808 bytes dist/jquery.bootgrid-1.3.2.zip | Bin 28898 -> 0 bytes dist/jquery.bootgrid-1.3.3.zip | Bin 0 -> 28467 bytes dist/jquery.bootgrid.css | 4 ++-- dist/jquery.bootgrid.fa.js | 4 ++-- dist/jquery.bootgrid.fa.min.js | 7 +------ dist/jquery.bootgrid.js | 4 ++-- dist/jquery.bootgrid.min.css | 6 +++--- dist/jquery.bootgrid.min.js | 7 +------ package.json | 2 +- 13 files changed, 14 insertions(+), 24 deletions(-) delete mode 100644 dist/jQuery.Bootgrid.1.3.2.nupkg create mode 100644 dist/jQuery.Bootgrid.1.3.3.nupkg delete mode 100644 dist/jquery.bootgrid-1.3.2.zip create mode 100644 dist/jquery.bootgrid-1.3.3.zip diff --git a/CHANGELOG.md b/CHANGELOG.md index 335cd58..4c35228 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,6 @@ # Changelog -## 1.3.3 (in development) +## 1.3.3 ### Enhancements & Features - Update dependencies to 2018 versions, eliminating 72 vulnerabilities. - The old, unused copy of Modernizr in the demo has been removed. diff --git a/bower.json b/bower.json index 08853e5..52513c4 100644 --- a/bower.json +++ b/bower.json @@ -13,7 +13,7 @@ "accessibility", "bootstrap" ], - "version": "1.3.2", + "version": "1.3.3", "authors": [ { "name": "Rafael Staib", diff --git a/dist/jQuery.Bootgrid.1.3.2.nupkg b/dist/jQuery.Bootgrid.1.3.2.nupkg deleted file mode 100644 index aa1a599d518ba5d60f167b7d447616e238d7f89e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 26227 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 z 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 Q2=(Vl`EzAac7J#O5A_ef0RR91 literal 0 HcmV?d00001 diff --git a/dist/jquery.bootgrid.css b/dist/jquery.bootgrid.css index f4be168..c9c6bba 100644 --- a/dist/jquery.bootgrid.css +++ b/dist/jquery.bootgrid.css @@ -1,6 +1,6 @@ /*! - * jQuery Bootgrid v1.3.2 - 11/06/2018 - * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) + * jQuery Bootgrid v1.3.3 - 11/06/2018 + * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Deciso B.V. (http://www.jquery-bootgrid.com) * Licensed under the MIT license. See LICENSE.txt for more details. */ .bootgrid-header, diff --git a/dist/jquery.bootgrid.fa.js b/dist/jquery.bootgrid.fa.js index 53c312b..1bfa4ec 100644 --- a/dist/jquery.bootgrid.fa.js +++ b/dist/jquery.bootgrid.fa.js @@ -1,6 +1,6 @@ /*! - * jQuery Bootgrid v1.3.2 - 11/06/2018 - * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) + * jQuery Bootgrid v1.3.3 - 11/06/2018 + * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Deciso B.V. (http://www.jquery-bootgrid.com) * Licensed under the MIT license. See LICENSE.txt for more details. */ ;(function ($, window, undefined) diff --git a/dist/jquery.bootgrid.fa.min.js b/dist/jquery.bootgrid.fa.min.js index faf00d3..89d0b40 100644 --- a/dist/jquery.bootgrid.fa.min.js +++ b/dist/jquery.bootgrid.fa.min.js @@ -1,6 +1 @@ -/*! - * jQuery Bootgrid v1.3.2 - 11/06/2018 - * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) - * Licensed under the MIT license. See LICENSE.txt for more details. - */ -!function(a,b,c){"use strict";a.extend(a.fn.bootgrid.Constructor.defaults.css,{icon:"icon fa",iconColumns:"fa-th-list",iconDown:"fa-sort-desc",iconRefresh:"fa-refresh",iconSearch:"fa-search",iconUp:"fa-sort-asc"})}(jQuery,window); \ No newline at end of file +!function(o,c,n){"use strict";o.extend(o.fn.bootgrid.Constructor.defaults.css,{icon:"icon fa",iconColumns:"fa-th-list",iconDown:"fa-sort-desc",iconRefresh:"fa-refresh",iconSearch:"fa-search",iconUp:"fa-sort-asc"})}(jQuery,window); \ No newline at end of file diff --git a/dist/jquery.bootgrid.js b/dist/jquery.bootgrid.js index 3c93baa..f0eca3e 100644 --- a/dist/jquery.bootgrid.js +++ b/dist/jquery.bootgrid.js @@ -1,6 +1,6 @@ /*! - * jQuery Bootgrid v1.3.2 - 11/06/2018 - * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) + * jQuery Bootgrid v1.3.3 - 11/06/2018 + * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Deciso B.V. (http://www.jquery-bootgrid.com) * Licensed under the MIT license. See LICENSE.txt for more details. */ ;(function ($, window, undefined) diff --git a/dist/jquery.bootgrid.min.css b/dist/jquery.bootgrid.min.css index a928e13..6ff3670 100644 --- a/dist/jquery.bootgrid.min.css +++ b/dist/jquery.bootgrid.min.css @@ -1,5 +1,5 @@ /*! - * jQuery Bootgrid v1.3.2 - 11/06/2018 - * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) + * jQuery Bootgrid v1.3.3 - 11/06/2018 + * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Deciso B.V. (http://www.jquery-bootgrid.com) * Licensed under the MIT license. See LICENSE.txt for more details. - */.bootgrid-footer,.bootgrid-header{margin:15px 0}.bootgrid-footer a,.bootgrid-header a{outline:0}.bootgrid-footer .search,.bootgrid-header .search{display:inline-block;margin:0 20px 0 0;vertical-align:middle;width:180px}.bootgrid-footer .search .glyphicon,.bootgrid-header .search .glyphicon{top:0}.bootgrid-footer .search .fa,.bootgrid-header .search .fa{display:table-cell}.bootgrid-footer .search .search-field::-ms-clear,.bootgrid-footer .search.search-field::-ms-clear,.bootgrid-header .search .search-field::-ms-clear,.bootgrid-header .search.search-field::-ms-clear{display:none}.bootgrid-footer .pagination,.bootgrid-header .pagination{margin:0!important}.bootgrid-footer .infoBar,.bootgrid-header .actionBar{text-align:right}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu{text-align:left}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item{cursor:pointer;display:block;margin:0;padding:3px 20px;white-space:nowrap}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item:focus,.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item:hover,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item:focus,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item:hover{color:#262626;text-decoration:none;background-color:#f5f5f5}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item .dropdown-item-checkbox,.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item.dropdown-item-checkbox,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item .dropdown-item-checkbox,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item.dropdown-item-checkbox{margin:0 2px 4px 0;vertical-align:middle}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item.disabled,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item.disabled{cursor:not-allowed}.bootgrid-table{table-layout:fixed}.bootgrid-table a{outline:0}.bootgrid-table th>.column-header-anchor{color:#333;cursor:not-allowed;display:block;position:relative;text-decoration:none}.bootgrid-table th>.column-header-anchor.sortable{cursor:pointer}.bootgrid-table th>.column-header-anchor>.text{display:block;margin:0 16px 0 0;overflow:hidden;-ms-text-overflow:ellipsis;-o-text-overflow:ellipsis;text-overflow:ellipsis;white-space:nowrap}.bootgrid-table th>.column-header-anchor>.icon{display:block;position:absolute;right:0;top:2px}.bootgrid-table th:active,.bootgrid-table th:hover{background:#fafafa}.bootgrid-table td{overflow:hidden;-ms-text-overflow:ellipsis;-o-text-overflow:ellipsis;text-overflow:ellipsis;white-space:nowrap}.bootgrid-table td.loading,.bootgrid-table td.no-results{background:#fff;text-align:center}.bootgrid-table td.select-cell,.bootgrid-table th.select-cell{text-align:center;width:30px}.bootgrid-table td.select-cell .select-box,.bootgrid-table th.select-cell .select-box{margin:0;outline:0}.table-responsive .bootgrid-table{table-layout:inherit!important}.table-responsive .bootgrid-table td,.table-responsive .bootgrid-table th>.column-header-anchor>.text{overflow:inherit!important;-ms-text-overflow:inherit!important;-o-text-overflow:inherit!important;text-overflow:inherit!important;white-space:inherit!important} \ No newline at end of file + */.bootgrid-footer,.bootgrid-header{margin:15px 0}.bootgrid-footer a,.bootgrid-header a{outline:0}.bootgrid-footer .search,.bootgrid-header .search{display:inline-block;margin:0 20px 0 0;vertical-align:middle;width:180px}.bootgrid-footer .search .glyphicon,.bootgrid-header .search .glyphicon{top:0}.bootgrid-footer .search .fa,.bootgrid-header .search .fa{display:table-cell}.bootgrid-footer .search .search-field::-ms-clear,.bootgrid-footer .search.search-field::-ms-clear,.bootgrid-header .search .search-field::-ms-clear,.bootgrid-header .search.search-field::-ms-clear{display:none}.bootgrid-footer .pagination,.bootgrid-header .pagination{margin:0!important}.bootgrid-footer .infoBar,.bootgrid-header .actionBar{text-align:right}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu{text-align:left}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item{cursor:pointer;display:block;margin:0;padding:3px 20px;white-space:nowrap}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item:focus,.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item:hover,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item:focus,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item:hover{color:#262626;text-decoration:none;background-color:#f5f5f5}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item .dropdown-item-checkbox,.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item.dropdown-item-checkbox,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item .dropdown-item-checkbox,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item.dropdown-item-checkbox{margin:0 2px 4px 0;vertical-align:middle}.bootgrid-footer .infoBar .btn-group>.btn-group .dropdown-menu .dropdown-item.disabled,.bootgrid-header .actionBar .btn-group>.btn-group .dropdown-menu .dropdown-item.disabled{cursor:not-allowed}.bootgrid-table{table-layout:fixed}.bootgrid-table a{outline:0}.bootgrid-table th>.column-header-anchor{color:#333;cursor:not-allowed;display:block;position:relative;text-decoration:none}.bootgrid-table th>.column-header-anchor.sortable{cursor:pointer}.bootgrid-table th>.column-header-anchor>.text{display:block;margin:0 16px 0 0;overflow:hidden;-ms-text-overflow:ellipsis;-o-text-overflow:ellipsis;text-overflow:ellipsis;white-space:nowrap}.bootgrid-table th>.column-header-anchor>.icon{display:block;position:absolute;right:0;top:2px}.bootgrid-table th:active,.bootgrid-table th:hover{background:#fafafa}.bootgrid-table td{overflow:hidden;-ms-text-overflow:ellipsis;-o-text-overflow:ellipsis;text-overflow:ellipsis;white-space:nowrap}.bootgrid-table td.loading,.bootgrid-table td.no-results{background:#fff;text-align:center}.bootgrid-table td.select-cell,.bootgrid-table th.select-cell{text-align:center;width:30px}.bootgrid-table td.select-cell .select-box,.bootgrid-table th.select-cell .select-box{margin:0;outline:0}.table-responsive .bootgrid-table{table-layout:inherit!important}.table-responsive .bootgrid-table th>.column-header-anchor>.text{overflow:inherit!important;-ms-text-overflow:inherit!important;-o-text-overflow:inherit!important;text-overflow:inherit!important;white-space:inherit!important}.table-responsive .bootgrid-table td{overflow:inherit!important;-ms-text-overflow:inherit!important;-o-text-overflow:inherit!important;text-overflow:inherit!important;white-space:inherit!important} \ No newline at end of file diff --git a/dist/jquery.bootgrid.min.js b/dist/jquery.bootgrid.min.js index 59069d6..956ae25 100644 --- a/dist/jquery.bootgrid.min.js +++ b/dist/jquery.bootgrid.min.js @@ -1,6 +1 @@ -/*! - * jQuery Bootgrid v1.3.2 - 11/06/2018 - * Copyright © 2014-2015 Rafael J. Staib; Copyright © 2018-2018 Rafael Staib (http://www.jquery-bootgrid.com) - * Licensed under the MIT license. See LICENSE.txt for more details. - */ -!function(a,b,c){"use strict";function d(a){function b(b){return c.identifier&&b[c.identifier]===a[c.identifier]}var c=this;return!this.rows.contains(b)&&(this.rows.push(a),!0)}function e(b){var c=this.footer?this.footer.find(b):a(),d=this.header?this.header.find(b):a();return a.merge(c,d)}function f(b){return b?a.extend({},this.cachedParams,{ctx:b}):this.cachedParams}function g(){var b={current:this.current,rowCount:this.rowCount,sort:this.sortDictionary,searchPhrase:this.searchPhrase},c=this.options.post;return c=a.isFunction(c)?c():c,this.options.requestHandler(a.extend(!0,b,c))}function h(b){return"."+a.trim(b).replace(/\s+/gm,".")}function i(){var b=this.options.url;return a.isFunction(b)?b():b}function j(){this.element.trigger("initialize"+H),m.call(this),this.selection=this.options.selection&&null!=this.identifier,o.call(this),q.call(this),C.call(this),A.call(this),r.call(this),n.call(this),this.element.trigger("initialized"+H)}function k(a){this.options.highlightRows}function l(a){return a.visible}function m(){var b=this,c=this.element.find("thead > tr").first(),d=!1;c.children().each(function(){var c=a(this),e=c.data(),f={id:e.columnId,identifier:null==b.identifier&&e.identifier||!1,converter:b.options.converters[e.converter||e.type]||b.options.converters.string,text:c.text(),align:e.align||"left",headerAlign:e.headerAlign||"left",cssClass:e.cssClass||"",headerCssClass:e.headerCssClass||"",formatter:b.options.formatters[e.formatter]||null,order:d||"asc"!==e.order&&"desc"!==e.order?null:e.order,searchable:!(!1===e.searchable),sortable:!(!1===e.sortable),visible:!(!1===e.visible),visibleInSelection:!(!1===e.visibleInSelection),width:a.isNumeric(e.width)?e.width+"px":"string"==typeof e.width?e.width:null};b.columns.push(f),null!=f.order&&(b.sortDictionary[f.id]=f.order),f.identifier&&(b.identifier=f.id,b.converter=f.converter),b.options.multiSort||null===f.order||(d=!0)})}function n(){function c(a){for(var b,c=new RegExp(e.searchPhrase,e.options.caseSensitive?"g":"gi"),d=0;d-1)return!0;return!1}function d(a,b){e.currentRows=a,p.call(e,b),e.options.keepSelection||(e.selectedRows=[]),y.call(e,a),t.call(e),v.call(e),e.element._bgBusyAria(!1).trigger("loaded"+H)}var e=this;if(this.element._bgBusyAria(!0).trigger("load"+H),F.call(this),this.options.ajax){var f=g.call(this),h=i.call(this);if(null==h||"string"!=typeof h||0===h.length)throw new Error("Url setting must be a none empty string or a function that returns one.");this.xqr&&this.xqr.abort();var j={url:h,data:f,success:function(b){e.xqr=null,"string"==typeof b&&(b=a.parseJSON(b)),b=e.options.responseHandler(b),e.current=b.current,d(b.rows,b.total)},error:function(a,b,c){e.xqr=null,"abort"!==b&&(u.call(e),e.element._bgBusyAria(!1).trigger("loaded"+H))}};j=a.extend(this.options.ajaxSettings,j),this.xqr=a.ajax(j)}else{var k=this.searchPhrase.length>0?this.rows.where(c):this.rows,l=k.length;-1!==this.rowCount&&(k=k.page(this.current,this.rowCount)),b.setTimeout(function(){d(k,l)},10)}}function o(){if(!this.options.ajax){var b=this;this.element.find("tbody > tr").each(function(){var c=a(this),e=c.children("td"),f={};a.each(b.columns,function(a,b){f[b.id]=b.converter.from(e.eq(a).text())}),d.call(b,f)}),p.call(this,this.rows.length),G.call(this)}}function p(a){this.total=a,this.totalPages=-1===this.rowCount?1:Math.ceil(this.total/this.rowCount)}function q(){var b=this.options.templates,c=this.element.parent().hasClass(this.options.css.responsiveTable)?this.element.parent():this.element;this.element.addClass(this.options.css.table),0===this.element.children("tbody").length&&this.element.append(b.body),1&this.options.navigation&&(this.header=a(b.header.resolve(f.call(this,{id:this.element._bgId()+"-header"}))),c.before(this.header)),2&this.options.navigation&&(this.footer=a(b.footer.resolve(f.call(this,{id:this.element._bgId()+"-footer"}))),c.after(this.footer))}function r(){if(0!==this.options.navigation){var b=this.options.css,c=h(b.actions),d=e.call(this,c);if(d.length>0){var g=this,i=this.options.templates,j=a(i.actions.resolve(f.call(this)));if(this.options.ajax){var k=i.icon.resolve(f.call(this,{iconCss:b.iconRefresh})),l=a(i.actionButton.resolve(f.call(this,{content:k,text:this.options.labels.refresh}))).on("click"+H,function(a){a.stopPropagation(),g.current=1,n.call(g)});j.append(l)}x.call(this,j),s.call(this,j),E.call(this,d,j)}}}function s(b){if(this.options.columnSelection&&this.columns.length>1){var c=this,d=this.options.css,e=this.options.templates,g=e.icon.resolve(f.call(this,{iconCss:d.iconColumns})),i=a(e.actionDropDown.resolve(f.call(this,{content:g}))),j=h(d.dropDownItem),k=h(d.dropDownItemCheckbox),m=h(d.dropDownMenuItems);a.each(this.columns,function(b,g){if(g.visibleInSelection){var o=a(e.actionDropDownCheckboxItem.resolve(f.call(c,{name:g.id,label:g.text,checked:g.visible}))).on("click"+H,j,function(b){b.stopPropagation();var d=a(this),e=d.find(k);if(!e.prop("disabled")){g.visible=e.prop("checked");var f=c.columns.where(l).length>1;d.parents(m).find(j+":has("+k+":checked)")._bgEnableAria(f).find(k)._bgEnableField(f),c.element.find("tbody").empty(),C.call(c),n.call(c)}});i.find(h(d.dropDownMenuItems)).append(o)}}),b.append(i)}}function t(){if(0!==this.options.navigation){var b=h(this.options.css.infos),c=e.call(this,b);if(c.length>0){var d=this.current*this.rowCount,g=a(this.options.templates.infos.resolve(f.call(this,{end:0===this.total||-1===d||d>this.total?this.total:d,start:0===this.total?0:d-this.rowCount+1,total:this.total})));E.call(this,c,g)}}}function u(){var a=this.element.children("tbody").first(),b=this.options.templates,c=this.columns.where(l).length;this.selection&&(c+=1),a.html(b.noResults.resolve(f.call(this,{columns:c})))}function v(){if(0!==this.options.navigation){var b=h(this.options.css.pagination),c=e.call(this,b)._bgShowAria(-1!==this.rowCount);if(-1!==this.rowCount&&c.length>0){var d=this.options.templates,g=this.current,i=this.totalPages,j=a(d.pagination.resolve(f.call(this))),k=i-g,l=-1*(this.options.padding-g),m=k>=this.options.padding?Math.max(l,1):Math.max(l-this.options.padding+k,1),n=2*this.options.padding+1,o=i>=n?n:i;w.call(this,j,"first","«","first")._bgEnableAria(g>1),w.call(this,j,"prev","‹","prev")._bgEnableAria(g>1);for(var p=0;pg),w.call(this,j,"last","»","last")._bgEnableAria(i>g),E.call(this,c,j)}}}function w(b,c,d,e){var g=this,i=this.options.templates,j=this.options.css,k=f.call(this,{css:e,text:d,page:c}),l=a(i.paginationItem.resolve(k)).on("click"+H,h(j.paginationButton),function(b){b.stopPropagation(),b.preventDefault();var c=a(this),d=c.parent();if(!d.hasClass("active")&&!d.hasClass("disabled")){var e={first:1,prev:g.current-1,next:g.current+1,last:g.totalPages},f=c.data("page");g.current=e[f]||f,n.call(g)}c.trigger("blur")});return b.append(l),l}function x(b){function c(a){return-1===a?d.options.labels.all:a}var d=this,e=this.options.rowCount;if(a.isArray(e)){var g=this.options.css,i=this.options.templates,j=a(i.actionDropDown.resolve(f.call(this,{content:c(this.rowCount)}))),k=h(g.dropDownMenu),l=h(g.dropDownMenuText),m=h(g.dropDownMenuItems),o=h(g.dropDownItemButton);a.each(e,function(b,e){var g=a(i.actionDropDownItem.resolve(f.call(d,{text:c(e),action:e})))._bgSelectAria(e===d.rowCount).on("click"+H,o,function(b){b.preventDefault();var e=a(this),f=e.data("action");f!==d.rowCount&&(d.current=1,d.rowCount=f,e.parents(m).children().each(function(){var b=a(this),c=b.find(o).data("action");b._bgSelectAria(c===f)}),e.parents(k).find(l).text(c(f)),n.call(d))});j.find(m).append(g)}),b.append(j)}}function y(b){if(b.length>0){var c=this,d=this.options.css,e=this.options.templates,g=this.element.children("tbody").first(),i=!0,j="";a.each(b,function(b,g){var h="",k=' data-row-id="'+(null==c.identifier?b:g[c.identifier])+'"',l="";if(c.selection){var m=-1!==a.inArray(g[c.identifier],c.selectedRows),n=e.select.resolve(f.call(c,{type:"checkbox",value:g[c.identifier],checked:m}));h+=e.cell.resolve(f.call(c,{content:n,css:d.selectCell})),i=i&&m,m&&(l+=d.selected,k+=' aria-selected="true"')}var o=null!=g.status&&c.options.statusMapping[g.status];o&&(l+=o),a.each(c.columns,function(b,i){if(i.visible){var j=a.isFunction(i.formatter)?i.formatter.call(c,i,g):i.converter.to(g[i.id]),k=i.cssClass.length>0?" "+i.cssClass:"";h+=e.cell.resolve(f.call(c,{content:null==j||""===j?" ":j,css:("right"===i.align?d.right:"center"===i.align?d.center:d.left)+k,style:null==i.width?"":"width:"+i.width+";"}))}}),l.length>0&&(k+=' class="'+l+'"'),j+=e.row.resolve(f.call(c,{attr:k,cells:h}))}),c.element.find("thead "+h(c.options.css.selectBox)).prop("checked",i),g.html(j),z.call(this,g)}else u.call(this)}function z(b){var c=this,d=h(this.options.css.selectBox);this.selection&&!this.options.rowSelect&&b.off("click"+H,d).on("click"+H,d,function(b){b.stopPropagation();var d=a(this),e=c.converter.from(d.val());d.prop("checked")?c.select([e]):c.deselect([e])}),b.off("click"+H,"> tr").on("click"+H,"> tr",function(b){b.stopPropagation();var d=a(this),e=null==c.identifier?d.data("row-id"):c.converter.from(d.data("row-id")+""),f=null==c.identifier?c.currentRows[e]:c.currentRows.first(function(a){return a[c.identifier]===e});c.selection&&c.options.rowSelect&&(d.hasClass(c.options.css.selected)?c.deselect([e]):c.select([e])),c.element.trigger("click"+H,[c.columns,f])})}function A(){if(0!==this.options.navigation){var c=this.options.css,d=h(c.search),g=e.call(this,d);if(g.length>0){var i=this,j=this.options.templates,k=null,l="",m=h(c.searchField),n=a(j.search.resolve(f.call(this)));(n.is(m)?n:n.find(m)).on("keyup"+H,function(c){c.stopPropagation();var d=a(this).val();(l!==d||13===c.which&&""!==d)&&(l=d,(13===c.which||0===d.length||d.length>=i.options.searchSettings.characters)&&(b.clearTimeout(k),k=b.setTimeout(function(){B.call(i,d)},i.options.searchSettings.delay)))}),E.call(this,g,n)}}}function B(a){this.searchPhrase!==a&&(this.current=1,this.searchPhrase=a,n.call(this))}function C(){var b=this,c=this.element.find("thead > tr"),d=this.options.css,e=this.options.templates,g="",i=this.options.sorting;if(this.selection){var j=this.options.multiSelect?e.select.resolve(f.call(b,{type:"checkbox",value:"all"})):"";g+=e.rawHeaderCell.resolve(f.call(b,{content:j,css:d.selectCell}))}if(a.each(this.columns,function(a,c){if(c.visible){var h=b.sortDictionary[c.id],j=i&&h&&"asc"===h?d.iconUp:i&&h&&"desc"===h?d.iconDown:"",k=e.icon.resolve(f.call(b,{iconCss:j})),l=c.headerAlign,m=c.headerCssClass.length>0?" "+c.headerCssClass:"";g+=e.headerCell.resolve(f.call(b,{column:c,icon:k,sortable:i&&c.sortable&&d.sortable||"",css:("right"===l?d.right:"center"===l?d.center:d.left)+m,style:null==c.width?"":"width:"+c.width+";"}))}}),c.html(g),i){var k=h(d.sortable);c.off("click"+H,k).on("click"+H,k,function(c){c.preventDefault(),D.call(b,a(this)),G.call(b),n.call(b)})}if(this.selection&&this.options.multiSelect){var l=h(d.selectBox);c.off("click"+H,l).on("click"+H,l,function(c){c.stopPropagation(),a(this).prop("checked")?b.select():b.deselect()})}}function D(a){var b=this.options.css,c=h(b.icon),d=a.data("column-id")||a.parents("th").first().data("column-id"),e=this.sortDictionary[d],f=a.find(c);if(this.options.multiSort||(a.parents("tr").first().find(c).removeClass(b.iconDown+" "+b.iconUp),this.sortDictionary={}),e&&"asc"===e)this.sortDictionary[d]="desc",f.removeClass(b.iconUp).addClass(b.iconDown);else if(e&&"desc"===e)if(this.options.multiSort){var g={};for(var i in this.sortDictionary)i!==d&&(g[i]=this.sortDictionary[i]);this.sortDictionary=g,f.removeClass(b.iconDown)}else this.sortDictionary[d]="asc",f.removeClass(b.iconDown).addClass(b.iconUp);else this.sortDictionary[d]="asc",f.addClass(b.iconUp)}function E(b,c){b.each(function(b,d){a(d).before(c.clone(!0)).remove()})}function F(){var a=this;b.setTimeout(function(){if("true"===a.element._bgAria("busy")){var b=a.options.templates,c=a.element.children("thead").first(),d=a.element.children("tbody").first(),e=d.find("tr > td").first(),g=a.element.height()-c.height()-(e.height()+20),h=a.columns.where(l).length;a.selection&&(h+=1),d.html(b.loading.resolve(f.call(a,{columns:h}))),-1!==a.rowCount&&g>0&&d.find("tr > td").css("padding","20px 0 "+g+"px")}},250)}function G(){function a(c,d,e){function f(a){return"asc"===h.order?a:-1*a}e=e||0;var g=e+1,h=b[e];return c[h.id]>d[h.id]?f(1):c[h.id]g?a(c,d,g):0}var b=[];if(!this.options.ajax){for(var c in this.sortDictionary)(this.options.multiSort||0===b.length)&&b.push({id:c,order:this.sortDictionary[c]});b.length>0&&this.rows.sort(a)}}var H=".rs.jquery.bootgrid",I=function(b,c){this.element=a(b),this.origin=this.element.clone(),this.options=a.extend(!0,{},I.defaults,this.element.data(),c);var d=this.options.rowCount=this.element.data().rowCount||c.rowCount||this.options.rowCount;this.columns=[],this.current=1,this.currentRows=[],this.identifier=null,this.selection=!1,this.converter=null,this.rowCount=a.isArray(d)?d[0]:d,this.rows=[],this.searchPhrase="",this.selectedRows=[],this.sortDictionary={},this.total=0,this.totalPages=0,this.cachedParams={lbl:this.options.labels,css:this.options.css,ctx:{}},this.header=null,this.footer=null,this.xqr=null};if(I.defaults={navigation:3,padding:2,columnSelection:!0,rowCount:[10,25,50,-1],selection:!1,multiSelect:!1,rowSelect:!1,keepSelection:!1,highlightRows:!1,sorting:!0,multiSort:!1,searchSettings:{delay:250,characters:1},ajax:!1,ajaxSettings:{method:"POST"},post:{},url:"",caseSensitive:!0,requestHandler:function(a){return a},responseHandler:function(a){return a},converters:{numeric:{from:function(a){return+a},to:function(a){return a+""}},string:{from:function(a){return a},to:function(a){return a}}},css:{actions:"actions btn-group",center:"text-center",columnHeaderAnchor:"column-header-anchor",columnHeaderText:"text",dropDownItem:"dropdown-item",dropDownItemButton:"dropdown-item-button",dropDownItemCheckbox:"dropdown-item-checkbox",dropDownMenu:"dropdown btn-group",dropDownMenuItems:"dropdown-menu pull-right",dropDownMenuText:"dropdown-text",footer:"bootgrid-footer container-fluid",header:"bootgrid-header container-fluid",icon:"icon glyphicon",iconColumns:"glyphicon-th-list",iconDown:"glyphicon-chevron-down",iconRefresh:"glyphicon-refresh",iconSearch:"glyphicon-search",iconUp:"glyphicon-chevron-up",infos:"infos",left:"text-left",pagination:"pagination",paginationButton:"button",responsiveTable:"table-responsive",right:"text-right",search:"search form-group",searchField:"search-field form-control",selectBox:"select-box",selectCell:"select-cell",selected:"active",sortable:"sortable",table:"bootgrid-table table"},formatters:{},labels:{all:"All",infos:"Showing {{ctx.start}} to {{ctx.end}} of {{ctx.total}} entries",loading:"Loading...",noResults:"No results found!",refresh:"Refresh",search:"Search"},statusMapping:{0:"success",1:"info",2:"warning",3:"danger"},templates:{actionButton:'',actionDropDown:'
                    ',actionDropDownItem:'
                  • {{ctx.text}}
                  • ',actionDropDownCheckboxItem:'
                  • ',actions:'
                    ',body:"",cell:'{{ctx.content}}',footer:'

                    ',header:'

                    ',headerCell:'{{ctx.column.text}}{{ctx.icon}}',icon:'',infos:'
                    {{lbl.infos}}
                    ',loading:'{{lbl.loading}}',noResults:'{{lbl.noResults}}',pagination:'
                      ',paginationItem:'
                    • {{ctx.text}}
                    • ',rawHeaderCell:'{{ctx.content}}',row:"{{ctx.cells}}",search:'
                      ',select:''}},I.prototype.append=function(a){if(this.options.ajax);else{for(var b=[],c=0;c0&&(this.options.multiSelect||1!==e.length);)if(c=b.pop(),-1===a.inArray(c,this.selectedRows))for(d=0;d0){var f=h(this.options.css.selectBox),g=this.selectedRows.length>=this.currentRows.length;for(d=0;!this.options.keepSelection&&g&&d tr "+f+":checked").trigger("click"+H),d=0;d tr[data-row-id="'+this.selectedRows[d]+'"]').addClass(this.options.css.selected)._bgAria("selected","true").find(f).prop("checked",!0);this.element.trigger("selected"+H,[e])}}return this},I.prototype.deselect=function(b){if(this.selection){b=b||this.currentRows.propValues(this.identifier);for(var c,d,e,f=[];b.length>0;)if(c=b.pop(),-1!==(e=a.inArray(c,this.selectedRows)))for(d=0;d0){var g=h(this.options.css.selectBox);for(this.element.find("thead "+g).prop("checked",!1),d=0;d tr[data-row-id="'+f[d][this.identifier]+'"]').removeClass(this.options.css.selected)._bgAria("selected","false").find(g).prop("checked",!1);this.element.trigger("deselected"+H,[f])}}return this},I.prototype.sort=function(b){var c=b?a.extend({},b):{};return c===this.sortDictionary?this:(this.sortDictionary=c,C.call(this),G.call(this),n.call(this),this)},I.prototype.getColumnSettings=function(){return a.merge([],this.columns)},I.prototype.getCurrentPage=function(){return this.current},I.prototype.getCurrentRows=function(){return a.merge([],this.currentRows)},I.prototype.getRowCount=function(){return this.rowCount},I.prototype.getSearchPhrase=function(){return this.searchPhrase},I.prototype.getSelectedRows=function(){return a.merge([],this.selectedRows)},I.prototype.getSortDictionary=function(){return a.extend({},this.sortDictionary)},I.prototype.getTotalPageCount=function(){return this.totalPages},I.prototype.getTotalRowCount=function(){return this.total},a.fn.extend({_bgAria:function(a,b){return b?this.attr("aria-"+a,b):this.attr("aria-"+a)},_bgBusyAria:function(a){return null==a||a?this._bgAria("busy","true"):this._bgAria("busy","false")},_bgRemoveAria:function(a){return this.removeAttr("aria-"+a)},_bgEnableAria:function(a){return null==a||a?this.removeClass("disabled")._bgAria("disabled","false"):this.addClass("disabled")._bgAria("disabled","true")},_bgEnableField:function(a){return null==a||a?this.removeAttr("disabled"):this.attr("disabled","disable")},_bgShowAria:function(a){return null==a||a?this.show()._bgAria("hidden","false"):this.hide()._bgAria("hidden","true")},_bgSelectAria:function(a){return null==a||a?this.addClass("active")._bgAria("selected","true"):this.removeClass("active")._bgAria("selected","false")},_bgId:function(a){return a?this.attr("id",a):this.attr("id")}}),!String.prototype.resolve){var J={checked:function(a){return"boolean"==typeof a?a?'checked="checked"':"":a}};String.prototype.resolve=function(b,c){var d=this;return a.each(b,function(b,e){if(null!=e&&"function"!=typeof e)if("object"==typeof e){var f=c?a.extend([],c):[];f.push(b),d=d.resolve(e,f)+""}else{J&&J[b]&&"function"==typeof J[b]&&(e=J[b](e)),b=c?c.join(".")+"."+b:b;var g=new RegExp("\\{\\{"+b+"\\}\\}","gm");d=d.replace(g,e.replace?e.replace(/\$/gi,"$"):e)}}),d}}Array.prototype.first||(Array.prototype.first=function(a){for(var b=0;bc?this.length>d?this.slice(c,d):this.slice(c):[]}),Array.prototype.where||(Array.prototype.where=function(a){for(var b=[],c=0;c tr").first(),o=!1;t.children().each(function(){var t=p(this),e=t.data(),i={id:e.columnId,identifier:null==s.identifier&&e.identifier||!1,converter:s.options.converters[e.converter||e.type]||s.options.converters.string,text:t.text(),align:e.align||"left",headerAlign:e.headerAlign||"left",cssClass:e.cssClass||"",headerCssClass:e.headerCssClass||"",formatter:s.options.formatters[e.formatter]||null,order:o||"asc"!==e.order&&"desc"!==e.order?null:e.order,searchable:!(!1===e.searchable),sortable:!(!1===e.sortable),visible:!(!1===e.visible),visibleInSelection:!(!1===e.visibleInSelection),width:p.isNumeric(e.width)?e.width+"px":"string"==typeof e.width?e.width:null};s.columns.push(i),null!=i.order&&(s.sortDictionary[i.id]=i.order),i.identifier&&(s.identifier=i.id,s.converter=i.converter),s.options.multiSort||null===i.order||(o=!0)})}.call(this),this.selection=this.options.selection&&null!=this.identifier,function(){if(!this.options.ajax){var e=this,t=this.element.find("tbody > tr");t.each(function(){var t=p(this),i=t.children("td"),s={};p.each(e.columns,function(t,e){s[e.id]=e.converter.from(i.eq(t).text())}),o.call(e,s)}),a.call(this,this.rows.length),n.call(this)}}.call(this),function(){var t=this.options.templates,e=this.element.parent().hasClass(this.options.css.responsiveTable)?this.element.parent():this.element;this.element.addClass(this.options.css.table),0===this.element.children("tbody").length&&this.element.append(t.body);1&this.options.navigation&&(this.header=p(t.header.resolve(v.call(this,{id:this.element._bgId()+"-header"}))),e.before(this.header));2&this.options.navigation&&(this.footer=p(t.footer.resolve(v.call(this,{id:this.element._bgId()+"-footer"}))),e.after(this.footer))}.call(this),x.call(this),function(){if(0!==this.options.navigation){var t=this.options.css,e=m(t.search),i=g.call(this,e);if(0=s.options.searchSettings.characters)&&(h.clearTimeout(n),n=h.setTimeout(function(){y.call(s,e)},s.options.searchSettings.delay)))}),C.call(this,i,a)}}}.call(this),function(){if(0!==this.options.navigation){var t=this.options.css,e=m(t.actions),i=g.call(this,e);if(0 tr").on("click"+f,"> tr",function(t){t.stopPropagation();var e=p(this),i=null==o.identifier?e.data("row-id"):o.converter.from(e.data("row-id")+""),s=null==o.identifier?o.currentRows[i]:o.currentRows.first(function(t){return t[o.identifier]===i});o.selection&&o.options.rowSelect&&(e.hasClass(o.options.css.selected)?o.deselect([i]):o.select([i])),o.element.trigger("click"+f,[o.columns,s])})}.call(this,e)}else b.call(this)}.call(o,t),function(){if(0!==this.options.navigation){var t=m(this.options.css.infos),e=g.call(this,t);if(0this.total?this.total:i,start:0===this.total?0:i-this.rowCount+1,total:this.total})));C.call(this,e,s)}}}.call(o),function(){if(0!==this.options.navigation){var t=m(this.options.css.pagination),e=g.call(this,t)._bgShowAria(-1!==this.rowCount);if(-1!==this.rowCount&&0=this.options.padding?Math.max(l,1):Math.max(l-this.options.padding+r,1),c=2*this.options.padding+1,h=c<=o?c:o;w.call(this,n,"first","«","first")._bgEnableAria(1 td").first(),o=r.element.height()-e.height()-(s.height()+20),n=r.columns.where(d).length;r.selection&&(n+=1),i.html(t.loading.resolve(v.call(r,{columns:n}))),-1!==r.rowCount&&0 td").css("padding","20px 0 "+o+"px")}},250)}.call(this),this.options.ajax){var t=function(){var t={current:this.current,rowCount:this.rowCount,sort:this.sortDictionary,searchPhrase:this.searchPhrase},e=this.options.post;return e=p.isFunction(e)?e():e,this.options.requestHandler(p.extend(!0,t,e))}.call(this),i=function(){var t=this.options.url;return p.isFunction(t)?t():t}.call(this);if(null==i||"string"!=typeof i||0===i.length)throw new Error("Url setting must be a none empty string or a function that returns one.");this.xqr&&this.xqr.abort();var s={url:i,data:t,success:function(t){o.xqr=null,"string"==typeof t&&(t=p.parseJSON(t)),t=o.options.responseHandler(t),o.current=t.current,e(t.rows,t.total)},error:function(t,e,i){o.xqr=null,"abort"!==e&&(b.call(o),o.element._bgBusyAria(!1).trigger("loaded"+f))}};s=p.extend(this.options.ajaxSettings,s),this.xqr=p.ajax(s)}else{var n=0 tr"),a=this.options.css,c=this.options.templates,h="",d=this.options.sorting;if(this.selection){var e=this.options.multiSelect?c.select.resolve(v.call(l,{type:"checkbox",value:"all"})):"";h+=c.rawHeaderCell.resolve(v.call(l,{content:e,css:a.selectCell}))}if(p.each(this.columns,function(t,e){if(e.visible){var i=l.sortDictionary[e.id],s=d&&i&&"asc"===i?a.iconUp:d&&i&&"desc"===i?a.iconDown:"",o=c.icon.resolve(v.call(l,{iconCss:s})),n=e.headerAlign,r=0i[n.id]?r(1):e[n.id]o?t(e,i,o):0})}}var c=function(t,e){this.element=p(t),this.origin=this.element.clone(),this.options=p.extend(!0,{},c.defaults,this.element.data(),e);var i=this.options.rowCount=this.element.data().rowCount||e.rowCount||this.options.rowCount;this.columns=[],this.current=1,this.currentRows=[],this.identifier=null,this.selection=!1,this.converter=null,this.rowCount=p.isArray(i)?i[0]:i,this.rows=[],this.searchPhrase="",this.selectedRows=[],this.sortDictionary={},this.total=0,this.totalPages=0,this.cachedParams={lbl:this.options.labels,css:this.options.css,ctx:{}},this.header=null,this.footer=null,this.xqr=null};if(c.defaults={navigation:3,padding:2,columnSelection:!0,rowCount:[10,25,50,-1],selection:!1,multiSelect:!1,rowSelect:!1,keepSelection:!1,highlightRows:!1,sorting:!0,multiSort:!1,searchSettings:{delay:250,characters:1},ajax:!1,ajaxSettings:{method:"POST"},post:{},url:"",caseSensitive:!0,requestHandler:function(t){return t},responseHandler:function(t){return t},converters:{numeric:{from:function(t){return+t},to:function(t){return t+""}},string:{from:function(t){return t},to:function(t){return t}}},css:{actions:"actions btn-group",center:"text-center",columnHeaderAnchor:"column-header-anchor",columnHeaderText:"text",dropDownItem:"dropdown-item",dropDownItemButton:"dropdown-item-button",dropDownItemCheckbox:"dropdown-item-checkbox",dropDownMenu:"dropdown btn-group",dropDownMenuItems:"dropdown-menu pull-right",dropDownMenuText:"dropdown-text",footer:"bootgrid-footer container-fluid",header:"bootgrid-header container-fluid",icon:"icon glyphicon",iconColumns:"glyphicon-th-list",iconDown:"glyphicon-chevron-down",iconRefresh:"glyphicon-refresh",iconSearch:"glyphicon-search",iconUp:"glyphicon-chevron-up",infos:"infos",left:"text-left",pagination:"pagination",paginationButton:"button",responsiveTable:"table-responsive",right:"text-right",search:"search form-group",searchField:"search-field form-control",selectBox:"select-box",selectCell:"select-cell",selected:"active",sortable:"sortable",table:"bootgrid-table table"},formatters:{},labels:{all:"All",infos:"Showing {{ctx.start}} to {{ctx.end}} of {{ctx.total}} entries",loading:"Loading...",noResults:"No results found!",refresh:"Refresh",search:"Search"},statusMapping:{0:"success",1:"info",2:"warning",3:"danger"},templates:{actionButton:'',actionDropDown:'
                      ',actionDropDownItem:'
                    • {{ctx.text}}
                    • ',actionDropDownCheckboxItem:'
                    • ',actions:'
                      ',body:"",cell:'{{ctx.content}}',footer:'

                      ',header:'

                      ',headerCell:'{{ctx.column.text}}{{ctx.icon}}',icon:'',infos:'
                      {{lbl.infos}}
                      ',loading:'{{lbl.loading}}',noResults:'{{lbl.noResults}}',pagination:'
                        ',paginationItem:'
                      • {{ctx.text}}
                      • ',rawHeaderCell:'{{ctx.content}}',row:"{{ctx.cells}}",search:'
                        ',select:''}},c.prototype.append=function(t){if(this.options.ajax);else{for(var e=[],i=0;i=this.currentRows.length;for(i=0;!this.options.keepSelection&&n&&i tr "+o+":checked").trigger("click"+f),i=0;i tr[data-row-id="'+this.selectedRows[i]+'"]').addClass(this.options.css.selected)._bgAria("selected","true").find(o).prop("checked",!0);this.element.trigger("selected"+f,[s])}}return this},c.prototype.deselect=function(t){if(this.selection){t=t||this.currentRows.propValues(this.identifier);for(var e,i,s,o=[];0 tr[data-row-id="'+o[i][this.identifier]+'"]').removeClass(this.options.css.selected)._bgAria("selected","false").find(n).prop("checked",!1);this.element.trigger("deselected"+f,[o])}}return this},c.prototype.sort=function(t){var e=t?p.extend({},t):{};return e===this.sortDictionary||(this.sortDictionary=e,x.call(this),n.call(this),u.call(this)),this},c.prototype.getColumnSettings=function(){return p.merge([],this.columns)},c.prototype.getCurrentPage=function(){return this.current},c.prototype.getCurrentRows=function(){return p.merge([],this.currentRows)},c.prototype.getRowCount=function(){return this.rowCount},c.prototype.getSearchPhrase=function(){return this.searchPhrase},c.prototype.getSelectedRows=function(){return p.merge([],this.selectedRows)},c.prototype.getSortDictionary=function(){return p.extend({},this.sortDictionary)},c.prototype.getTotalPageCount=function(){return this.totalPages},c.prototype.getTotalRowCount=function(){return this.total},p.fn.extend({_bgAria:function(t,e){return e?this.attr("aria-"+t,e):this.attr("aria-"+t)},_bgBusyAria:function(t){return null==t||t?this._bgAria("busy","true"):this._bgAria("busy","false")},_bgRemoveAria:function(t){return this.removeAttr("aria-"+t)},_bgEnableAria:function(t){return null==t||t?this.removeClass("disabled")._bgAria("disabled","false"):this.addClass("disabled")._bgAria("disabled","true")},_bgEnableField:function(t){return null==t||t?this.removeAttr("disabled"):this.attr("disabled","disable")},_bgShowAria:function(t){return null==t||t?this.show()._bgAria("hidden","false"):this.hide()._bgAria("hidden","true")},_bgSelectAria:function(t){return null==t||t?this.addClass("active")._bgAria("selected","true"):this.removeClass("active")._bgAria("selected","false")},_bgId:function(t){return t?this.attr("id",t):this.attr("id")}}),!String.prototype.resolve){var r={checked:function(t){return"boolean"==typeof t?t?'checked="checked"':"":t}};String.prototype.resolve=function(t,o){var n=this;return p.each(t,function(t,e){if(null!=e&&"function"!=typeof e)if("object"==typeof e){var i=o?p.extend([],o):[];i.push(t),n=n.resolve(e,i)+""}else{r&&r[t]&&"function"==typeof r[t]&&(e=r[t](e)),t=o?o.join(".")+"."+t:t;var s=new RegExp("\\{\\{"+t+"\\}\\}","gm");n=n.replace(s,e.replace?e.replace(/\$/gi,"$"):e)}}),n}}Array.prototype.first||(Array.prototype.first=function(t){for(var e=0;ei?this.length>s?this.slice(i,s):this.slice(i):[]}),Array.prototype.where||(Array.prototype.where=function(t){for(var e=[],i=0;i