123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- // Copyright 2008 The Closure Library Authors. All Rights Reserved.
- //
- // 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
- //
- // http://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.
- /**
- * @fileoverview A table sorting decorator.
- *
- * @author robbyw@google.com (Robby Walker)
- * @see ../demos/tablesorter.html
- */
- goog.provide('goog.ui.TableSorter');
- goog.provide('goog.ui.TableSorter.EventType');
- goog.require('goog.array');
- goog.require('goog.dom');
- goog.require('goog.dom.TagName');
- goog.require('goog.dom.classlist');
- goog.require('goog.events.EventType');
- goog.require('goog.functions');
- goog.require('goog.ui.Component');
- /**
- * A table sorter allows for sorting of a table by column. This component can
- * be used to decorate an already existing TABLE element with sorting
- * features.
- *
- * The TABLE should use a THEAD containing TH elements for the table column
- * headers.
- *
- * @param {goog.dom.DomHelper=} opt_domHelper Optional DOM helper, used for
- * document interaction.
- * @constructor
- * @extends {goog.ui.Component}
- */
- goog.ui.TableSorter = function(opt_domHelper) {
- goog.ui.Component.call(this, opt_domHelper);
- /**
- * The current sort header of the table, or null if none.
- * @type {HTMLTableCellElement}
- * @private
- */
- this.header_ = null;
- /**
- * Whether the last sort was in reverse.
- * @type {boolean}
- * @private
- */
- this.reversed_ = false;
- /**
- * The default sorting function.
- * @type {function(*, *) : number}
- * @private
- */
- this.defaultSortFunction_ = goog.ui.TableSorter.numericSort;
- /**
- * Array of custom sorting functions per colun.
- * @type {Array<function(*, *) : number>}
- * @private
- */
- this.sortFunctions_ = [];
- };
- goog.inherits(goog.ui.TableSorter, goog.ui.Component);
- goog.tagUnsealableClass(goog.ui.TableSorter);
- /**
- * Row number (in <thead>) to use for sorting.
- * @type {number}
- * @private
- */
- goog.ui.TableSorter.prototype.sortableHeaderRowIndex_ = 0;
- /**
- * Sets the row index (in <thead>) to be used for sorting.
- * By default, the first row (index 0) is used.
- * Must be called before decorate() is called.
- * @param {number} index The row index.
- */
- goog.ui.TableSorter.prototype.setSortableHeaderRowIndex = function(index) {
- if (this.isInDocument()) {
- throw Error(goog.ui.Component.Error.ALREADY_RENDERED);
- }
- this.sortableHeaderRowIndex_ = index;
- };
- /**
- * Table sorter events.
- * @enum {string}
- */
- goog.ui.TableSorter.EventType = {
- BEFORESORT: 'beforesort',
- SORT: 'sort'
- };
- /** @override */
- goog.ui.TableSorter.prototype.canDecorate = function(element) {
- return element.tagName == goog.dom.TagName.TABLE;
- };
- /** @override */
- goog.ui.TableSorter.prototype.enterDocument = function() {
- goog.ui.TableSorter.superClass_.enterDocument.call(this);
- var table = this.getElement();
- var headerRow = table.tHead.rows[this.sortableHeaderRowIndex_];
- this.getHandler().listen(headerRow, goog.events.EventType.CLICK, this.sort_);
- };
- /**
- * @return {number} The current sort column of the table, or -1 if none.
- */
- goog.ui.TableSorter.prototype.getSortColumn = function() {
- return this.header_ ? this.header_.cellIndex : -1;
- };
- /**
- * @return {boolean} Whether the last sort was in reverse.
- */
- goog.ui.TableSorter.prototype.isSortReversed = function() {
- return this.reversed_;
- };
- /**
- * @return {function(*, *) : number} The default sort function to be used by
- * all columns.
- */
- goog.ui.TableSorter.prototype.getDefaultSortFunction = function() {
- return this.defaultSortFunction_;
- };
- /**
- * Sets the default sort function to be used by all columns. If not set
- * explicitly, this defaults to numeric sorting.
- * @param {function(*, *) : number} sortFunction The new default sort function.
- */
- goog.ui.TableSorter.prototype.setDefaultSortFunction = function(sortFunction) {
- this.defaultSortFunction_ = sortFunction;
- };
- /**
- * Gets the sort function to be used by the given column. Returns the default
- * sort function if no sort function is explicitly set for this column.
- * @param {number} column The column index.
- * @return {function(*, *) : number} The sort function used by the column.
- */
- goog.ui.TableSorter.prototype.getSortFunction = function(column) {
- return this.sortFunctions_[column] || this.defaultSortFunction_;
- };
- /**
- * Set the sort function for the given column, overriding the default sort
- * function.
- * @param {number} column The column index.
- * @param {function(*, *) : number} sortFunction The new sort function.
- */
- goog.ui.TableSorter.prototype.setSortFunction = function(column, sortFunction) {
- this.sortFunctions_[column] = sortFunction;
- };
- /**
- * Sort the table contents by the values in the given column.
- * @param {goog.events.BrowserEvent} e The click event.
- * @private
- */
- goog.ui.TableSorter.prototype.sort_ = function(e) {
- // Determine what column was clicked.
- // TODO(robbyw): If this table cell contains another table, this could break.
- var target = e.target;
- var th = goog.dom.getAncestorByTagNameAndClass(target, goog.dom.TagName.TH);
- // If the user clicks on the same column, sort it in reverse of what it is
- // now. Otherwise, sort forward.
- var reverse = th == this.header_ ? !this.reversed_ : false;
- // Perform the sort.
- if (this.dispatchEvent(goog.ui.TableSorter.EventType.BEFORESORT)) {
- if (this.sort(th.cellIndex, reverse)) {
- this.dispatchEvent(goog.ui.TableSorter.EventType.SORT);
- }
- }
- };
- /**
- * Sort the table contents by the values in the given column.
- * @param {number} column The column to sort by.
- * @param {boolean=} opt_reverse Whether to sort in reverse.
- * @return {boolean} Whether the sort was executed.
- */
- goog.ui.TableSorter.prototype.sort = function(column, opt_reverse) {
- var sortFunction = this.getSortFunction(column);
- if (sortFunction === goog.ui.TableSorter.noSort) {
- return false;
- }
- // Remove old header classes.
- if (this.header_) {
- goog.dom.classlist.remove(
- this.header_, this.reversed_ ?
- goog.getCssName('goog-tablesorter-sorted-reverse') :
- goog.getCssName('goog-tablesorter-sorted'));
- }
- // If the user clicks on the same column, sort it in reverse of what it is
- // now. Otherwise, sort forward.
- this.reversed_ = !!opt_reverse;
- var multiplier = this.reversed_ ? -1 : 1;
- var cmpFn = function(a, b) {
- return multiplier * sortFunction(a[0], b[0]) || a[1] - b[1];
- };
- // Sort all tBodies
- var table = this.getElement();
- goog.array.forEach(table.tBodies, function(tBody) {
- // Collect all of the rows into an array.
- var values = goog.array.map(tBody.rows, function(row, rowIndex) {
- return [goog.dom.getTextContent(row.cells[column]), rowIndex, row];
- });
- goog.array.sort(values, cmpFn);
- // Remove the tBody temporarily since this speeds up the sort on some
- // browsers.
- var nextSibling = tBody.nextSibling;
- table.removeChild(tBody);
- // Sort the rows, using the resulting array.
- goog.array.forEach(values, function(row) { tBody.appendChild(row[2]); });
- // Reinstate the tBody.
- table.insertBefore(tBody, nextSibling);
- });
- // Mark this as the last sorted column.
- this.header_ = /** @type {!HTMLTableCellElement} */
- (table.tHead.rows[this.sortableHeaderRowIndex_].cells[column]);
- // Update the header class.
- goog.dom.classlist.add(
- this.header_, this.reversed_ ?
- goog.getCssName('goog-tablesorter-sorted-reverse') :
- goog.getCssName('goog-tablesorter-sorted'));
- return true;
- };
- /**
- * Disables sorting on the specified column
- * @param {*} a First sort value.
- * @param {*} b Second sort value.
- * @return {number} Negative if a < b, 0 if a = b, and positive if a > b.
- */
- goog.ui.TableSorter.noSort = goog.functions.error('no sort');
- /**
- * A numeric sort function. NaN values (or values that do not parse as float
- * numbers) compare equal to each other and greater to any other number.
- * @param {*} a First sort value.
- * @param {*} b Second sort value.
- * @return {number} Negative if a < b, 0 if a = b, and positive if a > b.
- */
- goog.ui.TableSorter.numericSort = function(a, b) {
- a = parseFloat(a);
- b = parseFloat(b);
- // foo == foo is false if and only if foo is NaN.
- if (a == a) {
- return b == b ? a - b : -1;
- } else {
- return b == b ? 1 : 0;
- }
- };
- /**
- * Alphabetic sort function.
- * @param {*} a First sort value.
- * @param {*} b Second sort value.
- * @return {number} Negative if a < b, 0 if a = b, and positive if a > b.
- */
- goog.ui.TableSorter.alphaSort = goog.array.defaultCompare;
- /**
- * Returns a function that is the given sort function in reverse.
- * @param {function(*, *) : number} sortFunction The original sort function.
- * @return {function(*, *) : number} A new sort function that reverses the
- * given sort function.
- */
- goog.ui.TableSorter.createReverseSort = function(sortFunction) {
- return function(a, b) { return -1 * sortFunction(a, b); };
- };
|