mirror of
https://github.com/Mottie/tablesorter.git
synced 2025-01-12 15:24:21 +00:00
293 lines
10 KiB
JavaScript
293 lines
10 KiB
JavaScript
/*!
|
|
Copyright (C) 2011 T. Connell & Associates, Inc.
|
|
|
|
Dual-licensed under the MIT and GPL licenses
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
|
|
FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
Resizable scroller widget for the jQuery tablesorter plugin
|
|
|
|
Version 2.0 - modified by Rob Garrison 4/12/2013; updated 3/5/2015 (v2.21.0)
|
|
Requires jQuery v1.7+
|
|
Requires the tablesorter plugin, v2.8+, available at http://mottie.github.com/tablesorter/docs/
|
|
|
|
Usage:
|
|
|
|
$(function() {
|
|
|
|
$('table.tablesorter').tablesorter({
|
|
widgets: ['zebra', 'scroller'],
|
|
widgetOptions : {
|
|
scroller_height : 300, // height of scroll window
|
|
scroller_barWidth : 18, // scroll bar width
|
|
scroller_jumpToHeader : true, // header snap to browser top when scrolling the tbody
|
|
}
|
|
});
|
|
|
|
});
|
|
|
|
Website: www.tconnell.com
|
|
*/
|
|
/*jshint browser:true, jquery:true, unused:false */
|
|
;(function($, window){
|
|
"use strict";
|
|
|
|
$.fn.hasScrollBar = function(){
|
|
return this.get(0).scrollHeight > this.height();
|
|
};
|
|
var ts = $.tablesorter;
|
|
|
|
ts.window_resize = function(){
|
|
if (this.resize_timer) {
|
|
clearTimeout(this.resize_timer);
|
|
}
|
|
this.resize_timer = setTimeout(function(){
|
|
$(this).trigger('resizeEnd');
|
|
}, 250);
|
|
};
|
|
|
|
// Add extra scroller css
|
|
$(function(){
|
|
var s = '<style>' +
|
|
'.tablesorter-scrollbar-measure { width: 100px; height: 100px; overflow: scroll; position: absolute; top: -9999px; } ' +
|
|
'.tablesorter-scroller-reset { width: auto !important; } ' +
|
|
'.tablesorter-scroller { text-align: left; overflow: hidden; }' +
|
|
'.tablesorter-scroller-header, .tablesorter-scroller-footer { overflow: hidden; }' +
|
|
'.tablesorter-scroller-header table.tablesorter { margin-bottom: 0; }' +
|
|
'.tablesorter-scroller-footer table.tablesorter thead { visibility: hidden, height: 0; overflow: hidden; }' +
|
|
'.tablesorter-scroller-table { overflow-y: scroll; }' +
|
|
'.tablesorter-scroller-table table.tablesorter { margin-top: 0; margin-bottom: 0; overflow: scroll; } ' +
|
|
'.tablesorter-scroller-table .tablesorter-filter-row,.tablesorter-scroller-footer .tablesorter-filter-row,.tablesorter-scroller-table tfoot { display: none; }' +
|
|
'.tablesorter-scroller-table table.tablesorter thead tr.tablesorter-headerRow *,.tablesorter-scroller-footer table.tablesorter thead * {' +
|
|
'line-height:0;height:0;border:none;background-image:none;padding-top:0;padding-bottom:0;margin-top:0;margin-bottom:0;overflow:hidden;' +
|
|
'}</style>';
|
|
$(s).appendTo('body');
|
|
});
|
|
|
|
ts.addWidget({
|
|
id: 'scroller',
|
|
priority: 60, // run after the filter widget
|
|
options: {
|
|
scroller_height : 300,
|
|
scroller_jumpToHeader: true,
|
|
scroller_upAfterSort: true,
|
|
// bar width is now calculated; set a value to override
|
|
scroller_barWidth : null
|
|
},
|
|
init: function(table, thisWidget, c, wo){
|
|
var $win = $(window),
|
|
namespace = c.namespace + 'tsscroller';
|
|
// Setup window.resizeEnd event
|
|
$win
|
|
.bind('resize' + namespace, ts.window_resize)
|
|
.bind('resizeEnd' + namespace, function() {
|
|
// init is run before format, so scroller_resizeWidth
|
|
// won't be defined within the "c" or "wo" parameters
|
|
if ($.isFunction(table.config.widgetOptions.scroller_resizeWidth)) {
|
|
// IE calls resize when you modify content, so we have to unbind the resize event
|
|
// so we don't end up with an infinite loop. we can rebind after we're done.
|
|
$win.unbind('resize' + namespace, ts.window_resize);
|
|
table.config.widgetOptions.scroller_resizeWidth();
|
|
$win.bind('resize' + namespace, ts.window_resize);
|
|
}
|
|
});
|
|
},
|
|
format: function(table, c, wo) {
|
|
var maxHt, tbHt, $hdr, $t, resize, getBarWidth, $hCells, $fCells, $tblWrap,
|
|
$ft = $(),
|
|
// c.namespace contains a unique tablesorter ID, per table
|
|
id = c.namespace.slice(1) + 'tsscroller',
|
|
$win = $(window),
|
|
$tbl = c.$table;
|
|
|
|
if (!c.isScrolling) {
|
|
// force developer to set fixedWidth to maintain column widths
|
|
c.widthFixed = true;
|
|
maxHt = wo.scroller_height || 300;
|
|
tbHt = $tbl.children('tbody').height();
|
|
if (tbHt !== 0 && maxHt > tbHt) { maxHt = tbHt + 10; } // Table is less than h px
|
|
|
|
$hdr = $('<table class="' + $tbl.attr('class') + '" cellpadding=0 cellspacing=0>' +
|
|
$tbl.children('thead')[0].outerHTML +
|
|
'</table>');
|
|
|
|
$t = $tbl.children('tfoot');
|
|
if ($t.length) {
|
|
$ft = $('<table class="' + $tbl.attr('class') + '" cellpadding=0 cellspacing=0 style="margin-top:0"></table>')
|
|
.append( $t.clone(true) ) // maintain any bindings on the tfoot cells
|
|
.append( $tbl.children('thead')[0].outerHTML )
|
|
.wrap('<div class="tablesorter-scroller-footer"/>');
|
|
$fCells = $ft.children('tfoot').eq(0).children('tr').children();
|
|
}
|
|
|
|
if (c.$extraTables && c.$extraTables.length) {
|
|
c.$extraTables = c.$extraTables.add($hdr).add($ft);
|
|
} else {
|
|
c.$extraTables = $hdr.add($ft);
|
|
}
|
|
$tbl
|
|
.wrap('<div id="' + id + '" class="tablesorter-scroller" />')
|
|
.before($hdr)
|
|
// shrink filter row but don't completely hide it because the inputs/selectors may distort the columns
|
|
.find('.tablesorter-filter-row').addClass('hideme');
|
|
|
|
if ($ft.length) {
|
|
// $ft.parent() to include <div> wrapper
|
|
$tbl.after( $ft.parent() );
|
|
}
|
|
|
|
$hCells = $hdr
|
|
.wrap('<div class="tablesorter-scroller-header" />')
|
|
.find('.' + ts.css.header);
|
|
|
|
// use max-height, so the height resizes dynamically while filtering
|
|
$tbl.wrap('<div class="tablesorter-scroller-table" style="max-height:' + maxHt + 'px;" />');
|
|
$tblWrap = $tbl.parent();
|
|
|
|
// make scroller header sortable
|
|
ts.bindEvents(table, $hCells);
|
|
|
|
// look for filter widget
|
|
if ($tbl.hasClass('hasFilters')) {
|
|
ts.filter.bindSearch( $tbl, $hdr.find('.' + ts.css.filter) );
|
|
}
|
|
|
|
// modified from http://davidwalsh.name/detect-scrollbar-width
|
|
getBarWidth = function(){
|
|
var $scrollDiv = $('<div class="tablesorter-scrollbar-measure">').appendTo('body'),
|
|
div = $scrollDiv[0],
|
|
barWidth = div.offsetWidth - div.clientWidth;
|
|
$scrollDiv.remove();
|
|
return barWidth;
|
|
};
|
|
|
|
resize = function(){
|
|
var b, $h, $f, w,
|
|
// Hide other scrollers so we can resize
|
|
$div = $('div.tablesorter-scroller[id != "' + id + '"]').hide();
|
|
|
|
$tbl.children('thead').show();
|
|
// only remove colgroup if it was added by the plugin
|
|
// the $.tablesorter.fixColumnWidth() function already does this (v2.19.0)
|
|
// but we need to get "accurate" resized measurements here - see issue #680
|
|
$tbl.add( $hdr ).add( $ft ).children('colgroup').remove();
|
|
|
|
// Reset sizes so parent can resize.
|
|
$tbl
|
|
.addClass('tablesorter-scroller-reset')
|
|
.children('thead')
|
|
.find('.tablesorter-header-inner').addClass('tablesorter-scroller-reset').end()
|
|
.find('.tablesorter-filter-row').show();
|
|
$tblWrap.addClass('tablesorter-scroller-reset');
|
|
|
|
$tblWrap.parent().trigger('resize');
|
|
|
|
// include left & right border widths
|
|
b = parseInt( $tbl.css('border-left-width'), 10 ) + parseInt( $tbl.css('border-right-width'), 10 );
|
|
|
|
// Shrink a bit to accommodate scrollbar
|
|
w = ( wo.scroller_barWidth || getBarWidth() ) + b;
|
|
|
|
$tblWrap.width( $tblWrap.parent().innerWidth() - ( $tblWrap.parent().hasScrollBar() ? w : 0 ) );
|
|
w = $tblWrap.innerWidth() - ( $tblWrap.hasScrollBar() ? w : 0 );
|
|
$hdr.parent().add( $ft.parent() ).width( w );
|
|
|
|
w = $tbl.width();
|
|
|
|
$h = $hdr.children('thead').children().children('th, td').filter(':visible');
|
|
$f = $ft.children('tfoot').children().children('th, td').filter(':visible');
|
|
$tbl.children('thead').children().eq(0).children('th, td').each(function(indx, el) {
|
|
var width, border,
|
|
$this = $(this);
|
|
// code from https://github.com/jmosbech/StickyTableHeaders
|
|
if ($this.css('box-sizing') === 'border-box') {
|
|
width = $this.outerWidth();
|
|
} else {
|
|
if ($h.eq(indx).css('border-collapse') === 'collapse') {
|
|
if (window.getComputedStyle) {
|
|
width = parseFloat( window.getComputedStyle(this, null).width );
|
|
} else {
|
|
// ie8 only
|
|
border = parseFloat( $this.css('border-width') );
|
|
width = $this.outerWidth() - parseFloat( $this.css('padding-left') ) - parseFloat( $this.css('padding-right') ) - border;
|
|
}
|
|
} else {
|
|
width = $this.width();
|
|
}
|
|
}
|
|
$h.eq(indx).add( $f.eq(indx) ).css({
|
|
'min-width': width,
|
|
'max-width': width
|
|
});
|
|
});
|
|
|
|
$tbl
|
|
.closest('.tablesorter-scroller')
|
|
.find('.tablesorter-scroller-reset')
|
|
.removeClass('tablesorter-scroller-reset');
|
|
|
|
// refresh colgroup & copy to cloned header
|
|
$.tablesorter.fixColumnWidth( table );
|
|
$h = $tbl.children('colgroup');
|
|
if ($h.length) {
|
|
b = $h[0].outerHTML;
|
|
$hdr.prepend(b);
|
|
if ($ft.length) {
|
|
$ft.prepend(b);
|
|
}
|
|
}
|
|
|
|
// hide filter row because filterEnd event fires
|
|
$tbl.children('thead').find('.tablesorter-filter-row').hide();
|
|
|
|
$div.show();
|
|
};
|
|
|
|
// Expose to external calls
|
|
wo.scroller_resizeWidth = resize;
|
|
|
|
resize();
|
|
|
|
$tbl.find('thead').css('visibility', 'hidden');
|
|
c.isScrolling = true;
|
|
|
|
tbHt = $tblWrap.parent().height();
|
|
|
|
// The header will always jump into view if scrolling the table body
|
|
$tblWrap.bind('scroll', function(){
|
|
if (wo.scroller_jumpToHeader) {
|
|
var pos = $win.scrollTop() - $hdr.offset().top;
|
|
if ($(this).scrollTop() !== 0 && pos < tbHt && pos > 0) {
|
|
$win.scrollTop( $hdr.offset().top );
|
|
}
|
|
}
|
|
$hdr.parent().add( $ft.parent() ).scrollLeft( $(this).scrollLeft() );
|
|
});
|
|
|
|
}
|
|
|
|
// Sorting, so scroll to top
|
|
if (wo.scroller_upAfterSort) {
|
|
$tbl.parent().animate({ scrollTop: 0 }, 'fast');
|
|
}
|
|
|
|
},
|
|
remove : function(table, c){
|
|
var $table = c.$table,
|
|
namespace = c.namespace + 'tsscroller';
|
|
$table.closest('.tablesorter-scroller').find('.tablesorter-scroller-header').remove();
|
|
$table.closest('.tablesorter-scroller').find('.tablesorter-scroller-footer').remove();
|
|
$table
|
|
.unwrap()
|
|
.find('.tablesorter-filter-row').removeClass('hideme').end()
|
|
.find('thead').show().css('visibility', 'visible');
|
|
$(window).unbind('resize' + namespace + ' resizeEnd' + namespace);
|
|
c.isScrolling = false;
|
|
}
|
|
});
|
|
|
|
})(jQuery, window);
|