summaryrefslogtreecommitdiff
path: root/sitestatic/jquery.tablesorter-2.7.min.js
diff options
context:
space:
mode:
Diffstat (limited to 'sitestatic/jquery.tablesorter-2.7.min.js')
-rw-r--r--sitestatic/jquery.tablesorter-2.7.min.js1401
1 files changed, 43 insertions, 1358 deletions
diff --git a/sitestatic/jquery.tablesorter-2.7.min.js b/sitestatic/jquery.tablesorter-2.7.min.js
index f8c443ab..eff1e09f 100644
--- a/sitestatic/jquery.tablesorter-2.7.min.js
+++ b/sitestatic/jquery.tablesorter-2.7.min.js
@@ -14,1361 +14,46 @@
* @author Christian Bach/christian.bach@polyester.se
* @contributor Rob Garrison/https://github.com/Mottie/tablesorter
*/
-/*jshint browser:true, jquery:true, unused:false, expr: true */
-/*global console:false, alert:false */
-!(function($) {
- "use strict";
- $.extend({
- /*jshint supernew:true */
- tablesorter: new function() {
-
- var ts = this;
-
- ts.version = "2.7";
-
- ts.parsers = [];
- ts.widgets = [];
- ts.defaults = {
-
- // *** appearance
- theme : 'default', // adds tablesorter-{theme} to the table for styling
- widthFixed : false, // adds colgroup to fix widths of columns
- showProcessing : false, // show an indeterminate timer icon in the header when the table is sorted or filtered.
-
- headerTemplate : '{content}',// header layout template (HTML ok); {content} = innerHTML, {icon} = <i/> (class from cssIcon)
- onRenderTemplate : null, // function(index, template){ return template; }, (template is a string)
- onRenderHeader : null, // function(index){}, (nothing to return)
-
- // *** functionality
- cancelSelection : true, // prevent text selection in the header
- dateFormat : 'mmddyyyy', // other options: "ddmmyyy" or "yyyymmdd"
- sortMultiSortKey : 'shiftKey', // key used to select additional columns
- sortResetKey : 'ctrlKey', // key used to remove sorting on a column
- usNumberFormat : true, // false for German "1.234.567,89" or French "1 234 567,89"
- delayInit : false, // if false, the parsed table contents will not update until the first sort
- serverSideSorting: false, // if true, server-side sorting should be performed because client-side sorting will be disabled, but the ui and events will still be used.
-
- // *** sort options
- headers : {}, // set sorter, string, empty, locked order, sortInitialOrder, filter, etc.
- ignoreCase : true, // ignore case while sorting
- sortForce : null, // column(s) first sorted; always applied
- sortList : [], // Initial sort order; applied initially; updated when manually sorted
- sortAppend : null, // column(s) sorted last; always applied
-
- sortInitialOrder : 'asc', // sort direction on first click
- sortLocaleCompare: false, // replace equivalent character (accented characters)
- sortReset : false, // third click on the header will reset column to default - unsorted
- sortRestart : false, // restart sort to "sortInitialOrder" when clicking on previously unsorted columns
-
- emptyTo : 'bottom', // sort empty cell to bottom, top, none, zero
- stringTo : 'max', // sort strings in numerical column as max, min, top, bottom, zero
- textExtraction : 'simple', // text extraction method/function - function(node, table, cellIndex){}
- textSorter : null, // use custom text sorter - function(a,b){ return a.sort(b); } // basic sort
-
- // *** widget options
- widgets: [], // method to add widgets, e.g. widgets: ['zebra']
- widgetOptions : {
- zebra : [ 'even', 'odd' ] // zebra widget alternating row class names
- },
- initWidgets : true, // apply widgets on tablesorter initialization
-
- // *** callbacks
- initialized : null, // function(table){},
-
- // *** css class names
- tableClass : 'tablesorter',
- cssAsc : 'tablesorter-headerAsc',
- cssChildRow : 'tablesorter-childRow', // previously "expand-child"
- cssDesc : 'tablesorter-headerDesc',
- cssHeader : 'tablesorter-header',
- cssHeaderRow : 'tablesorter-headerRow',
- cssIcon : 'tablesorter-icon', // if this class exists, a <i> will be added to the header automatically
- cssInfoBlock : 'tablesorter-infoOnly', // don't sort tbody with this class name
- cssProcessing : 'tablesorter-processing', // processing icon applied to header during sort/filter
-
- // *** selectors
- selectorHeaders : '> thead th, > thead td',
- selectorSort : 'th, td', // jQuery selector of content within selectorHeaders that is clickable to trigger a sort
- selectorRemove : '.remove-me',
-
- // *** advanced
- debug : false,
-
- // *** Internal variables
- headerList: [],
- empties: {},
- strings: {},
- parsers: []
-
- // deprecated; but retained for backwards compatibility
- // widgetZebra: { css: ["even", "odd"] }
-
- };
-
- /* debuging utils */
- function log(s) {
- if (typeof console !== "undefined" && typeof console.log !== "undefined") {
- console.log(s);
- } else {
- alert(s);
- }
- }
-
- function benchmark(s, d) {
- log(s + " (" + (new Date().getTime() - d.getTime()) + "ms)");
- }
-
- ts.benchmark = benchmark;
-
- function getElementText(table, node, cellIndex) {
- if (!node) { return ""; }
- var c = table.config,
- t = c.textExtraction, text = "";
- if (t === "simple") {
- if (c.supportsTextContent) {
- text = node.textContent; // newer browsers support this
- } else {
- text = $(node).text();
- }
- } else {
- if (typeof(t) === "function") {
- text = t(node, table, cellIndex);
- } else if (typeof(t) === "object" && t.hasOwnProperty(cellIndex)) {
- text = t[cellIndex](node, table, cellIndex);
- } else {
- text = c.supportsTextContent ? node.textContent : $(node).text();
- }
- }
- return $.trim(text);
- }
-
- function detectParserForColumn(table, rows, rowIndex, cellIndex) {
- var i, l = ts.parsers.length,
- node = false,
- nodeValue = '',
- keepLooking = true;
- while (nodeValue === '' && keepLooking) {
- rowIndex++;
- if (rows[rowIndex]) {
- node = rows[rowIndex].cells[cellIndex];
- nodeValue = getElementText(table, node, cellIndex);
- if (table.config.debug) {
- log('Checking if value was empty on row ' + rowIndex + ', column: ' + cellIndex + ': ' + nodeValue);
- }
- } else {
- keepLooking = false;
- }
- }
- for (i = 1; i < l; i++) {
- if (ts.parsers[i].is(nodeValue, table, node)) {
- return ts.parsers[i];
- }
- }
- // 0 is always the generic parser (text)
- return ts.parsers[0];
- }
-
- function buildParserCache(table) {
- var c = table.config,
- tb = $(table.tBodies).filter(':not(.' + c.cssInfoBlock + ')'),
- rows, list, l, i, h, ch, p, parsersDebug = "";
- if ( tb.length === 0) {
- return c.debug ? log('*Empty table!* Not building a parser cache') : '';
- }
- rows = tb[0].rows;
- if (rows[0]) {
- list = [];
- l = rows[0].cells.length;
- for (i = 0; i < l; i++) {
- // tons of thanks to AnthonyM1229 for working out the following selector (issue #74) to make this work in IE8!
- // More fixes to this selector to work properly in iOS and jQuery 1.8+ (issue #132 & #174)
- h = c.$headers.filter(':not([colspan])');
- h = h.add( c.$headers.filter('[colspan="1"]') ) // ie8 fix
- .filter('[data-column="' + i + '"]:last');
- ch = c.headers[i];
- // get column parser
- p = ts.getParserById( ts.getData(h, ch, 'sorter') );
- // empty cells behaviour - keeping emptyToBottom for backwards compatibility
- c.empties[i] = ts.getData(h, ch, 'empty') || c.emptyTo || (c.emptyToBottom ? 'bottom' : 'top' );
- // text strings behaviour in numerical sorts
- c.strings[i] = ts.getData(h, ch, 'string') || c.stringTo || 'max';
- if (!p) {
- p = detectParserForColumn(table, rows, -1, i);
- }
- if (c.debug) {
- parsersDebug += "column:" + i + "; parser:" + p.id + "; string:" + c.strings[i] + '; empty: ' + c.empties[i] + "\n";
- }
- list.push(p);
- }
- }
- if (c.debug) {
- log(parsersDebug);
- }
- return list;
- }
-
- /* utils */
- function buildCache(table) {
- var b = table.tBodies,
- tc = table.config,
- totalRows,
- totalCells,
- parsers = tc.parsers,
- t, v, i, j, k, c, cols, cacheTime, colMax = [];
- tc.cache = {};
- // if no parsers found, return - it's an empty table.
- if (!parsers) {
- return tc.debug ? log('*Empty table!* Not building a cache') : '';
- }
- if (tc.debug) {
- cacheTime = new Date();
- }
- // processing icon
- if (tc.showProcessing) {
- ts.isProcessing(table, true);
- }
- for (k = 0; k < b.length; k++) {
- tc.cache[k] = { row: [], normalized: [] };
- // ignore tbodies with class name from css.cssInfoBlock
- if (!$(b[k]).hasClass(tc.cssInfoBlock)) {
- totalRows = (b[k] && b[k].rows.length) || 0;
- totalCells = (b[k].rows[0] && b[k].rows[0].cells.length) || 0;
- for (i = 0; i < totalRows; ++i) {
- /** Add the table data to main data array */
- c = $(b[k].rows[i]);
- cols = [];
- // if this is a child row, add it to the last row's children and continue to the next row
- if (c.hasClass(tc.cssChildRow)) {
- tc.cache[k].row[tc.cache[k].row.length - 1] = tc.cache[k].row[tc.cache[k].row.length - 1].add(c);
- // go to the next for loop
- continue;
- }
- tc.cache[k].row.push(c);
- for (j = 0; j < totalCells; ++j) {
- t = getElementText(table, c[0].cells[j], j);
- // allow parsing if the string is empty, previously parsing would change it to zero,
- // in case the parser needs to extract data from the table cell attributes
- v = parsers[j].format(t, table, c[0].cells[j], j);
- cols.push(v);
- if ((parsers[j].type || '').toLowerCase() === "numeric") {
- colMax[j] = Math.max(Math.abs(v), colMax[j] || 0); // determine column max value (ignore sign)
- }
- }
- cols.push(tc.cache[k].normalized.length); // add position for rowCache
- tc.cache[k].normalized.push(cols);
- }
- tc.cache[k].colMax = colMax;
- }
- }
- if (tc.showProcessing) {
- ts.isProcessing(table); // remove processing icon
- }
- if (tc.debug) {
- benchmark("Building cache for " + totalRows + " rows", cacheTime);
- }
- }
-
- // init flag (true) used by pager plugin to prevent widget application
- function appendToTable(table, init) {
- var c = table.config,
- b = table.tBodies,
- rows = [],
- c2 = c.cache,
- r, n, totalRows, checkCell, $bk, $tb,
- i, j, k, l, pos, appendTime;
- if (!c2[0]) { return; } // empty table - fixes #206
- if (c.debug) {
- appendTime = new Date();
- }
- for (k = 0; k < b.length; k++) {
- $bk = $(b[k]);
- if (!$bk.hasClass(c.cssInfoBlock)) {
- // get tbody
- $tb = ts.processTbody(table, $bk, true);
- r = c2[k].row;
- n = c2[k].normalized;
- totalRows = n.length;
- checkCell = totalRows ? (n[0].length - 1) : 0;
- for (i = 0; i < totalRows; i++) {
- pos = n[i][checkCell];
- rows.push(r[pos]);
- // removeRows used by the pager plugin
- if (!c.appender || !c.removeRows) {
- l = r[pos].length;
- for (j = 0; j < l; j++) {
- $tb.append(r[pos][j]);
- }
- }
- }
- // restore tbody
- ts.processTbody(table, $tb, false);
- }
- }
- if (c.appender) {
- c.appender(table, rows);
- }
- if (c.debug) {
- benchmark("Rebuilt table", appendTime);
- }
- // apply table widgets
- if (!init) { ts.applyWidget(table); }
- // trigger sortend
- $(table).trigger("sortEnd", table);
- }
-
- // computeTableHeaderCellIndexes from:
- // http://www.javascripttoolbox.com/lib/table/examples.php
- // http://www.javascripttoolbox.com/temp/table_cellindex.html
- function computeThIndexes(t) {
- var matrix = [],
- lookup = {},
- trs = $(t).find('thead:eq(0), tfoot').children('tr'), // children tr in tfoot - see issue #196
- i, j, k, l, c, cells, rowIndex, cellId, rowSpan, colSpan, firstAvailCol, matrixrow;
- for (i = 0; i < trs.length; i++) {
- cells = trs[i].cells;
- for (j = 0; j < cells.length; j++) {
- c = cells[j];
- rowIndex = c.parentNode.rowIndex;
- cellId = rowIndex + "-" + c.cellIndex;
- rowSpan = c.rowSpan || 1;
- colSpan = c.colSpan || 1;
- if (typeof(matrix[rowIndex]) === "undefined") {
- matrix[rowIndex] = [];
- }
- // Find first available column in the first row
- for (k = 0; k < matrix[rowIndex].length + 1; k++) {
- if (typeof(matrix[rowIndex][k]) === "undefined") {
- firstAvailCol = k;
- break;
- }
- }
- lookup[cellId] = firstAvailCol;
- // add data-column
- $(c).attr({ 'data-column' : firstAvailCol }); // 'data-row' : rowIndex
- for (k = rowIndex; k < rowIndex + rowSpan; k++) {
- if (typeof(matrix[k]) === "undefined") {
- matrix[k] = [];
- }
- matrixrow = matrix[k];
- for (l = firstAvailCol; l < firstAvailCol + colSpan; l++) {
- matrixrow[l] = "x";
- }
- }
- }
- }
- return lookup;
- }
-
- function formatSortingOrder(v) {
- // look for "d" in "desc" order; return true
- return (/^d/i.test(v) || v === 1);
- }
-
- function buildHeaders(table) {
- var header_index = computeThIndexes(table), ch, $t,
- h, i, t, lock, time, $tableHeaders, c = table.config;
- c.headerList = [], c.headerContent = [];
- if (c.debug) {
- time = new Date();
- }
- i = c.cssIcon ? '<i class="' + c.cssIcon + '"></i>' : ''; // add icon if cssIcon option exists
- $tableHeaders = $(table).find(c.selectorHeaders).each(function(index) {
- $t = $(this);
- ch = c.headers[index];
- c.headerContent[index] = this.innerHTML; // save original header content
- // set up header template
- t = c.headerTemplate.replace(/\{content\}/g, this.innerHTML).replace(/\{icon\}/g, i);
- if (c.onRenderTemplate) {
- h = c.onRenderTemplate.apply($t, [index, t]);
- if (h && typeof h === 'string') { t = h; } // only change t if something is returned
- }
- this.innerHTML = '<div class="tablesorter-header-inner">' + t + '</div>'; // faster than wrapInner
-
- if (c.onRenderHeader) { c.onRenderHeader.apply($t, [index]); }
-
- this.column = header_index[this.parentNode.rowIndex + "-" + this.cellIndex];
- this.order = formatSortingOrder( ts.getData($t, ch, 'sortInitialOrder') || c.sortInitialOrder ) ? [1,0,2] : [0,1,2];
- this.count = -1; // set to -1 because clicking on the header automatically adds one
- if (ts.getData($t, ch, 'sorter') === 'false') {
- this.sortDisabled = true;
- $t.addClass('sorter-false');
- } else {
- $t.removeClass('sorter-false');
- }
- this.lockedOrder = false;
- lock = ts.getData($t, ch, 'lockedOrder') || false;
- if (typeof(lock) !== 'undefined' && lock !== false) {
- this.order = this.lockedOrder = formatSortingOrder(lock) ? [1,1,1] : [0,0,0];
- }
- $t.addClass( (this.sortDisabled ? 'sorter-false ' : ' ') + c.cssHeader );
- // add cell to headerList
- c.headerList[index] = this;
- // add to parent in case there are multiple rows
- $t.parent().addClass(c.cssHeaderRow);
- });
- if (table.config.debug) {
- benchmark("Built headers:", time);
- log($tableHeaders);
- }
- return $tableHeaders;
- }
-
- function setHeadersCss(table) {
- var f, i, j, l,
- c = table.config,
- list = c.sortList,
- css = [c.cssAsc, c.cssDesc],
- // find the footer
- $t = $(table).find('tfoot tr').children().removeClass(css.join(' '));
- // remove all header information
- c.$headers.removeClass(css.join(' '));
- l = list.length;
- for (i = 0; i < l; i++) {
- // direction = 2 means reset!
- if (list[i][1] !== 2) {
- // multicolumn sorting updating - choose the :last in case there are nested columns
- f = c.$headers.not('.sorter-false').filter('[data-column="' + list[i][0] + '"]' + (l === 1 ? ':last' : '') );
- if (f.length) {
- for (j = 0; j < f.length; j++) {
- if (!f[j].sortDisabled) {
- f.eq(j).addClass(css[list[i][1]]);
- // add sorted class to footer, if it exists
- if ($t.length) {
- $t.filter('[data-column="' + list[i][0] + '"]').eq(j).addClass(css[list[i][1]]);
- }
- }
- }
- }
- }
- }
- }
-
- function fixColumnWidth(table) {
- if (table.config.widthFixed && $(table).find('colgroup').length === 0) {
- var colgroup = $('<colgroup>'),
- overallWidth = $(table).width();
- $("tr:first td", table.tBodies[0]).each(function() {
- colgroup.append($('<col>').css('width', parseInt(($(this).width()/overallWidth)*1000, 10)/10 + '%'));
- });
- $(table).prepend(colgroup);
- }
- }
-
- function updateHeaderSortCount(table, list) {
- var s, t, o, c = table.config,
- sl = list || c.sortList;
- c.sortList = [];
- $.each(sl, function(i,v){
- // ensure all sortList values are numeric - fixes #127
- s = [ parseInt(v[0], 10), parseInt(v[1], 10) ];
- // make sure header exists
- o = c.headerList[s[0]];
- if (o) { // prevents error if sorton array is wrong
- c.sortList.push(s);
- t = $.inArray(s[1], o.order); // fixes issue #167
- o.count = t >= 0 ? t : s[1] % (c.sortReset ? 3 : 2);
- }
- });
- }
-
- function getCachedSortType(parsers, i) {
- return (parsers && parsers[i]) ? parsers[i].type || '' : '';
- }
-
- // sort multiple columns
- function multisort(table) { /*jshint loopfunc:true */
- var dynamicExp, sortWrapper, col, mx = 0, dir = 0, tc = table.config,
- sortList = tc.sortList, l = sortList.length, bl = table.tBodies.length,
- sortTime, i, j, k, c, colMax, cache, lc, s, e, order, orgOrderCol;
- if (tc.serverSideSorting || !tc.cache[0]) { // empty table - fixes #206
- return;
- }
- if (tc.debug) { sortTime = new Date(); }
- for (k = 0; k < bl; k++) {
- colMax = tc.cache[k].colMax;
- cache = tc.cache[k].normalized;
- lc = cache.length;
- orgOrderCol = (cache && cache[0]) ? cache[0].length - 1 : 0;
- cache.sort(function(a, b) {
- // cache is undefined here in IE, so don't use it!
- for (i = 0; i < l; i++) {
- c = sortList[i][0];
- order = sortList[i][1];
- // fallback to natural sort since it is more robust
- s = /n/i.test(getCachedSortType(tc.parsers, c)) ? "Numeric" : "Text";
- s += order === 0 ? "" : "Desc";
- if (/Numeric/.test(s) && tc.strings[c]) {
- // sort strings in numerical columns
- if (typeof (tc.string[tc.strings[c]]) === 'boolean') {
- dir = (order === 0 ? 1 : -1) * (tc.string[tc.strings[c]] ? -1 : 1);
- } else {
- dir = (tc.strings[c]) ? tc.string[tc.strings[c]] || 0 : 0;
- }
- }
- var sort = $.tablesorter["sort" + s](table, a[c], b[c], c, colMax[c], dir);
- if (sort) { return sort; }
- }
- return a[orgOrderCol] - b[orgOrderCol];
- });
- }
- if (tc.debug) { benchmark("Sorting on " + sortList.toString() + " and dir " + order + " time", sortTime); }
- }
-
- function resortComplete($table, callback){
- $table.trigger('updateComplete');
- if (typeof callback === "function") {
- callback($table[0]);
- }
- }
-
- function checkResort($table, flag, callback) {
- if (flag !== false) {
- $table.trigger("sorton", [$table[0].config.sortList, function(){
- resortComplete($table, callback);
- }]);
- } else {
- resortComplete($table, callback);
- }
- }
-
- /* public methods */
- ts.construct = function(settings) {
- return this.each(function() {
- // if no thead or tbody, or tablesorter is already present, quit
- if (!this.tHead || this.tBodies.length === 0 || this.hasInitialized === true) {
- return (this.config.debug) ? log('stopping initialization! No thead, tbody or tablesorter has already been initialized') : '';
- }
- // declare
- var $cell, $this = $(this),
- c, i, j, k = '', a, s, o, downTime,
- m = $.metadata;
- // initialization flag
- this.hasInitialized = false;
- // new blank config object
- this.config = {};
- // merge and extend
- c = $.extend(true, this.config, ts.defaults, settings);
- // save the settings where they read
- $.data(this, "tablesorter", c);
- if (c.debug) { $.data( this, 'startoveralltimer', new Date()); }
- // constants
- c.supportsTextContent = $('<span>x</span>')[0].textContent === 'x';
- c.supportsDataObject = parseFloat($.fn.jquery) >= 1.4;
- // digit sort text location; keeping max+/- for backwards compatibility
- c.string = { 'max': 1, 'min': -1, 'max+': 1, 'max-': -1, 'zero': 0, 'none': 0, 'null': 0, 'top': true, 'bottom': false };
- // add table theme class only if there isn't already one there
- if (!/tablesorter\-/.test($this.attr('class'))) {
- k = (c.theme !== '' ? ' tablesorter-' + c.theme : '');
- }
- $this.addClass(c.tableClass + k);
- // build headers
- c.$headers = buildHeaders(this);
- // try to auto detect column type, and store in tables config
- c.parsers = buildParserCache(this);
- // build the cache for the tbody cells
- // delayInit will delay building the cache until the user starts a sort
- if (!c.delayInit) { buildCache(this); }
- // apply event handling to headers
- // this is to big, perhaps break it out?
- c.$headers
- // http://stackoverflow.com/questions/5312849/jquery-find-self
- .find('*').andSelf().filter(c.selectorSort)
- .unbind('mousedown.tablesorter mouseup.tablesorter')
- .bind('mousedown.tablesorter mouseup.tablesorter', function(e, external) {
- // jQuery v1.2.6 doesn't have closest()
- var $cell = this.tagName.match('TH|TD') ? $(this) : $(this).parents('th, td').filter(':last'), cell = $cell[0];
- // only recognize left clicks
- if ((e.which || e.button) !== 1) { return false; }
- // set timer on mousedown
- if (e.type === 'mousedown') {
- downTime = new Date().getTime();
- return e.target.tagName === "INPUT" ? '' : !c.cancelSelection;
- }
- // ignore long clicks (prevents resizable widget from initializing a sort)
- if (external !== true && (new Date().getTime() - downTime > 250)) { return false; }
- if (c.delayInit && !c.cache) { buildCache($this[0]); }
- if (!cell.sortDisabled) {
- // Only call sortStart if sorting is enabled
- $this.trigger("sortStart", $this[0]);
- // store exp, for speed
- // $cell = $(this);
- k = !e[c.sortMultiSortKey];
- // get current column sort order
- cell.count = e[c.sortResetKey] ? 2 : (cell.count + 1) % (c.sortReset ? 3 : 2);
- // reset all sorts on non-current column - issue #30
- if (c.sortRestart) {
- i = cell;
- c.$headers.each(function() {
- // only reset counts on columns that weren't just clicked on and if not included in a multisort
- if (this !== i && (k || !$(this).is('.' + c.cssDesc + ',.' + c.cssAsc))) {
- this.count = -1;
- }
- });
- }
- // get current column index
- i = cell.column;
- // user only wants to sort on one column
- if (k) {
- // flush the sort list
- c.sortList = [];
- if (c.sortForce !== null) {
- a = c.sortForce;
- for (j = 0; j < a.length; j++) {
- if (a[j][0] !== i) {
- c.sortList.push(a[j]);
- }
- }
- }
- // add column to sort list
- o = cell.order[cell.count];
- if (o < 2) {
- c.sortList.push([i, o]);
- // add other columns if header spans across multiple
- if (cell.colSpan > 1) {
- for (j = 1; j < cell.colSpan; j++) {
- c.sortList.push([i + j, o]);
- }
- }
- }
- // multi column sorting
- } else {
- // get rid of the sortAppend before adding more - fixes issue #115
- if (c.sortAppend && c.sortList.length > 1) {
- if (ts.isValueInArray(c.sortAppend[0][0], c.sortList)) {
- c.sortList.pop();
- }
- }
- // the user has clicked on an already sorted column
- if (ts.isValueInArray(i, c.sortList)) {
- // reverse the sorting direction for all tables
- for (j = 0; j < c.sortList.length; j++) {
- s = c.sortList[j];
- o = c.headerList[s[0]];
- if (s[0] === i) {
- s[1] = o.order[o.count];
- if (s[1] === 2) {
- c.sortList.splice(j,1);
- o.count = -1;
- }
- }
- }
- } else {
- // add column to sort list array
- o = cell.order[cell.count];
- if (o < 2) {
- c.sortList.push([i, o]);
- // add other columns if header spans across multiple
- if (cell.colSpan > 1) {
- for (j = 1; j < cell.colSpan; j++) {
- c.sortList.push([i + j, o]);
- }
- }
- }
- }
- }
- if (c.sortAppend !== null) {
- a = c.sortAppend;
- for (j = 0; j < a.length; j++) {
- if (a[j][0] !== i) {
- c.sortList.push(a[j]);
- }
- }
- }
- // sortBegin event triggered immediately before the sort
- $this.trigger("sortBegin", $this[0]);
- // setTimeout needed so the processing icon shows up
- setTimeout(function(){
- // set css for headers
- setHeadersCss($this[0]);
- multisort($this[0]);
- appendToTable($this[0]);
- }, 1);
- }
- });
- if (c.cancelSelection) {
- // cancel selection
- c.$headers.each(function() {
- this.onselectstart = function() {
- return false;
- };
- });
- }
- // apply easy methods that trigger binded events
- $this
- .unbind('sortReset update updateCell addRows sorton appendCache applyWidgetId applyWidgets refreshWidgets destroy mouseup mouseleave')
- .bind("sortReset", function(){
- c.sortList = [];
- setHeadersCss(this);
- multisort(this);
- appendToTable(this);
- })
- .bind("update", function(e, resort, callback) {
- // remove rows/elements before update
- $(c.selectorRemove, this).remove();
- // rebuild parsers
- c.parsers = buildParserCache(this);
- // rebuild the cache map
- buildCache(this);
- checkResort($this, resort, callback);
- })
- .bind("updateCell", function(e, cell, resort, callback) {
- // get position from the dom
- var l, row, icell,
- t = this, $tb = $(this).find('tbody'),
- // update cache - format: function(s, table, cell, cellIndex)
- // no closest in jQuery v1.2.6 - tbdy = $tb.index( $(cell).closest('tbody') ),$row = $(cell).closest('tr');
- tbdy = $tb.index( $(cell).parents('tbody').filter(':last') ),
- $row = $(cell).parents('tr').filter(':last');
- cell = $(cell)[0]; // in case cell is a jQuery object
- // tbody may not exist if update is initialized while tbody is removed for processing
- if ($tb.length && tbdy >= 0) {
- row = $tb.eq(tbdy).find('tr').index( $row );
- icell = cell.cellIndex;
- l = t.config.cache[tbdy].normalized[row].length - 1;
- t.config.cache[tbdy].row[t.config.cache[tbdy].normalized[row][l]] = $row;
- t.config.cache[tbdy].normalized[row][icell] = c.parsers[icell].format( getElementText(t, cell, icell), t, cell, icell );
- checkResort($this, resort, callback);
- }
- })
- .bind("addRows", function(e, $row, resort, callback) {
- var i, rows = $row.filter('tr').length,
- dat = [], l = $row[0].cells.length, t = this,
- tbdy = $(this).find('tbody').index( $row.closest('tbody') );
- // fixes adding rows to an empty table - see issue #179
- if (!c.parsers) {
- c.parsers = buildParserCache(t);
- }
- // add each row
- for (i = 0; i < rows; i++) {
- // add each cell
- for (j = 0; j < l; j++) {
- dat[j] = c.parsers[j].format( getElementText(t, $row[i].cells[j], j), t, $row[i].cells[j], j );
- }
- // add the row index to the end
- dat.push(c.cache[tbdy].row.length);
- // update cache
- c.cache[tbdy].row.push([$row[i]]);
- c.cache[tbdy].normalized.push(dat);
- dat = [];
- }
- // resort using current settings
- checkResort($this, resort, callback);
- })
- .bind("sorton", function(e, list, callback, init) {
- $(this).trigger("sortStart", this);
- // update header count index
- updateHeaderSortCount(this, list);
- // set css for headers
- setHeadersCss(this);
- // sort the table and append it to the dom
- multisort(this);
- appendToTable(this, init);
- if (typeof callback === "function") {
- callback(this);
- }
- })
- .bind("appendCache", function(e, callback, init) {
- appendToTable(this, init);
- if (typeof callback === "function") {
- callback(this);
- }
- })
- .bind("applyWidgetId", function(e, id) {
- ts.getWidgetById(id).format(this, c, c.widgetOptions);
- })
- .bind("applyWidgets", function(e, init) {
- // apply widgets
- ts.applyWidget(this, init);
- })
- .bind("refreshWidgets", function(e, all, dontapply){
- ts.refreshWidgets(this, all, dontapply);
- })
- .bind("destroy", function(e, c, cb){
- ts.destroy(this, c, cb);
- });
-
- // get sort list from jQuery data or metadata
- // in jQuery < 1.4, an error occurs when calling $this.data()
- if (c.supportsDataObject && typeof $this.data().sortlist !== 'undefined') {
- c.sortList = $this.data().sortlist;
- } else if (m && ($this.metadata() && $this.metadata().sortlist)) {
- c.sortList = $this.metadata().sortlist;
- }
- // apply widget init code
- ts.applyWidget(this, true);
- // if user has supplied a sort list to constructor
- if (c.sortList.length > 0) {
- $this.trigger("sorton", [c.sortList, {}, !c.initWidgets]);
- } else if (c.initWidgets) {
- // apply widget format
- ts.applyWidget(this);
- }
-
- // fixate columns if the users supplies the fixedWidth option
- // do this after theme has been applied
- fixColumnWidth(this);
-
- // show processesing icon
- if (c.showProcessing) {
- $this
- .unbind('sortBegin sortEnd')
- .bind('sortBegin sortEnd', function(e) {
- ts.isProcessing($this[0], e.type === 'sortBegin');
- });
- }
-
- // initialized
- this.hasInitialized = true;
- if (c.debug) {
- ts.benchmark("Overall initialization time", $.data( this, 'startoveralltimer'));
- }
- $this.trigger('tablesorter-initialized', this);
- if (typeof c.initialized === 'function') { c.initialized(this); }
- });
- };
-
- // *** Process table ***
- // add processing indicator
- ts.isProcessing = function(table, toggle, $ths) {
- var c = table.config,
- // default to all headers
- $h = $ths || $(table).find('.' + c.cssHeader);
- if (toggle) {
- if (c.sortList.length > 0) {
- // get headers from the sortList
- $h = $h.filter(function(){
- // get data-column from attr to keep compatibility with jQuery 1.2.6
- return this.sortDisabled ? false : ts.isValueInArray( parseFloat($(this).attr('data-column')), c.sortList);
- });
- }
- $h.addClass(c.cssProcessing);
- } else {
- $h.removeClass(c.cssProcessing);
- }
- };
-
- // detach tbody but save the position
- // don't use tbody because there are portions that look for a tbody index (updateCell)
- ts.processTbody = function(table, $tb, getIt){
- var t, holdr;
- if (getIt) {
- $tb.before('<span class="tablesorter-savemyplace"/>');
- holdr = ($.fn.detach) ? $tb.detach() : $tb.remove();
- return holdr;
- }
- holdr = $(table).find('span.tablesorter-savemyplace');
- $tb.insertAfter( holdr );
- holdr.remove();
- };
-
- ts.clearTableBody = function(table) {
- $(table.tBodies).filter(':not(.' + table.config.cssInfoBlock + ')').empty();
- };
-
- ts.destroy = function(table, removeClasses, callback){
- var $t = $(table), c = table.config,
- $h = $t.find('thead:first');
- // clear flag in case the plugin is initialized again
- table.hasInitialized = false;
- // remove widget added rows
- $h.find('tr:not(.' + c.cssHeaderRow + ')').remove();
- // remove resizer widget stuff
- $h.find('.tablesorter-resizer').remove();
- // remove all widgets
- ts.refreshWidgets(table, true, true);
- // disable tablesorter
- $t
- .removeData('tablesorter')
- .unbind('sortReset update updateCell addRows sorton appendCache applyWidgetId applyWidgets refreshWidgets destroy mouseup mouseleave')
- .find('.' + c.cssHeader)
- .unbind('click mousedown mousemove mouseup')
- .removeClass(c.cssHeader + ' ' + c.cssAsc + ' ' + c.cssDesc)
- .find('.tablesorter-header-inner').each(function(){
- if (c.cssIcon !== '') { $(this).find('.' + c.cssIcon).remove(); }
- $(this).replaceWith( $(this).contents() );
- });
- if (removeClasses !== false) {
- $t.removeClass(c.tableClass);
- }
- if (typeof callback === 'function') {
- callback(table);
- }
- };
-
- // *** sort functions ***
- // regex used in natural sort
- ts.regex = [
- /(^-?[0-9]+(\.?[0-9]*)[df]?e?[0-9]?$|^0x[0-9a-f]+$|[0-9]+)/gi, // chunk/tokenize numbers & letters
- /(^([\w ]+,?[\w ]+)?[\w ]+,?[\w ]+\d+:\d+(:\d+)?[\w ]?|^\d{1,4}[\/\-]\d{1,4}[\/\-]\d{1,4}|^\w+, \w+ \d+, \d{4})/, //date
- /^0x[0-9a-f]+$/i // hex
- ];
-
- // Natural sort - https://github.com/overset/javascript-natural-sort
- ts.sortText = function(table, a, b, col) {
- if (a === b) { return 0; }
- var c = table.config, e = c.string[ (c.empties[col] || c.emptyTo ) ],
- r = ts.regex, xN, xD, yN, yD, xF, yF, i, mx;
- if (a === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? -1 : 1) : -e || -1; }
- if (b === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? 1 : -1) : e || 1; }
- if (typeof c.textSorter === 'function') { return c.textSorter(a, b, table, col); }
- // chunk/tokenize
- xN = a.replace(r[0], '\\0$1\\0').replace(/\\0$/, '').replace(/^\\0/, '').split('\\0');
- yN = b.replace(r[0], '\\0$1\\0').replace(/\\0$/, '').replace(/^\\0/, '').split('\\0');
- // numeric, hex or date detection
- xD = parseInt(a.match(r[2]),16) || (xN.length !== 1 && a.match(r[1]) && Date.parse(a));
- yD = parseInt(b.match(r[2]),16) || (xD && b.match(r[1]) && Date.parse(b)) || null;
- // first try and sort Hex codes or Dates
- if (yD) {
- if ( xD < yD ) { return -1; }
- if ( xD > yD ) { return 1; }
- }
- mx = Math.max(xN.length, yN.length);
- // natural sorting through split numeric strings and default strings
- for (i = 0; i < mx; i++) {
- // find floats not starting with '0', string or 0 if not defined
- xF = isNaN(xN[i]) ? xN[i] || 0 : parseFloat(xN[i]) || 0;
- yF = isNaN(yN[i]) ? yN[i] || 0 : parseFloat(yN[i]) || 0;
- // handle numeric vs string comparison - number < string - (Kyle Adams)
- if (isNaN(xF) !== isNaN(yF)) { return (isNaN(xF)) ? 1 : -1; }
- // rely on string comparison if different types - i.e. '02' < 2 != '02' < '2'
- if (typeof xF !== typeof yF) {
- xF += '';
- yF += '';
- }
- if (xF < yF) { return -1; }
- if (xF > yF) { return 1; }
- }
- return 0;
- };
-
- ts.sortTextDesc = function(table, a, b, col) {
- if (a === b) { return 0; }
- var c = table.config, e = c.string[ (c.empties[col] || c.emptyTo ) ];
- if (a === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? -1 : 1) : e || 1; }
- if (b === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? 1 : -1) : -e || -1; }
- if (typeof c.textSorter === 'function') { return c.textSorter(b, a, table, col); }
- return ts.sortText(table, b, a);
- };
-
- // return text string value by adding up ascii value
- // so the text is somewhat sorted when using a digital sort
- // this is NOT an alphanumeric sort
- ts.getTextValue = function(a, mx, d) {
- if (mx) {
- // make sure the text value is greater than the max numerical value (mx)
- var i, l = a.length, n = mx + d;
- for (i = 0; i < l; i++) {
- n += a.charCodeAt(i);
- }
- return d * n;
- }
- return 0;
- };
-
- ts.sortNumeric = function(table, a, b, col, mx, d) {
- if (a === b) { return 0; }
- var c = table.config, e = c.string[ (c.empties[col] || c.emptyTo ) ];
- if (a === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? -1 : 1) : -e || -1; }
- if (b === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? 1 : -1) : e || 1; }
- if (isNaN(a)) { a = ts.getTextValue(a, mx, d); }
- if (isNaN(b)) { b = ts.getTextValue(b, mx, d); }
- return a - b;
- };
-
- ts.sortNumericDesc = function(table, a, b, col, mx, d) {
- if (a === b) { return 0; }
- var c = table.config, e = c.string[ (c.empties[col] || c.emptyTo ) ];
- if (a === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? -1 : 1) : e || 1; }
- if (b === '' && e !== 0) { return (typeof(e) === 'boolean') ? (e ? 1 : -1) : -e || -1; }
- if (isNaN(a)) { a = ts.getTextValue(a, mx, d); }
- if (isNaN(b)) { b = ts.getTextValue(b, mx, d); }
- return b - a;
- };
-
- // used when replacing accented characters during sorting
- ts.characterEquivalents = {
- "a" : "\u00e1\u00e0\u00e2\u00e3\u00e4\u0105\u00e5", // áàâãäąå
- "A" : "\u00c1\u00c0\u00c2\u00c3\u00c4\u0104\u00c5", // ÁÀÂÃÄĄÅ
- "c" : "\u00e7\u0107\u010d", // çćč
- "C" : "\u00c7\u0106\u010c", // ÇĆČ
- "e" : "\u00e9\u00e8\u00ea\u00eb\u011b\u0119", // éèêëěę
- "E" : "\u00c9\u00c8\u00ca\u00cb\u011a\u0118", // ÉÈÊËĚĘ
- "i" : "\u00ed\u00ec\u0130\u00ee\u00ef\u0131", // íìİîïı
- "I" : "\u00cd\u00cc\u0130\u00ce\u00cf", // ÍÌİÎÏ
- "o" : "\u00f3\u00f2\u00f4\u00f5\u00f6", // óòôõö
- "O" : "\u00d3\u00d2\u00d4\u00d5\u00d6", // ÓÒÔÕÖ
- "ss": "\u00df", // ß (s sharp)
- "SS": "\u1e9e", // ẞ (Capital sharp s)
- "u" : "\u00fa\u00f9\u00fb\u00fc\u016f", // úùûüů
- "U" : "\u00da\u00d9\u00db\u00dc\u016e" // ÚÙÛÜŮ
- };
- ts.replaceAccents = function(s) {
- var a, acc = '[', eq = ts.characterEquivalents;
- if (!ts.characterRegex) {
- ts.characterRegexArray = {};
- for (a in eq) {
- if (typeof a === 'string') {
- acc += eq[a];
- ts.characterRegexArray[a] = new RegExp('[' + eq[a] + ']', 'g');
- }
- }
- ts.characterRegex = new RegExp(acc + ']');
- }
- if (ts.characterRegex.test(s)) {
- for (a in eq) {
- if (typeof a === 'string') {
- s = s.replace( ts.characterRegexArray[a], a );
- }
- }
- }
- return s;
- };
-
- // *** utilities ***
- ts.isValueInArray = function(v, a) {
- var i, l = a.length;
- for (i = 0; i < l; i++) {
- if (a[i][0] === v) {
- return true;
- }
- }
- return false;
- };
-
- ts.addParser = function(parser) {
- var i, l = ts.parsers.length, a = true;
- for (i = 0; i < l; i++) {
- if (ts.parsers[i].id.toLowerCase() === parser.id.toLowerCase()) {
- a = false;
- }
- }
- if (a) {
- ts.parsers.push(parser);
- }
- };
-
- ts.getParserById = function(name) {
- var i, l = ts.parsers.length;
- for (i = 0; i < l; i++) {
- if (ts.parsers[i].id.toLowerCase() === (name.toString()).toLowerCase()) {
- return ts.parsers[i];
- }
- }
- return false;
- };
-
- ts.addWidget = function(widget) {
- ts.widgets.push(widget);
- };
-
- ts.getWidgetById = function(name) {
- var i, w, l = ts.widgets.length;
- for (i = 0; i < l; i++) {
- w = ts.widgets[i];
- if (w && w.hasOwnProperty('id') && w.id.toLowerCase() === name.toLowerCase()) {
- return w;
- }
- }
- };
-
- ts.applyWidget = function(table, init) {
- var c = table.config,
- wo = c.widgetOptions,
- ws = c.widgets.sort().reverse(), // ensure that widgets are always applied in a certain order
- time, i, w, l = ws.length;
- // make zebra last
- i = $.inArray('zebra', c.widgets);
- if (i >= 0) {
- c.widgets.splice(i,1);
- c.widgets.push('zebra');
- }
- if (c.debug) {
- time = new Date();
- }
- // add selected widgets
- for (i = 0; i < l; i++) {
- w = ts.getWidgetById(ws[i]);
- if ( w ) {
- if (init === true && w.hasOwnProperty('init')) {
- w.init(table, w, c, wo);
- } else if (!init && w.hasOwnProperty('format')) {
- w.format(table, c, wo);
- }
- }
- }
- if (c.debug) {
- benchmark("Completed " + (init === true ? "initializing" : "applying") + " widgets", time);
- }
- };
-
- ts.refreshWidgets = function(table, doAll, dontapply) {
- var i, c = table.config,
- cw = c.widgets,
- w = ts.widgets, l = w.length;
- // remove previous widgets
- for (i = 0; i < l; i++){
- if ( w[i] && w[i].id && (doAll || $.inArray( w[i].id, cw ) < 0) ) {
- if (c.debug) { log( 'Refeshing widgets: Removing ' + w[i].id ); }
- if (w[i].hasOwnProperty('remove')) { w[i].remove(table, c, c.widgetOptions); }
- }
- }
- if (dontapply !== true) {
- ts.applyWidget(table, doAll);
- }
- };
-
- // get sorter, string, empty, etc options for each column from
- // jQuery data, metadata, header option or header class name ("sorter-false")
- // priority = jQuery data > meta > headers option > header class name
- ts.getData = function(h, ch, key) {
- var val = '', $h = $(h), m, cl;
- if (!$h.length) { return ''; }
- m = $.metadata ? $h.metadata() : false;
- cl = ' ' + ($h.attr('class') || '');
- if (typeof $h.data(key) !== 'undefined' || typeof $h.data(key.toLowerCase()) !== 'undefined'){
- // "data-lockedOrder" is assigned to "lockedorder"; but "data-locked-order" is assigned to "lockedOrder"
- // "data-sort-initial-order" is assigned to "sortInitialOrder"
- val += $h.data(key) || $h.data(key.toLowerCase());
- } else if (m && typeof m[key] !== 'undefined') {
- val += m[key];
- } else if (ch && typeof ch[key] !== 'undefined') {
- val += ch[key];
- } else if (cl !== ' ' && cl.match(' ' + key + '-')) {
- // include sorter class name "sorter-text", etc
- val = cl.match( new RegExp(' ' + key + '-(\\w+)') )[1] || '';
- }
- return $.trim(val);
- };
-
- ts.formatFloat = function(s, table) {
- if (typeof(s) !== 'string' || s === '') { return s; }
- // allow using formatFloat without a table; defaults to US number format
- var i,
- t = table && table.config ? table.config.usNumberFormat !== false :
- typeof table !== "undefined" ? table : true;
- if (t) {
- // US Format - 1,234,567.89 -> 1234567.89
- s = s.replace(/,/g,'');
- } else {
- // German Format = 1.234.567,89 -> 1234567.89
- // French Format = 1 234 567,89 -> 1234567.89
- s = s.replace(/[\s|\.]/g,'').replace(/,/g,'.');
- }
- if(/^\s*\([.\d]+\)/.test(s)) {
- // make (#) into a negative number -> (10) = -10
- s = s.replace(/^\s*\(/,'-').replace(/\)/,'');
- }
- i = parseFloat(s);
- // return the text instead of zero
- return isNaN(i) ? $.trim(s) : i;
- };
-
- ts.isDigit = function(s) {
- // replace all unwanted chars and match
- return isNaN(s) ? (/^[\-+(]?\d+[)]?$/).test(s.toString().replace(/[,.'"\s]/g, '')) : true;
- };
-
- }()
- });
-
- // make shortcut
- var ts = $.tablesorter;
-
- // extend plugin scope
- $.fn.extend({
- tablesorter: ts.construct
- });
-
- // add default parsers
- ts.addParser({
- id: "text",
- is: function(s, table, node) {
- return true;
- },
- format: function(s, table, cell, cellIndex) {
- var c = table.config;
- s = $.trim( c.ignoreCase ? s.toLocaleLowerCase() : s );
- return c.sortLocaleCompare ? ts.replaceAccents(s) : s;
- },
- type: "text"
- });
-
- ts.addParser({
- id: "currency",
- is: function(s) {
- return (/^\(?\d+[\u00a3$\u20ac\u00a4\u00a5\u00a2?.]|[\u00a3$\u20ac\u00a4\u00a5\u00a2?.]\d+\)?$/).test(s); // £$€¤¥¢
- },
- format: function(s, table) {
- return ts.formatFloat(s.replace(/[^\w,. \-()]/g, ""), table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "ipAddress",
- is: function(s) {
- return (/^\d{1,3}[\.]\d{1,3}[\.]\d{1,3}[\.]\d{1,3}$/).test(s);
- },
- format: function(s, table) {
- var i, a = s.split("."),
- r = "",
- l = a.length;
- for (i = 0; i < l; i++) {
- r += ("00" + a[i]).slice(-3);
- }
- return ts.formatFloat(r, table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "url",
- is: function(s) {
- return (/^(https?|ftp|file):\/\//).test(s);
- },
- format: function(s) {
- return $.trim(s.replace(/(https?|ftp|file):\/\//, ''));
- },
- type: "text"
- });
-
- ts.addParser({
- id: "isoDate",
- is: function(s) {
- return (/^\d{4}[\/\-]\d{1,2}[\/\-]\d{1,2}$/).test(s);
- },
- format: function(s, table) {
- return ts.formatFloat((s !== "") ? (new Date(s.replace(/-/g, "/")).getTime() || "") : "", table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "percent",
- is: function(s) {
- return (/(\d\s?%|%\s?\d)/).test(s);
- },
- format: function(s, table) {
- return ts.formatFloat(s.replace(/%/g, ""), table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "usLongDate",
- is: function(s) {
- // two digit years are not allowed cross-browser
- return (/^[A-Z]{3,10}\.?\s+\d{1,2},?\s+(\d{4})(\s+\d{1,2}:\d{2}(:\d{2})?(\s+[AP]M)?)?$/i).test(s);
- },
- format: function(s, table) {
- return ts.formatFloat( (new Date(s.replace(/(\S)([AP]M)$/i, "$1 $2")).getTime() || ''), table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "shortDate", // "mmddyyyy", "ddmmyyyy" or "yyyymmdd"
- is: function(s) {
- // testing for ####-##-####, so it's not perfect
- return (/^(\d{1,2}|\d{4})[\/\-\,\.\s+]\d{1,2}[\/\-\.\,\s+](\d{1,2}|\d{4})$/).test(s);
- },
- format: function(s, table, cell, cellIndex) {
- var c = table.config, ci = c.headerList[cellIndex],
- format = ci.shortDateFormat;
- if (typeof format === 'undefined') {
- // cache header formatting so it doesn't getData for every cell in the column
- format = ci.shortDateFormat = ts.getData( ci, c.headers[cellIndex], 'dateFormat') || c.dateFormat;
- }
- s = s.replace(/\s+/g," ").replace(/[\-|\.|\,]/g, "/");
- if (format === "mmddyyyy") {
- s = s.replace(/(\d{1,2})[\/\s](\d{1,2})[\/\s](\d{4})/, "$3/$1/$2");
- } else if (format === "ddmmyyyy") {
- s = s.replace(/(\d{1,2})[\/\s](\d{1,2})[\/\s](\d{4})/, "$3/$2/$1");
- } else if (format === "yyyymmdd") {
- s = s.replace(/(\d{4})[\/\s](\d{1,2})[\/\s](\d{1,2})/, "$1/$2/$3");
- }
- return ts.formatFloat( (new Date(s).getTime() || ''), table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "time",
- is: function(s) {
- return (/^(([0-2]?\d:[0-5]\d)|([0-1]?\d:[0-5]\d\s?([AP]M)))$/i).test(s);
- },
- format: function(s, table) {
- return ts.formatFloat( (new Date("2000/01/01 " + s.replace(/(\S)([AP]M)$/i, "$1 $2")).getTime() || ""), table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "digit",
- is: function(s) {
- return ts.isDigit(s);
- },
- format: function(s, table) {
- return ts.formatFloat(s.replace(/[^\w,. \-()]/g, ""), table);
- },
- type: "numeric"
- });
-
- ts.addParser({
- id: "metadata",
- is: function(s) {
- return false;
- },
- format: function(s, table, cell) {
- var c = table.config,
- p = (!c.parserMetadataName) ? 'sortValue' : c.parserMetadataName;
- return $(cell).metadata()[p];
- },
- type: "numeric"
- });
-
- // add default widgets
- ts.addWidget({
- id: "zebra",
- format: function(table, c, wo) {
- var $tb, $tv, $tr, row, even, time, k, l,
- child = new RegExp(c.cssChildRow, 'i'),
- b = $(table).children('tbody:not(.' + c.cssInfoBlock + ')');
- if (c.debug) {
- time = new Date();
- }
- for (k = 0; k < b.length; k++ ) {
- // loop through the visible rows
- $tb = $(b[k]);
- l = $tb.children('tr').length;
- if (l > 1) {
- row = 0;
- $tv = $tb.children('tr:visible');
- // revered back to using jQuery each - strangely it's the fastest method
- /*jshint loopfunc:true */
- $tv.each(function(){
- $tr = $(this);
- // style children rows the same way the parent row was styled
- if (!child.test(this.className)) { row++; }
- even = (row % 2 === 0);
- $tr.removeClass(wo.zebra[even ? 1 : 0]).addClass(wo.zebra[even ? 0 : 1]);
- });
- }
- }
- if (c.debug) {
- ts.benchmark("Applying Zebra widget", time);
- }
- },
- remove: function(table, c, wo){
- var k, $tb,
- b = $(table).children('tbody:not(.' + c.cssInfoBlock + ')'),
- rmv = (c.widgetOptions.zebra || [ "even", "odd" ]).join(' ');
- for (k = 0; k < b.length; k++ ){
- $tb = $.tablesorter.processTbody(table, $(b[k]), true); // remove tbody
- $tb.children().removeClass(rmv);
- $.tablesorter.processTbody(table, $tb, false); // restore tbody
- }
- }
- });
-
-})(jQuery); \ No newline at end of file
+!function(l){l.extend({tablesorter:new function(){function e(b){"undefined"!==typeof console&&"undefined"!==typeof console.log?console.log(b):alert(b)}function q(b,c){e(b+" ("+((new Date).getTime()-c.getTime())+"ms)")}function k(b,c,a){if(!c)return"";var d=b.config,g=d.textExtraction,m="",m="simple"===g?d.supportsTextContent?c.textContent:l(c).text():"function"===typeof g?g(c,b,a):"object"===typeof g&&g.hasOwnProperty(a)?g[a](c,b,a):d.supportsTextContent?c.textContent:l(c).text();return l.trim(m)}
+function r(b){var c=b.config,a=l(b.tBodies).filter(":not(."+c.cssInfoBlock+")"),d,g,m,h,w,n,p="";if(0===a.length)return c.debug?e("*Empty table!* Not building a parser cache"):"";a=a[0].rows;if(a[0])for(d=[],g=a[0].cells.length,m=0;m<g;m++){h=c.$headers.filter(":not([colspan])");h=h.add(c.$headers.filter('[colspan="1"]')).filter('[data-column="'+m+'"]:last');w=c.headers[m];n=f.getParserById(f.getData(h,w,"sorter"));c.empties[m]=f.getData(h,w,"empty")||c.emptyTo||(c.emptyToBottom?"bottom":"top");c.strings[m]=
+f.getData(h,w,"string")||c.stringTo||"max";if(!n)a:{h=b;w=a;n=-1;for(var q=m,t=void 0,u=f.parsers.length,v=!1,r="",t=!0;""===r&&t;)n++,w[n]?(v=w[n].cells[q],r=k(h,v,q),h.config.debug&&e("Checking if value was empty on row "+n+", column: "+q+": "+r)):t=!1;for(t=1;t<u;t++)if(f.parsers[t].is(r,h,v)){n=f.parsers[t];break a}n=f.parsers[0]}c.debug&&(p+="column:"+m+"; parser:"+n.id+"; string:"+c.strings[m]+"; empty: "+c.empties[m]+"\n");d.push(n)}c.debug&&e(p);return d}function x(b){var c=b.tBodies,a=b.config,
+d,g,m=a.parsers,h,w,n,p,z,t,u,v=[];a.cache={};if(!m)return a.debug?e("*Empty table!* Not building a cache"):"";a.debug&&(u=new Date);a.showProcessing&&f.isProcessing(b,!0);for(p=0;p<c.length;p++)if(a.cache[p]={row:[],normalized:[]},!l(c[p]).hasClass(a.cssInfoBlock)){d=c[p]&&c[p].rows.length||0;g=c[p].rows[0]&&c[p].rows[0].cells.length||0;for(w=0;w<d;++w)if(z=l(c[p].rows[w]),t=[],z.hasClass(a.cssChildRow))a.cache[p].row[a.cache[p].row.length-1]=a.cache[p].row[a.cache[p].row.length-1].add(z);else{a.cache[p].row.push(z);
+for(n=0;n<g;++n)h=k(b,z[0].cells[n],n),h=m[n].format(h,b,z[0].cells[n],n),t.push(h),"numeric"===(m[n].type||"").toLowerCase()&&(v[n]=Math.max(Math.abs(h),v[n]||0));t.push(a.cache[p].normalized.length);a.cache[p].normalized.push(t)}a.cache[p].colMax=v}a.showProcessing&&f.isProcessing(b);a.debug&&q("Building cache for "+d+" rows",u)}function y(b,c){var a=b.config,d=b.tBodies,g=[],m=a.cache,h,e,n,p,z,t,u,v,k,r,s;if(m[0]){a.debug&&(s=new Date);for(v=0;v<d.length;v++)if(h=l(d[v]),!h.hasClass(a.cssInfoBlock)){z=
+f.processTbody(b,h,!0);h=m[v].row;e=m[v].normalized;p=(n=e.length)?e[0].length-1:0;for(t=0;t<n;t++)if(r=e[t][p],g.push(h[r]),!a.appender||!a.removeRows)for(k=h[r].length,u=0;u<k;u++)z.append(h[r][u]);f.processTbody(b,z,!1)}a.appender&&a.appender(b,g);a.debug&&q("Rebuilt table",s);c||f.applyWidget(b);l(b).trigger("sortEnd",b)}}function B(b){var c=[],a={};b=l(b).find("thead:eq(0), tfoot").children("tr");var d,g,m,h,e,f,p,q,t,u;for(d=0;d<b.length;d++)for(e=b[d].cells,g=0;g<e.length;g++){h=e[g];f=h.parentNode.rowIndex;
+p=f+"-"+h.cellIndex;q=h.rowSpan||1;t=h.colSpan||1;"undefined"===typeof c[f]&&(c[f]=[]);for(m=0;m<c[f].length+1;m++)if("undefined"===typeof c[f][m]){u=m;break}a[p]=u;l(h).attr({"data-column":u});for(m=f;m<f+q;m++)for("undefined"===typeof c[m]&&(c[m]=[]),p=c[m],h=u;h<u+t;h++)p[h]="x"}return a}function C(b){var c=B(b),a,d,g,m,h,w,n,p,k=b.config;k.headerList=[];k.headerContent=[];k.debug&&(n=new Date);m=k.cssIcon?'<i class="'+k.cssIcon+'"></i>':"";p=l(b).find(k.selectorHeaders).each(function(b){d=l(this);
+a=k.headers[b];k.headerContent[b]=this.innerHTML;h=k.headerTemplate.replace(/\{content\}/g,this.innerHTML).replace(/\{icon\}/g,m);k.onRenderTemplate&&(g=k.onRenderTemplate.apply(d,[b,h]))&&"string"===typeof g&&(h=g);this.innerHTML='<div class="tablesorter-header-inner">'+h+"</div>";k.onRenderHeader&&k.onRenderHeader.apply(d,[b]);this.column=c[this.parentNode.rowIndex+"-"+this.cellIndex];var e=f.getData(d,a,"sortInitialOrder")||k.sortInitialOrder;this.order=/^d/i.test(e)||1===e?[1,0,2]:[0,1,2];this.count=
+-1;"false"===f.getData(d,a,"sorter")?(this.sortDisabled=!0,d.addClass("sorter-false")):d.removeClass("sorter-false");this.lockedOrder=!1;w=f.getData(d,a,"lockedOrder")||!1;"undefined"!==typeof w&&!1!==w&&(this.order=this.lockedOrder=/^d/i.test(w)||1===w?[1,1,1]:[0,0,0]);d.addClass((this.sortDisabled?"sorter-false ":" ")+k.cssHeader);k.headerList[b]=this;d.parent().addClass(k.cssHeaderRow)});b.config.debug&&(q("Built headers:",n),e(p));return p}function A(b){var c,a,d,g=b.config,m=g.sortList,f=[g.cssAsc,
+g.cssDesc],e=l(b).find("tfoot tr").children().removeClass(f.join(" "));g.$headers.removeClass(f.join(" "));d=m.length;for(c=0;c<d;c++)if(2!==m[c][1]&&(b=g.$headers.not(".sorter-false").filter('[data-column="'+m[c][0]+'"]'+(1===d?":last":"")),b.length))for(a=0;a<b.length;a++)b[a].sortDisabled||(b.eq(a).addClass(f[m[c][1]]),e.length&&e.filter('[data-column="'+m[c][0]+'"]').eq(a).addClass(f[m[c][1]]))}function E(b){if(b.config.widthFixed&&0===l(b).find("colgroup").length){var c=l("<colgroup>"),a=l(b).width();
+l("tr:first td",b.tBodies[0]).each(function(){c.append(l("<col>").css("width",parseInt(1E3*(l(this).width()/a),10)/10+"%"))});l(b).prepend(c)}}function D(b,c){var a,d,g,f=b.config,e=c||f.sortList;f.sortList=[];l.each(e,function(c,b){a=[parseInt(b[0],10),parseInt(b[1],10)];if(g=f.headerList[a[0]])f.sortList.push(a),d=l.inArray(a[1],g.order),g.count=0<=d?d:a[1]%(f.sortReset?3:2)})}function F(b){var c=0,a=b.config,d=a.sortList,g=d.length,f=b.tBodies.length,e,k,n,p,r,t,u,v,s;if(!a.serverSideSorting&&
+a.cache[0]){a.debug&&(e=new Date);for(n=0;n<f;n++)r=a.cache[n].colMax,s=(t=a.cache[n].normalized)&&t[0]?t[0].length-1:0,t.sort(function(f,e){for(k=0;k<g;k++){p=d[k][0];v=d[k][1];u=/n/i.test(a.parsers&&a.parsers[p]?a.parsers[p].type||"":"")?"Numeric":"Text";u+=0===v?"":"Desc";/Numeric/.test(u)&&a.strings[p]&&(c="boolean"===typeof a.string[a.strings[p]]?(0===v?1:-1)*(a.string[a.strings[p]]?-1:1):a.strings[p]?a.string[a.strings[p]]||0:0);var m=l.tablesorter["sort"+u](b,f[p],e[p],p,r[p],c);if(m)return m}return f[s]-
+e[s]});a.debug&&q("Sorting on "+d.toString()+" and dir "+v+" time",e)}}function H(b,c){b.trigger("updateComplete");"function"===typeof c&&c(b[0])}function G(b,c,a){!1!==c?b.trigger("sorton",[b[0].config.sortList,function(){H(b,a)}]):H(b,a)}var f=this;f.version="2.7";f.parsers=[];f.widgets=[];f.defaults={theme:"default",widthFixed:!1,showProcessing:!1,headerTemplate:"{content}",onRenderTemplate:null,onRenderHeader:null,cancelSelection:!0,dateFormat:"mmddyyyy",sortMultiSortKey:"shiftKey",sortResetKey:"ctrlKey",
+usNumberFormat:!0,delayInit:!1,serverSideSorting:!1,headers:{},ignoreCase:!0,sortForce:null,sortList:[],sortAppend:null,sortInitialOrder:"asc",sortLocaleCompare:!1,sortReset:!1,sortRestart:!1,emptyTo:"bottom",stringTo:"max",textExtraction:"simple",textSorter:null,widgets:[],widgetOptions:{zebra:["even","odd"]},initWidgets:!0,initialized:null,tableClass:"tablesorter",cssAsc:"tablesorter-headerAsc",cssChildRow:"tablesorter-childRow",cssDesc:"tablesorter-headerDesc",cssHeader:"tablesorter-header",cssHeaderRow:"tablesorter-headerRow",
+cssIcon:"tablesorter-icon",cssInfoBlock:"tablesorter-infoOnly",cssProcessing:"tablesorter-processing",selectorHeaders:"> thead th, > thead td",selectorSort:"th, td",selectorRemove:".remove-me",debug:!1,headerList:[],empties:{},strings:{},parsers:[]};f.benchmark=q;f.construct=function(b){return this.each(function(){if(!this.tHead||0===this.tBodies.length||!0===this.hasInitialized)return this.config.debug?e("stopping initialization! No thead, tbody or tablesorter has already been initialized"):"";var c=
+l(this),a,d,g,m="",h,q,n,p,z=l.metadata;this.hasInitialized=!1;this.config={};a=l.extend(!0,this.config,f.defaults,b);l.data(this,"tablesorter",a);a.debug&&l.data(this,"startoveralltimer",new Date);a.supportsTextContent="x"===l("<span>x</span>")[0].textContent;a.supportsDataObject=1.4<=parseFloat(l.fn.jquery);a.string={max:1,min:-1,"max+":1,"max-":-1,zero:0,none:0,"null":0,top:!0,bottom:!1};/tablesorter\-/.test(c.attr("class"))||(m=""!==a.theme?" tablesorter-"+a.theme:"");c.addClass(a.tableClass+
+m);a.$headers=C(this);a.parsers=r(this);a.delayInit||x(this);a.$headers.find("*").andSelf().filter(a.selectorSort).unbind("mousedown.tablesorter mouseup.tablesorter").bind("mousedown.tablesorter mouseup.tablesorter",function(b,e){var k=(this.tagName.match("TH|TD")?l(this):l(this).parents("th, td").filter(":last"))[0];if(1!==(b.which||b.button))return!1;if("mousedown"===b.type)return p=(new Date).getTime(),"INPUT"===b.target.tagName?"":!a.cancelSelection;if(!0!==e&&250<(new Date).getTime()-p)return!1;
+a.delayInit&&!a.cache&&x(c[0]);if(!k.sortDisabled){c.trigger("sortStart",c[0]);m=!b[a.sortMultiSortKey];k.count=b[a.sortResetKey]?2:(k.count+1)%(a.sortReset?3:2);a.sortRestart&&(d=k,a.$headers.each(function(){this===d||!m&&l(this).is("."+a.cssDesc+",."+a.cssAsc)||(this.count=-1)}));d=k.column;if(m){a.sortList=[];if(null!==a.sortForce)for(h=a.sortForce,g=0;g<h.length;g++)h[g][0]!==d&&a.sortList.push(h[g]);n=k.order[k.count];if(2>n&&(a.sortList.push([d,n]),1<k.colSpan))for(g=1;g<k.colSpan;g++)a.sortList.push([d+
+g,n])}else if(a.sortAppend&&1<a.sortList.length&&f.isValueInArray(a.sortAppend[0][0],a.sortList)&&a.sortList.pop(),f.isValueInArray(d,a.sortList))for(g=0;g<a.sortList.length;g++)q=a.sortList[g],n=a.headerList[q[0]],q[0]===d&&(q[1]=n.order[n.count],2===q[1]&&(a.sortList.splice(g,1),n.count=-1));else if(n=k.order[k.count],2>n&&(a.sortList.push([d,n]),1<k.colSpan))for(g=1;g<k.colSpan;g++)a.sortList.push([d+g,n]);if(null!==a.sortAppend)for(h=a.sortAppend,g=0;g<h.length;g++)h[g][0]!==d&&a.sortList.push(h[g]);
+c.trigger("sortBegin",c[0]);setTimeout(function(){A(c[0]);F(c[0]);y(c[0])},1)}});a.cancelSelection&&a.$headers.each(function(){this.onselectstart=function(){return!1}});c.unbind("sortReset update updateCell addRows sorton appendCache applyWidgetId applyWidgets refreshWidgets destroy mouseup mouseleave").bind("sortReset",function(){a.sortList=[];A(this);F(this);y(this)}).bind("update",function(b,d,g){l(a.selectorRemove,this).remove();a.parsers=r(this);x(this);G(c,d,g)}).bind("updateCell",function(b,
+d,g,f){var e,m,h;e=l(this).find("tbody");b=e.index(l(d).parents("tbody").filter(":last"));var q=l(d).parents("tr").filter(":last");d=l(d)[0];e.length&&0<=b&&(m=e.eq(b).find("tr").index(q),h=d.cellIndex,e=this.config.cache[b].normalized[m].length-1,this.config.cache[b].row[this.config.cache[b].normalized[m][e]]=q,this.config.cache[b].normalized[m][h]=a.parsers[h].format(k(this,d,h),this,d,h),G(c,g,f))}).bind("addRows",function(b,d,e,f){var m=d.filter("tr").length,h=[],q=d[0].cells.length,n=l(this).find("tbody").index(d.closest("tbody"));
+a.parsers||(a.parsers=r(this));for(b=0;b<m;b++){for(g=0;g<q;g++)h[g]=a.parsers[g].format(k(this,d[b].cells[g],g),this,d[b].cells[g],g);h.push(a.cache[n].row.length);a.cache[n].row.push([d[b]]);a.cache[n].normalized.push(h);h=[]}G(c,e,f)}).bind("sorton",function(a,b,c,d){l(this).trigger("sortStart",this);D(this,b);A(this);F(this);y(this,d);"function"===typeof c&&c(this)}).bind("appendCache",function(a,b,c){y(this,c);"function"===typeof b&&b(this)}).bind("applyWidgetId",function(b,c){f.getWidgetById(c).format(this,
+a,a.widgetOptions)}).bind("applyWidgets",function(a,b){f.applyWidget(this,b)}).bind("refreshWidgets",function(a,b,c){f.refreshWidgets(this,b,c)}).bind("destroy",function(a,b,c){f.destroy(this,b,c)});a.supportsDataObject&&"undefined"!==typeof c.data().sortlist?a.sortList=c.data().sortlist:z&&(c.metadata()&&c.metadata().sortlist)&&(a.sortList=c.metadata().sortlist);f.applyWidget(this,!0);0<a.sortList.length?c.trigger("sorton",[a.sortList,{},!a.initWidgets]):a.initWidgets&&f.applyWidget(this);E(this);
+a.showProcessing&&c.unbind("sortBegin sortEnd").bind("sortBegin sortEnd",function(a){f.isProcessing(c[0],"sortBegin"===a.type)});this.hasInitialized=!0;a.debug&&f.benchmark("Overall initialization time",l.data(this,"startoveralltimer"));c.trigger("tablesorter-initialized",this);"function"===typeof a.initialized&&a.initialized(this)})};f.isProcessing=function(b,c,a){var d=b.config;b=a||l(b).find("."+d.cssHeader);c?(0<d.sortList.length&&(b=b.filter(function(){return this.sortDisabled?!1:f.isValueInArray(parseFloat(l(this).attr("data-column")),
+d.sortList)})),b.addClass(d.cssProcessing)):b.removeClass(d.cssProcessing)};f.processTbody=function(b,c,a){if(a)return c.before('<span class="tablesorter-savemyplace"/>'),b=l.fn.detach?c.detach():c.remove();b=l(b).find("span.tablesorter-savemyplace");c.insertAfter(b);b.remove()};f.clearTableBody=function(b){l(b.tBodies).filter(":not(."+b.config.cssInfoBlock+")").empty()};f.destroy=function(b,c,a){var d=l(b),g=b.config,e=d.find("thead:first");b.hasInitialized=!1;e.find("tr:not(."+g.cssHeaderRow+")").remove();
+e.find(".tablesorter-resizer").remove();f.refreshWidgets(b,!0,!0);d.removeData("tablesorter").unbind("sortReset update updateCell addRows sorton appendCache applyWidgetId applyWidgets refreshWidgets destroy mouseup mouseleave").find("."+g.cssHeader).unbind("click mousedown mousemove mouseup").removeClass(g.cssHeader+" "+g.cssAsc+" "+g.cssDesc).find(".tablesorter-header-inner").each(function(){""!==g.cssIcon&&l(this).find("."+g.cssIcon).remove();l(this).replaceWith(l(this).contents())});!1!==c&&d.removeClass(g.tableClass);
+"function"===typeof a&&a(b)};f.regex=[/(^-?[0-9]+(\.?[0-9]*)[df]?e?[0-9]?$|^0x[0-9a-f]+$|[0-9]+)/gi,/(^([\w ]+,?[\w ]+)?[\w ]+,?[\w ]+\d+:\d+(:\d+)?[\w ]?|^\d{1,4}[\/\-]\d{1,4}[\/\-]\d{1,4}|^\w+, \w+ \d+, \d{4})/,/^0x[0-9a-f]+$/i];f.sortText=function(b,c,a,d){if(c===a)return 0;var g=b.config,e=g.string[g.empties[d]||g.emptyTo],h=f.regex;if(""===c&&0!==e)return"boolean"===typeof e?e?-1:1:-e||-1;if(""===a&&0!==e)return"boolean"===typeof e?e?1:-1:e||1;if("function"===typeof g.textSorter)return g.textSorter(c,
+a,b,d);b=c.replace(h[0],"\\0$1\\0").replace(/\\0$/,"").replace(/^\\0/,"").split("\\0");d=a.replace(h[0],"\\0$1\\0").replace(/\\0$/,"").replace(/^\\0/,"").split("\\0");c=parseInt(c.match(h[2]),16)||1!==b.length&&c.match(h[1])&&Date.parse(c);if(a=parseInt(a.match(h[2]),16)||c&&a.match(h[1])&&Date.parse(a)||null){if(c<a)return-1;if(c>a)return 1}g=Math.max(b.length,d.length);for(c=0;c<g;c++){a=isNaN(b[c])?b[c]||0:parseFloat(b[c])||0;h=isNaN(d[c])?d[c]||0:parseFloat(d[c])||0;if(isNaN(a)!==isNaN(h))return isNaN(a)?
+1:-1;typeof a!==typeof h&&(a+="",h+="");if(a<h)return-1;if(a>h)return 1}return 0};f.sortTextDesc=function(b,c,a,d){if(c===a)return 0;var e=b.config,m=e.string[e.empties[d]||e.emptyTo];return""===c&&0!==m?"boolean"===typeof m?m?-1:1:m||1:""===a&&0!==m?"boolean"===typeof m?m?1:-1:-m||-1:"function"===typeof e.textSorter?e.textSorter(a,c,b,d):f.sortText(b,a,c)};f.getTextValue=function(b,c,a){if(c){var d=b.length,e=c+a;for(c=0;c<d;c++)e+=b.charCodeAt(c);return a*e}return 0};f.sortNumeric=function(b,c,
+a,d,e,m){if(c===a)return 0;b=b.config;d=b.string[b.empties[d]||b.emptyTo];if(""===c&&0!==d)return"boolean"===typeof d?d?-1:1:-d||-1;if(""===a&&0!==d)return"boolean"===typeof d?d?1:-1:d||1;isNaN(c)&&(c=f.getTextValue(c,e,m));isNaN(a)&&(a=f.getTextValue(a,e,m));return c-a};f.sortNumericDesc=function(b,c,a,d,e,m){if(c===a)return 0;b=b.config;d=b.string[b.empties[d]||b.emptyTo];if(""===c&&0!==d)return"boolean"===typeof d?d?-1:1:d||1;if(""===a&&0!==d)return"boolean"===typeof d?d?1:-1:-d||-1;isNaN(c)&&
+(c=f.getTextValue(c,e,m));isNaN(a)&&(a=f.getTextValue(a,e,m));return a-c};f.characterEquivalents={a:"\u00e1\u00e0\u00e2\u00e3\u00e4\u0105\u00e5",A:"\u00c1\u00c0\u00c2\u00c3\u00c4\u0104\u00c5",c:"\u00e7\u0107\u010d",C:"\u00c7\u0106\u010c",e:"\u00e9\u00e8\u00ea\u00eb\u011b\u0119",E:"\u00c9\u00c8\u00ca\u00cb\u011a\u0118",i:"\u00ed\u00ec\u0130\u00ee\u00ef\u0131",I:"\u00cd\u00cc\u0130\u00ce\u00cf",o:"\u00f3\u00f2\u00f4\u00f5\u00f6",O:"\u00d3\u00d2\u00d4\u00d5\u00d6",ss:"\u00df",SS:"\u1e9e",u:"\u00fa\u00f9\u00fb\u00fc\u016f",
+U:"\u00da\u00d9\u00db\u00dc\u016e"};f.replaceAccents=function(b){var c,a="[",d=f.characterEquivalents;if(!f.characterRegex){f.characterRegexArray={};for(c in d)"string"===typeof c&&(a+=d[c],f.characterRegexArray[c]=RegExp("["+d[c]+"]","g"));f.characterRegex=RegExp(a+"]")}if(f.characterRegex.test(b))for(c in d)"string"===typeof c&&(b=b.replace(f.characterRegexArray[c],c));return b};f.isValueInArray=function(b,c){var a,d=c.length;for(a=0;a<d;a++)if(c[a][0]===b)return!0;return!1};f.addParser=function(b){var c,
+a=f.parsers.length,d=!0;for(c=0;c<a;c++)f.parsers[c].id.toLowerCase()===b.id.toLowerCase()&&(d=!1);d&&f.parsers.push(b)};f.getParserById=function(b){var c,a=f.parsers.length;for(c=0;c<a;c++)if(f.parsers[c].id.toLowerCase()===b.toString().toLowerCase())return f.parsers[c];return!1};f.addWidget=function(b){f.widgets.push(b)};f.getWidgetById=function(b){var c,a,d=f.widgets.length;for(c=0;c<d;c++)if((a=f.widgets[c])&&a.hasOwnProperty("id")&&a.id.toLowerCase()===b.toLowerCase())return a};f.applyWidget=
+function(b,c){var a=b.config,d=a.widgetOptions,e=a.widgets.sort().reverse(),m,h,k,n=e.length;h=l.inArray("zebra",a.widgets);0<=h&&(a.widgets.splice(h,1),a.widgets.push("zebra"));a.debug&&(m=new Date);for(h=0;h<n;h++)(k=f.getWidgetById(e[h]))&&(!0===c&&k.hasOwnProperty("init")?k.init(b,k,a,d):!c&&k.hasOwnProperty("format")&&k.format(b,a,d));a.debug&&q("Completed "+(!0===c?"initializing":"applying")+" widgets",m)};f.refreshWidgets=function(b,c,a){var d,g=b.config,m=g.widgets,h=f.widgets,k=h.length;
+for(d=0;d<k;d++)h[d]&&(h[d].id&&(c||0>l.inArray(h[d].id,m)))&&(g.debug&&e("Refeshing widgets: Removing "+h[d].id),h[d].hasOwnProperty("remove")&&h[d].remove(b,g,g.widgetOptions));!0!==a&&f.applyWidget(b,c)};f.getData=function(b,c,a){var d="";b=l(b);var e,f;if(!b.length)return"";e=l.metadata?b.metadata():!1;f=" "+(b.attr("class")||"");"undefined"!==typeof b.data(a)||"undefined"!==typeof b.data(a.toLowerCase())?d+=b.data(a)||b.data(a.toLowerCase()):e&&"undefined"!==typeof e[a]?d+=e[a]:c&&"undefined"!==
+typeof c[a]?d+=c[a]:" "!==f&&f.match(" "+a+"-")&&(d=f.match(RegExp(" "+a+"-(\\w+)"))[1]||"");return l.trim(d)};f.formatFloat=function(b,c){if("string"!==typeof b||""===b)return b;var a;b=(c&&c.config?!1!==c.config.usNumberFormat:"undefined"!==typeof c?c:1)?b.replace(/,/g,""):b.replace(/[\s|\.]/g,"").replace(/,/g,".");/^\s*\([.\d]+\)/.test(b)&&(b=b.replace(/^\s*\(/,"-").replace(/\)/,""));a=parseFloat(b);return isNaN(a)?l.trim(b):a};f.isDigit=function(b){return isNaN(b)?/^[\-+(]?\d+[)]?$/.test(b.toString().replace(/[,.'"\s]/g,
+"")):!0}}});var k=l.tablesorter;l.fn.extend({tablesorter:k.construct});k.addParser({id:"text",is:function(e,k,l){return!0},format:function(e,q,s,r){q=q.config;e=l.trim(q.ignoreCase?e.toLocaleLowerCase():e);return q.sortLocaleCompare?k.replaceAccents(e):e},type:"text"});k.addParser({id:"currency",is:function(e){return/^\(?\d+[\u00a3$\u20ac\u00a4\u00a5\u00a2?.]|[\u00a3$\u20ac\u00a4\u00a5\u00a2?.]\d+\)?$/.test(e)},format:function(e,l){return k.formatFloat(e.replace(/[^\w,. \-()]/g,""),l)},type:"numeric"});
+k.addParser({id:"ipAddress",is:function(e){return/^\d{1,3}[\.]\d{1,3}[\.]\d{1,3}[\.]\d{1,3}$/.test(e)},format:function(e,l){var s,r=e.split("."),x="",y=r.length;for(s=0;s<y;s++)x+=("00"+r[s]).slice(-3);return k.formatFloat(x,l)},type:"numeric"});k.addParser({id:"url",is:function(e){return/^(https?|ftp|file):\/\//.test(e)},format:function(e){return l.trim(e.replace(/(https?|ftp|file):\/\//,""))},type:"text"});k.addParser({id:"isoDate",is:function(e){return/^\d{4}[\/\-]\d{1,2}[\/\-]\d{1,2}$/.test(e)},
+format:function(e,l){return k.formatFloat(""!==e?(new Date(e.replace(/-/g,"/"))).getTime()||"":"",l)},type:"numeric"});k.addParser({id:"percent",is:function(e){return/(\d\s?%|%\s?\d)/.test(e)},format:function(e,l){return k.formatFloat(e.replace(/%/g,""),l)},type:"numeric"});k.addParser({id:"usLongDate",is:function(e){return/^[A-Z]{3,10}\.?\s+\d{1,2},?\s+(\d{4})(\s+\d{1,2}:\d{2}(:\d{2})?(\s+[AP]M)?)?$/i.test(e)},format:function(e,l){return k.formatFloat((new Date(e.replace(/(\S)([AP]M)$/i,"$1 $2"))).getTime()||
+"",l)},type:"numeric"});k.addParser({id:"shortDate",is:function(e){return/^(\d{1,2}|\d{4})[\/\-\,\.\s+]\d{1,2}[\/\-\.\,\s+](\d{1,2}|\d{4})$/.test(e)},format:function(e,l,s,r){s=l.config;var x=s.headerList[r],y=x.shortDateFormat;"undefined"===typeof y&&(y=x.shortDateFormat=k.getData(x,s.headers[r],"dateFormat")||s.dateFormat);e=e.replace(/\s+/g," ").replace(/[\-|\.|\,]/g,"/");"mmddyyyy"===y?e=e.replace(/(\d{1,2})[\/\s](\d{1,2})[\/\s](\d{4})/,"$3/$1/$2"):"ddmmyyyy"===y?e=e.replace(/(\d{1,2})[\/\s](\d{1,2})[\/\s](\d{4})/,
+"$3/$2/$1"):"yyyymmdd"===y&&(e=e.replace(/(\d{4})[\/\s](\d{1,2})[\/\s](\d{1,2})/,"$1/$2/$3"));return k.formatFloat((new Date(e)).getTime()||"",l)},type:"numeric"});k.addParser({id:"time",is:function(e){return/^(([0-2]?\d:[0-5]\d)|([0-1]?\d:[0-5]\d\s?([AP]M)))$/i.test(e)},format:function(e,l){return k.formatFloat((new Date("2000/01/01 "+e.replace(/(\S)([AP]M)$/i,"$1 $2"))).getTime()||"",l)},type:"numeric"});k.addParser({id:"digit",is:function(e){return k.isDigit(e)},format:function(e,l){return k.formatFloat(e.replace(/[^\w,. \-()]/g,
+""),l)},type:"numeric"});k.addParser({id:"metadata",is:function(e){return!1},format:function(e,k,s){e=k.config;e=e.parserMetadataName?e.parserMetadataName:"sortValue";return l(s).metadata()[e]},type:"numeric"});k.addWidget({id:"zebra",format:function(e,q,s){var r,x,y,B,C,A,E=RegExp(q.cssChildRow,"i"),D=l(e).children("tbody:not(."+q.cssInfoBlock+")");q.debug&&(C=new Date);for(e=0;e<D.length;e++)r=l(D[e]),A=r.children("tr").length,1<A&&(y=0,r=r.children("tr:visible"),r.each(function(){x=l(this);E.test(this.className)||
+y++;B=0===y%2;x.removeClass(s.zebra[B?1:0]).addClass(s.zebra[B?0:1])}));q.debug&&k.benchmark("Applying Zebra widget",C)},remove:function(e,k,s){var r;s=l(e).children("tbody:not(."+k.cssInfoBlock+")");var x=(k.widgetOptions.zebra||["even","odd"]).join(" ");for(k=0;k<s.length;k++)r=l.tablesorter.processTbody(e,l(s[k]),!0),r.children().removeClass(x),l.tablesorter.processTbody(e,r,!1)}})}(jQuery);