update jQuery.floatThead to 1.2.7
This commit is contained in:
		| @@ -1,60 +1,49 @@ | |||||||
| /*! | // @preserve jQuery.floatThead 1.2.7 - http://mkoryak.github.io/floatThead/ - Copyright (c) 2012 - 2014 Misha Koryak | ||||||
|  * jQuery.floatThead | // @license MIT | ||||||
|  * Copyright (c) 2012 - 2013 Misha Koryak |  | ||||||
|  * Licensed under Attribution-ShareAlike 4.0 International - http://creativecommons.org/licenses/by-sa/4.0/ | /* @author Misha Koryak | ||||||
|  * Date: 12/12/13 |  | ||||||
|  * |  | ||||||
|  * @author Misha Koryak |  | ||||||
|  * @version 1.2.0 |  | ||||||
|  * @projectDescription lock a table header in place while scrolling - without breaking styles or events bound to the header |  * @projectDescription lock a table header in place while scrolling - without breaking styles or events bound to the header | ||||||
|  * |  * | ||||||
|  * Dependencies: |  * Dependencies: | ||||||
|  * jquery 1.9.0 + [required] OR jquery 1.7.0 + jquery UI core |  * jquery 1.9.0 + [required] OR jquery 1.7.0 + jquery UI core | ||||||
|  * underscore.js 1.3.0 + [required] |  | ||||||
|  * |  * | ||||||
|  * http://mkoryak.github.io/floatThead/ |  * http://mkoryak.github.io/floatThead/ | ||||||
|  * |  * | ||||||
|  * Tested on FF13+, Chrome 21+, IE8, IE9, IE10 |  * Tested on FF13+, Chrome 21+, IE8, IE9, IE10, IE11 | ||||||
|  * |  * | ||||||
|  */ |  */ | ||||||
| // ==ClosureCompiler== |  | ||||||
| // @compilation_level SIMPLE_OPTIMIZATIONS |  | ||||||
| // @output_file_name jquery.floatThead.min.js |  | ||||||
| // ==/ClosureCompiler== |  | ||||||
| /** |  | ||||||
|  * @preserve jQuery.floatThead 1.2.0 |  | ||||||
|  * Copyright (c) 2013 Misha Koryak - http://mkoryak.github.io/floatThead/ |  | ||||||
|  * Licensed under http://creativecommons.org/licenses/by-sa/4.0/ |  | ||||||
|  */ |  | ||||||
| (function( $ ) { | (function( $ ) { | ||||||
|   /** |   /** | ||||||
|    * provides a default config object. You can modify this after including this script if you want to change the init defaults |    * provides a default config object. You can modify this after including this script if you want to change the init defaults | ||||||
|    * @type {Object} |    * @type {Object} | ||||||
|    */ |    */ | ||||||
|   $.floatThead = { |   $.floatThead = $.floatThead || {}; | ||||||
|     defaults: { |   $.floatThead.defaults = { | ||||||
|       cellTag: 'th', |     cellTag: 'th:visible', //thead cells are this | ||||||
|     zIndex: 1001, //zindex of the floating thead (actually a container div) |     zIndex: 1001, //zindex of the floating thead (actually a container div) | ||||||
|       debounceResizeMs: 1, |     debounceResizeMs: 10, | ||||||
|     useAbsolutePositioning: true, //if set to NULL - defaults: has scrollContainer=true, doesn't have scrollContainer=false |     useAbsolutePositioning: true, //if set to NULL - defaults: has scrollContainer=true, doesn't have scrollContainer=false | ||||||
|     scrollingTop: 0, //String or function($table) - offset from top of window where the header should not pass above |     scrollingTop: 0, //String or function($table) - offset from top of window where the header should not pass above | ||||||
|     scrollingBottom: 0, //String or function($table) - offset from the bottom of the table where the header should stop scrolling |     scrollingBottom: 0, //String or function($table) - offset from the bottom of the table where the header should stop scrolling | ||||||
|     scrollContainer: function($table){ |     scrollContainer: function($table){ | ||||||
|       return $([]); //if the table has horizontal scroll bars then this is the container that has overflow:auto and causes those scroll bars |       return $([]); //if the table has horizontal scroll bars then this is the container that has overflow:auto and causes those scroll bars | ||||||
|     }, |     }, | ||||||
|       getSizingRow: function($table, $cols, $fthCells){ // this is only called when using IE8, |     getSizingRow: function($table, $cols, $fthCells){ // this is only called when using IE, | ||||||
|       // override it if the first row of the table is going to contain colgroups (any cell spans greater then one col) |       // override it if the first row of the table is going to contain colgroups (any cell spans greater then one col) | ||||||
|       // it should return a jquery object containing a wrapped set of table cells comprising a row that contains no col spans and is visible |       // it should return a jquery object containing a wrapped set of table cells comprising a row that contains no col spans and is visible | ||||||
|       return $table.find('tbody tr:visible:first>td'); |       return $table.find('tbody tr:visible:first>td'); | ||||||
|     }, |     }, | ||||||
|     floatTableClass: 'floatThead-table', |     floatTableClass: 'floatThead-table', | ||||||
|  |     floatWrapperClass: 'floatThead-wrapper', | ||||||
|  |     floatContainerClass: 'floatThead-container', | ||||||
|  |     copyTableClass: true, //copy 'class' attribute from table into the floated table so that the styles match. | ||||||
|     debug: false //print possible issues (that don't prevent script loading) to console, if console exists. |     debug: false //print possible issues (that don't prevent script loading) to console, if console exists. | ||||||
|     } |  | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|  |   var util = window._; | ||||||
|  |  | ||||||
|   //browser stuff |   //browser stuff | ||||||
|   var ieVersion = function(){for(var a=3,b=document.createElement("b"),c=b.all||[];b.innerHTML="<!--[if gt IE "+ ++a+"]><i><![endif]-->",c[0];);return 4<a?a:document.documentMode}(); |   var ieVersion = function(){for(var a=3,b=document.createElement("b"),c=b.all||[];a = 1+a,b.innerHTML="<!--[if gt IE "+ a +"]><i><![endif]-->",c[0];);return 4<a?a:document.documentMode}(); | ||||||
|   var isChrome = null; |   var isChrome = null; | ||||||
|   var isChromeCheck = function(){ |   var isChromeCheck = function(){ | ||||||
|     if(ieVersion){ |     if(ieVersion){ | ||||||
| @@ -72,21 +61,24 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * debounce and fix window resize event for ie7. ie7 is evil and will fire window resize event when ANY dom element is resized. |  | ||||||
|    * @param debounceMs |    * @param debounceMs | ||||||
|    * @param cb |    * @param cb | ||||||
|    */ |    */ | ||||||
|  |  | ||||||
|   function windowResize(debounceMs, cb){ |   function windowResize(debounceMs, eventName, cb){ | ||||||
|  |     if(ieVersion == 8){ //ie8 is crap: https://github.com/mkoryak/floatThead/issues/65 | ||||||
|       var winWidth = $window.width(); |       var winWidth = $window.width(); | ||||||
|     var debouncedCb = _.debounce(function(){ |       var debouncedCb = util.debounce(function(){ | ||||||
|         var winWidthNew = $window.width(); |         var winWidthNew = $window.width(); | ||||||
|         if(winWidth != winWidthNew){ |         if(winWidth != winWidthNew){ | ||||||
|           winWidth = winWidthNew; |           winWidth = winWidthNew; | ||||||
|           cb(); |           cb(); | ||||||
|         } |         } | ||||||
|       }, debounceMs); |       }, debounceMs); | ||||||
|     $window.bind('resize.floatTHead', debouncedCb); |       $window.on(eventName, debouncedCb); | ||||||
|  |     } else { | ||||||
|  |       $window.on(eventName, util.debounce(cb, debounceMs)); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -127,6 +119,14 @@ | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|   $.fn.floatThead = function(map){ |   $.fn.floatThead = function(map){ | ||||||
|  |     map = map || {}; | ||||||
|  |     if(!util){ //may have been included after the script? lets try to grab it again. | ||||||
|  |       util = window._ || $.floatThead._; | ||||||
|  |       if(!util){ | ||||||
|  |         throw new Error("jquery.floatThead-slim.js requires underscore. You should use the non-lite version since you do not have underscore."); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     if(ieVersion < 8){ |     if(ieVersion < 8){ | ||||||
|       return this; //no more crappy browser support. |       return this; //no more crappy browser support. | ||||||
|     } |     } | ||||||
| @@ -140,12 +140,12 @@ | |||||||
|         document.createElement('fthfoot'); //tfoot |         document.createElement('fthfoot'); //tfoot | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if(_.isString(map)){ |     if(util.isString(map)){ | ||||||
|       var command = map; |       var command = map; | ||||||
|       var ret = this; |       var ret = this; | ||||||
|       this.filter('table').each(function(){ |       this.filter('table').each(function(){ | ||||||
|         var obj = $(this).data('floatThead-attached'); |         var obj = $(this).data('floatThead-attached'); | ||||||
|         if(obj && _.isFunction(obj[command])){ |         if(obj && util.isFunction(obj[command])){ | ||||||
|           var r = obj[command](); |           var r = obj[command](); | ||||||
|           if(typeof r !== 'undefined'){ |           if(typeof r !== 'undefined'){ | ||||||
|             ret = r; |             ret = r; | ||||||
| @@ -154,15 +154,16 @@ | |||||||
|       }); |       }); | ||||||
|       return ret; |       return ret; | ||||||
|     } |     } | ||||||
|     var opts = $.extend({}, $.floatThead.defaults, map); |     var opts = $.extend({}, $.floatThead.defaults || {}, map); | ||||||
|  |  | ||||||
|     _.each(map, function(val, key){ |     $.each(map, function(key, val){ | ||||||
|       if((!(key in $.floatThead.defaults)) && opts.debug){ |       if((!(key in $.floatThead.defaults)) && opts.debug){ | ||||||
|         debug("jQuery.floatThead: used ["+key+"] key to init plugin, but that param is not an option for the plugin. Valid options are: "+ (_.keys($.floatThead.defaults)).join(', ')); |         debug("jQuery.floatThead: used ["+key+"] key to init plugin, but that param is not an option for the plugin. Valid options are: "+ (util.keys($.floatThead.defaults)).join(', ')); | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     this.filter(':not(.'+opts.floatTableClass+')').each(function(){ |     this.filter(':not(.'+opts.floatTableClass+')').each(function(){ | ||||||
|  |       var floatTheadId = floatTheadCreated; | ||||||
|       var $table = $(this); |       var $table = $(this); | ||||||
|       if($table.data('floatThead-attached')){ |       if($table.data('floatThead-attached')){ | ||||||
|         return true; //continue the each loop |         return true; //continue the each loop | ||||||
| @@ -175,7 +176,7 @@ | |||||||
|       if($header.length == 0){ |       if($header.length == 0){ | ||||||
|         throw new Error('jQuery.floatThead must be run on a table that contains a <thead> element'); |         throw new Error('jQuery.floatThead must be run on a table that contains a <thead> element'); | ||||||
|       } |       } | ||||||
|       var headerFloated = true; |       var headerFloated = false; | ||||||
|       var scrollingTop, scrollingBottom; |       var scrollingTop, scrollingBottom; | ||||||
|       var scrollbarOffset = {vertical: 0, horizontal: 0}; |       var scrollbarOffset = {vertical: 0, horizontal: 0}; | ||||||
|       var scWidth = scrollbarWidth(); |       var scWidth = scrollbarWidth(); | ||||||
| @@ -196,10 +197,16 @@ | |||||||
|  |  | ||||||
|       var locked = $scrollContainer.length > 0; |       var locked = $scrollContainer.length > 0; | ||||||
|       var wrappedContainer = false; //used with absolute positioning enabled. did we need to wrap the scrollContainer/table with a relative div? |       var wrappedContainer = false; //used with absolute positioning enabled. did we need to wrap the scrollContainer/table with a relative div? | ||||||
|  |       var $wrapper = $([]); //used when absolute positioning enabled - wraps the table and the float container | ||||||
|       var absoluteToFixedOnScroll = ieVersion <= 9 && !locked && useAbsolutePositioning; //on ie using absolute positioning doesnt look good with window scrolling, so we change positon to fixed on scroll, and then change it back to absolute when done. |       var absoluteToFixedOnScroll = ieVersion <= 9 && !locked && useAbsolutePositioning; //on ie using absolute positioning doesnt look good with window scrolling, so we change positon to fixed on scroll, and then change it back to absolute when done. | ||||||
|       var $floatTable = $("<table/>"); |       var $floatTable = $("<table/>"); | ||||||
|       var $floatColGroup = $("<colgroup/>"); |       var $floatColGroup = $("<colgroup/>"); | ||||||
|       var $tableColGroup = $("<colgroup/>"); |       var $tableColGroup = $table.find('colgroup:first'); | ||||||
|  |       var existingColGroup = true; | ||||||
|  |       if($tableColGroup.length == 0){ | ||||||
|  |         $tableColGroup = $("<colgroup/>"); | ||||||
|  |         existingColGroup = false; | ||||||
|  |       } | ||||||
|       var $fthRow = $('<fthrow style="display:table-row;height:0;"/>'); //created unstyled elements |       var $fthRow = $('<fthrow style="display:table-row;height:0;"/>'); //created unstyled elements | ||||||
|       var $floatContainer = $('<div style="overflow: hidden;"></div>'); |       var $floatContainer = $('<div style="overflow: hidden;"></div>'); | ||||||
|       var $newHeader = $("<thead/>"); |       var $newHeader = $("<thead/>"); | ||||||
| @@ -210,9 +217,6 @@ | |||||||
|       var $fthCells = $([]); //created elements |       var $fthCells = $([]); //created elements | ||||||
|  |  | ||||||
|       $newHeader.append($sizerRow); |       $newHeader.append($sizerRow); | ||||||
|       $header.detach(); |  | ||||||
|  |  | ||||||
|       $table.prepend($newHeader); |  | ||||||
|       $table.prepend($tableColGroup); |       $table.prepend($tableColGroup); | ||||||
|       if(isChrome){ |       if(isChrome){ | ||||||
|         $fthGrp.append($fthRow); |         $fthGrp.append($fthRow); | ||||||
| @@ -221,7 +225,15 @@ | |||||||
|  |  | ||||||
|       $floatTable.append($floatColGroup); |       $floatTable.append($floatColGroup); | ||||||
|       $floatContainer.append($floatTable); |       $floatContainer.append($floatTable); | ||||||
|  |       if(opts.copyTableClass){ | ||||||
|         $floatTable.attr('class', $table.attr('class')); |         $floatTable.attr('class', $table.attr('class')); | ||||||
|  |       } | ||||||
|  |       $floatTable.attr({ //copy over some deprecated table attributes that people still like to use. Good thing poeple dont use colgroups... | ||||||
|  |         'cellpadding': $table.attr('cellpadding'), | ||||||
|  |         'cellspacing': $table.attr('cellspacing'), | ||||||
|  |         'border': $table.attr('border') | ||||||
|  |       }); | ||||||
|  |  | ||||||
|       $floatTable.addClass(opts.floatTableClass).css('margin', 0); //must have no margins or you wont be able to click on things under floating table |       $floatTable.addClass(opts.floatTableClass).css('margin', 0); //must have no margins or you wont be able to click on things under floating table | ||||||
|  |  | ||||||
|       if(useAbsolutePositioning){ |       if(useAbsolutePositioning){ | ||||||
| @@ -231,16 +243,16 @@ | |||||||
|           if(!relativeToScrollContainer || alwaysWrap){ |           if(!relativeToScrollContainer || alwaysWrap){ | ||||||
|             var css = {"paddingLeft": $container.css('paddingLeft'), "paddingRight": $container.css('paddingRight')}; |             var css = {"paddingLeft": $container.css('paddingLeft'), "paddingRight": $container.css('paddingRight')}; | ||||||
|             $floatContainer.css(css); |             $floatContainer.css(css); | ||||||
|             $container = $container.wrap("<div style='position: relative; clear:both;'></div>").parent(); |             $container = $container.wrap("<div class='"+opts.floatWrapperClass+"' style='position: relative; clear:both;'></div>").parent(); | ||||||
|             wrappedContainer = true; |             wrappedContainer = true; | ||||||
|           } |           } | ||||||
|           return $container; |           return $container; | ||||||
|         }; |         }; | ||||||
|         if(locked){ |         if(locked){ | ||||||
|           var $relative = makeRelative($scrollContainer, true); |           $wrapper = makeRelative($scrollContainer, true); | ||||||
|           $relative.append($floatContainer); |           $wrapper.append($floatContainer); | ||||||
|         } else { |         } else { | ||||||
|           makeRelative($table); |           $wrapper = makeRelative($table); | ||||||
|           $table.after($floatContainer); |           $table.after($floatContainer); | ||||||
|         } |         } | ||||||
|       } else { |       } else { | ||||||
| @@ -254,13 +266,22 @@ | |||||||
|         top:  useAbsolutePositioning ? 0 : 'auto', |         top:  useAbsolutePositioning ? 0 : 'auto', | ||||||
|         zIndex: opts.zIndex |         zIndex: opts.zIndex | ||||||
|       }); |       }); | ||||||
|  |       $floatContainer.addClass(opts.floatContainerClass); | ||||||
|       updateScrollingOffsets(); |       updateScrollingOffsets(); | ||||||
|  |  | ||||||
|       var layoutFixed = {'table-layout': 'fixed'}; |       var layoutFixed = {'table-layout': 'fixed'}; | ||||||
|       var layoutAuto = {'table-layout': $table.css('tableLayout') || 'auto'}; |       var layoutAuto = {'table-layout': $table.css('tableLayout') || 'auto'}; | ||||||
|  |       var originalTableWidth = $table[0].style.width || ""; //setting this to auto is bad: #70 | ||||||
|  |  | ||||||
|  |       function eventName(name){ | ||||||
|  |         return name+'.fth-'+floatTheadId+'.floatTHead' | ||||||
|  |       } | ||||||
|  |  | ||||||
|       function setHeaderHeight(){ |       function setHeaderHeight(){ | ||||||
|         var headerHeight = $header.outerHeight(true); |         var headerHeight = 0; | ||||||
|  |         $header.find("tr").each(function(){ | ||||||
|  |           headerHeight += $(this).outerHeight(true); | ||||||
|  |         }); | ||||||
|         $sizerRow.outerHeight(headerHeight); |         $sizerRow.outerHeight(headerHeight); | ||||||
|         $sizerCells.outerHeight(headerHeight); |         $sizerCells.outerHeight(headerHeight); | ||||||
|       } |       } | ||||||
| @@ -279,25 +300,29 @@ | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       function updateScrollingOffsets(){ |       function updateScrollingOffsets(){ | ||||||
|         scrollingTop = (_.isFunction(opts.scrollingTop) ? opts.scrollingTop($table) : opts.scrollingTop) || 0; |         scrollingTop = (util.isFunction(opts.scrollingTop) ? opts.scrollingTop($table) : opts.scrollingTop) || 0; | ||||||
|         scrollingBottom = (_.isFunction(opts.scrollingBottom) ? opts.scrollingBottom($table) : opts.scrollingBottom) || 0; |         scrollingBottom = (util.isFunction(opts.scrollingBottom) ? opts.scrollingBottom($table) : opts.scrollingBottom) || 0; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       /** |       /** | ||||||
|        * get the number of columns and also rebuild resizer rows if the count is different then the last count |        * get the number of columns and also rebuild resizer rows if the count is different then the last count | ||||||
|        */ |        */ | ||||||
|       function columnNum(){ |       function columnNum(){ | ||||||
|         var $headerColumns = $header.find('tr:first>'+opts.cellTag); |         var count, $headerColumns; | ||||||
|  |         if(existingColGroup){ | ||||||
|         var count =  _.reduce($headerColumns, function(sum, cell){ |           count = $tableColGroup.find('col').length; | ||||||
|           var colspan = parseInt(($(cell).attr('colspan') || 1), 10); |         } else { | ||||||
|           return sum + colspan; |           $headerColumns = $header.find('tr:first>'+opts.cellTag); | ||||||
|         }, 0); |           count = 0; | ||||||
|  |           $headerColumns.each(function(){ | ||||||
|  |             count += parseInt(($(this).attr('colspan') || 1), 10); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|         if(count != lastColumnCount){ |         if(count != lastColumnCount){ | ||||||
|           lastColumnCount = count; |           lastColumnCount = count; | ||||||
|           var cells = [], cols = [], psuedo = []; |           var cells = [], cols = [], psuedo = []; | ||||||
|           for(var x = 0; x < count; x++){ |           for(var x = 0; x < count; x++){ | ||||||
|             cells.push('<'+opts.cellTag+' class="floatThead-col-'+x+'"/>'); |             cells.push('<th class="floatThead-col"/>'); | ||||||
|             cols.push('<col/>'); |             cols.push('<col/>'); | ||||||
|             psuedo.push("<fthtd style='display:table-cell;height:0;width:auto;'/>"); |             psuedo.push("<fthtd style='display:table-cell;height:0;width:auto;'/>"); | ||||||
|           } |           } | ||||||
| @@ -312,7 +337,10 @@ | |||||||
|           } |           } | ||||||
|  |  | ||||||
|           $sizerRow.html(cells); |           $sizerRow.html(cells); | ||||||
|  |           $sizerCells = $sizerRow.find("th"); | ||||||
|  |           if(!existingColGroup){ | ||||||
|             $tableColGroup.html(cols); |             $tableColGroup.html(cols); | ||||||
|  |           } | ||||||
|           $tableCells = $tableColGroup.find('col'); |           $tableCells = $tableColGroup.find('col'); | ||||||
|           $floatColGroup.html(cols); |           $floatColGroup.html(cols); | ||||||
|           $headerCells = $floatColGroup.find("col"); |           $headerCells = $floatColGroup.find("col"); | ||||||
| @@ -324,6 +352,13 @@ | |||||||
|       function refloat(){ //make the thing float |       function refloat(){ //make the thing float | ||||||
|         if(!headerFloated){ |         if(!headerFloated){ | ||||||
|           headerFloated = true; |           headerFloated = true; | ||||||
|  |           if(useAbsolutePositioning){ //#53, #56 | ||||||
|  |             var tableWidth = $table.width(); | ||||||
|  |             var wrapperWidth = $wrapper.width(); | ||||||
|  |             if(tableWidth > wrapperWidth){ | ||||||
|  |               $table.css('minWidth', tableWidth); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|           $table.css(layoutFixed); |           $table.css(layoutFixed); | ||||||
|           $floatTable.css(layoutFixed); |           $floatTable.css(layoutFixed); | ||||||
|           $floatTable.append($header); //append because colgroup must go first in chrome |           $floatTable.append($header); //append because colgroup must go first in chrome | ||||||
| @@ -334,6 +369,9 @@ | |||||||
|       function unfloat(){ //put the header back into the table |       function unfloat(){ //put the header back into the table | ||||||
|         if(headerFloated){ |         if(headerFloated){ | ||||||
|           headerFloated = false; |           headerFloated = false; | ||||||
|  |           if(useAbsolutePositioning){ //#53, #56 | ||||||
|  |             $table.width(originalTableWidth); | ||||||
|  |           } | ||||||
|           $newHeader.detach(); |           $newHeader.detach(); | ||||||
|           $table.prepend($header); |           $table.prepend($header); | ||||||
|           $table.css(layoutAuto); |           $table.css(layoutAuto); | ||||||
| @@ -368,6 +406,11 @@ | |||||||
|         return function(){ |         return function(){ | ||||||
|           var $rowCells = getSizingRow($table, $tableCells, $fthCells, ieVersion); |           var $rowCells = getSizingRow($table, $tableCells, $fthCells, ieVersion); | ||||||
|           if($rowCells.length == numCols && numCols > 0){ |           if($rowCells.length == numCols && numCols > 0){ | ||||||
|  |             if(!existingColGroup){ | ||||||
|  |               for(i=0; i < numCols; i++){ | ||||||
|  |                 $tableCells.eq(i).css('width', ''); | ||||||
|  |               } | ||||||
|  |             } | ||||||
|             unfloat(); |             unfloat(); | ||||||
|             for(i=0; i < numCols; i++){ |             for(i=0; i < numCols; i++){ | ||||||
|               var _rowcell = $rowCells.get(i); |               var _rowcell = $rowCells.get(i); | ||||||
| @@ -397,6 +440,7 @@ | |||||||
|         var floatEnd; |         var floatEnd; | ||||||
|         var tableContainerGap = 0; |         var tableContainerGap = 0; | ||||||
|         var captionHeight = haveCaption ? $caption.outerHeight(true) : 0; |         var captionHeight = haveCaption ? $caption.outerHeight(true) : 0; | ||||||
|  |         var captionScrollOffset = captionAlignTop ? captionHeight : -captionHeight; | ||||||
|  |  | ||||||
|         var floatContainerHeight = $floatContainer.height(); |         var floatContainerHeight = $floatContainer.height(); | ||||||
|         var tableOffset = $table.offset(); |         var tableOffset = $table.offset(); | ||||||
| @@ -461,13 +505,13 @@ | |||||||
|             left = 0; |             left = 0; | ||||||
|           } else if(!locked && useAbsolutePositioning) { //window scrolling, absolute positioning |           } else if(!locked && useAbsolutePositioning) { //window scrolling, absolute positioning | ||||||
|             tableHeight = $table.outerHeight(); |             tableHeight = $table.outerHeight(); | ||||||
|             if(windowTop > floatEnd + tableHeight + captionHeight){ |             if(windowTop > floatEnd + tableHeight + captionScrollOffset){ | ||||||
|               top = tableHeight - floatContainerHeight + captionHeight; //scrolled past table |               top = tableHeight - floatContainerHeight + captionScrollOffset; //scrolled past table | ||||||
|             } else if (tableOffset.top > windowTop + scrollingTop) { |             } else if (tableOffset.top > windowTop + scrollingTop) { | ||||||
|               top = 0; //scrolling to table |               top = 0; //scrolling to table | ||||||
|               unfloat(); |               unfloat(); | ||||||
|             } else { |             } else { | ||||||
|               top = scrollingTop + windowTop - tableOffset.top + tableContainerGap + captionHeight; |               top = scrollingTop + windowTop - tableOffset.top + tableContainerGap + (captionAlignTop ? captionHeight : 0); | ||||||
|               refloat(); //scrolling within table. header floated |               refloat(); //scrolling within table. header floated | ||||||
|             } |             } | ||||||
|             left =  0; |             left =  0; | ||||||
| @@ -483,8 +527,8 @@ | |||||||
|             left = tableOffset.left + scrollContainerLeft - windowLeft; |             left = tableOffset.left + scrollContainerLeft - windowLeft; | ||||||
|           } else if(!locked && !useAbsolutePositioning) { //window scrolling, fixed positioning |           } else if(!locked && !useAbsolutePositioning) { //window scrolling, fixed positioning | ||||||
|             tableHeight = $table.outerHeight(); |             tableHeight = $table.outerHeight(); | ||||||
|             if(windowTop > floatEnd + tableHeight + captionHeight){ |             if(windowTop > floatEnd + tableHeight + captionScrollOffset){ | ||||||
|               top = tableHeight + scrollingTop - windowTop + floatEnd + captionHeight; |               top = tableHeight + scrollingTop - windowTop + floatEnd + captionScrollOffset; | ||||||
|               //scrolled past the bottom of the table |               //scrolled past the bottom of the table | ||||||
|             } else if (tableOffset.top > windowTop + scrollingTop) { |             } else if (tableOffset.top > windowTop + scrollingTop) { | ||||||
|               top = tableOffset.top - windowTop; |               top = tableOffset.top - windowTop; | ||||||
| @@ -542,15 +586,21 @@ | |||||||
|       //finish up. create all calculation functions and bind them to events |       //finish up. create all calculation functions and bind them to events | ||||||
|       calculateScrollBarSize(); |       calculateScrollBarSize(); | ||||||
|  |  | ||||||
|       var flow = reflow(); |       var flow; | ||||||
|  |  | ||||||
|  |       var ensureReflow = function(){ | ||||||
|  |         flow = reflow(); | ||||||
|         flow(); |         flow(); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       ensureReflow(); | ||||||
|  |  | ||||||
|       var calculateFloatContainerPos = calculateFloatContainerPosFn(); |       var calculateFloatContainerPos = calculateFloatContainerPosFn(); | ||||||
|       var repositionFloatContainer = repositionFloatContainerFn(); |       var repositionFloatContainer = repositionFloatContainerFn(); | ||||||
|  |  | ||||||
|       repositionFloatContainer(calculateFloatContainerPos('init'), true, true); |       repositionFloatContainer(calculateFloatContainerPos('init'), true); //this must come after reflow because reflow changes scrollLeft back to 0 when it rips out the thead | ||||||
|       //this must come after reflow because reflow changes scrollLeft back to 0 when it rips out the thead |  | ||||||
|  |  | ||||||
|       var windowScrollDoneEvent = _.debounce(function(){ |       var windowScrollDoneEvent = util.debounce(function(){ | ||||||
|         repositionFloatContainer(calculateFloatContainerPos('windowScrollDone'), false); |         repositionFloatContainer(calculateFloatContainerPos('windowScrollDone'), false); | ||||||
|       }, 300); |       }, 300); | ||||||
|  |  | ||||||
| @@ -562,10 +612,6 @@ | |||||||
|         repositionFloatContainer(calculateFloatContainerPos('containerScroll'), false); |         repositionFloatContainer(calculateFloatContainerPos('containerScroll'), false); | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       var ensureReflow = function(){ |  | ||||||
|         flow = reflow(); |  | ||||||
|         flow(); |  | ||||||
|       }; |  | ||||||
|  |  | ||||||
|       var windowResizeEvent = function(){ |       var windowResizeEvent = function(){ | ||||||
|         updateScrollingOffsets(); |         updateScrollingOffsets(); | ||||||
| @@ -575,7 +621,7 @@ | |||||||
|         repositionFloatContainer = repositionFloatContainerFn(); |         repositionFloatContainer = repositionFloatContainerFn(); | ||||||
|         repositionFloatContainer(calculateFloatContainerPos('resize'), true, true); |         repositionFloatContainer(calculateFloatContainerPos('resize'), true, true); | ||||||
|       }; |       }; | ||||||
|       var reflowEvent = _.debounce(function(){ |       var reflowEvent = util.debounce(function(){ | ||||||
|         calculateScrollBarSize(); |         calculateScrollBarSize(); | ||||||
|         updateScrollingOffsets(); |         updateScrollingOffsets(); | ||||||
|         ensureReflow(); |         ensureReflow(); | ||||||
| @@ -584,46 +630,46 @@ | |||||||
|       }, 1); |       }, 1); | ||||||
|       if(locked){ //internal scrolling |       if(locked){ //internal scrolling | ||||||
|         if(useAbsolutePositioning){ |         if(useAbsolutePositioning){ | ||||||
|           $scrollContainer.bind('scroll.floatTHead', containerScrollEvent); |           $scrollContainer.on(eventName('scroll'), containerScrollEvent); | ||||||
|         } else { |         } else { | ||||||
|           $scrollContainer.bind('scroll.floatTHead', containerScrollEvent); |           $scrollContainer.on(eventName('scroll'), containerScrollEvent); | ||||||
|           $window.bind('scroll.floatTHead', windowScrollEvent); |           $window.on(eventName('scroll'), windowScrollEvent); | ||||||
|         } |         } | ||||||
|       } else { //window scrolling |       } else { //window scrolling | ||||||
|         $window.bind('scroll.floatTHead', windowScrollEvent); |         $window.on(eventName('scroll'), windowScrollEvent); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       $window.bind('load.floatTHead', reflowEvent); //for tables with images |       $window.on(eventName('load'), reflowEvent); //for tables with images | ||||||
|  |  | ||||||
|       windowResize(opts.debounceResizeMs, windowResizeEvent); |       windowResize(opts.debounceResizeMs, eventName('resize'), windowResizeEvent); | ||||||
|       $table.bind('reflow', reflowEvent); |       $table.on('reflow', reflowEvent); | ||||||
|       if(isDatatable($table)){ |       if(isDatatable($table)){ | ||||||
|         $table |         $table | ||||||
|           .bind('filter', reflowEvent) |           .on('filter', reflowEvent) | ||||||
|           .bind('sort',   reflowEvent) |           .on('sort',   reflowEvent) | ||||||
|           .bind('page',   reflowEvent); |           .on('page',   reflowEvent); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       //attach some useful functions to the table. |       //attach some useful functions to the table. | ||||||
|       $table.data('floatThead-attached', { |       $table.data('floatThead-attached', { | ||||||
|         destroy: function(){ |         destroy: function(){ | ||||||
|  |           var ns = '.fth-'+floatTheadId; | ||||||
|  |           unfloat(); | ||||||
|           $table.css(layoutAuto); |           $table.css(layoutAuto); | ||||||
|           $tableColGroup.remove(); |           $tableColGroup.remove(); | ||||||
|           isChrome && $fthGrp.remove(); |           isChrome && $fthGrp.remove(); | ||||||
|           if($newHeader.parent().length){ //only if its in the dom |           if($newHeader.parent().length){ //only if its in the dom | ||||||
|             $newHeader.replaceWith($header); |             $newHeader.replaceWith($header); | ||||||
|           } |           } | ||||||
|           $table.unbind('reflow'); |           $table.off('reflow'); | ||||||
|           reflowEvent = windowResizeEvent = containerScrollEvent = windowScrollEvent = function() {}; |           $scrollContainer.off(ns); | ||||||
|           $scrollContainer.unbind('scroll.floatTHead'); |           if (wrappedContainer) { | ||||||
|  |             $scrollContainer.unwrap(); | ||||||
|  |           } | ||||||
|           $floatContainer.remove(); |           $floatContainer.remove(); | ||||||
|           $table.data('floatThead-attached', false); |           $table.data('floatThead-attached', false); | ||||||
|           floatTheadCreated--; |  | ||||||
|           if(floatTheadCreated == 0){ |           $window.off(ns); | ||||||
|             $window.unbind('scroll.floatTHead'); |  | ||||||
|             $window.unbind('resize.floatTHead'); |  | ||||||
|             $window.unbind('load.floatTHead'); |  | ||||||
|           } |  | ||||||
|         }, |         }, | ||||||
|         reflow: function(){ |         reflow: function(){ | ||||||
|           reflowEvent(); |           reflowEvent(); | ||||||
| @@ -633,6 +679,13 @@ | |||||||
|         }, |         }, | ||||||
|         getFloatContainer: function(){ |         getFloatContainer: function(){ | ||||||
|           return $floatContainer; |           return $floatContainer; | ||||||
|  |         }, | ||||||
|  |         getRowGroups: function(){ | ||||||
|  |           if(headerFloated){ | ||||||
|  |             return $floatContainer.find("thead").add($table.find("tbody,tfoot")); | ||||||
|  |           } else { | ||||||
|  |             return $table.find("thead,tbody,tfoot"); | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|       }); |       }); | ||||||
|       floatTheadCreated++; |       floatTheadCreated++; | ||||||
| @@ -640,3 +693,62 @@ | |||||||
|     return this; |     return this; | ||||||
|   }; |   }; | ||||||
| })(jQuery); | })(jQuery); | ||||||
|  | /* jQuery.floatThead.utils - http://mkoryak.github.io/floatThead/ - Copyright (c) 2012 - 2014 Misha Koryak | ||||||
|  |  * License: MIT | ||||||
|  |  * | ||||||
|  |  * This file is required if you do not use underscore in your project and you want to use floatThead. | ||||||
|  |  * It contains functions from underscore that the plugin uses. | ||||||
|  |  * | ||||||
|  |  * YOU DON'T NEED TO INCLUDE THIS IF YOU ALREADY INCLUDE UNDERSCORE! | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | (function(){ | ||||||
|  |  | ||||||
|  |   $.floatThead = $.floatThead || {}; | ||||||
|  |  | ||||||
|  |   $.floatThead._  = window._ || (function(){ | ||||||
|  |     var that = {}; | ||||||
|  |     var hasOwnProperty = Object.prototype.hasOwnProperty, isThings = ['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp']; | ||||||
|  |     that.has = function(obj, key) { | ||||||
|  |       return hasOwnProperty.call(obj, key); | ||||||
|  |     }; | ||||||
|  |     that.keys = function(obj) { | ||||||
|  |       if (obj !== Object(obj)) throw new TypeError('Invalid object'); | ||||||
|  |       var keys = []; | ||||||
|  |       for (var key in obj) if (that.has(obj, key)) keys.push(key); | ||||||
|  |       return keys; | ||||||
|  |     }; | ||||||
|  |     $.each(isThings, function(){ | ||||||
|  |       var name = this; | ||||||
|  |       that['is' + name] = function(obj) { | ||||||
|  |         return Object.prototype.toString.call(obj) == '[object ' + name + ']'; | ||||||
|  |       }; | ||||||
|  |     }); | ||||||
|  |     that.debounce = function(func, wait, immediate) { | ||||||
|  |       var timeout, args, context, timestamp, result; | ||||||
|  |       return function() { | ||||||
|  |         context = this; | ||||||
|  |         args = arguments; | ||||||
|  |         timestamp = new Date(); | ||||||
|  |         var later = function() { | ||||||
|  |           var last = (new Date()) - timestamp; | ||||||
|  |           if (last < wait) { | ||||||
|  |             timeout = setTimeout(later, wait - last); | ||||||
|  |           } else { | ||||||
|  |             timeout = null; | ||||||
|  |             if (!immediate) result = func.apply(context, args); | ||||||
|  |           } | ||||||
|  |         }; | ||||||
|  |         var callNow = immediate && !timeout; | ||||||
|  |         if (!timeout) { | ||||||
|  |           timeout = setTimeout(later, wait); | ||||||
|  |         } | ||||||
|  |         if (callNow) result = func.apply(context, args); | ||||||
|  |         return result; | ||||||
|  |       }; | ||||||
|  |     }; | ||||||
|  |     return that; | ||||||
|  |   })(); | ||||||
|  | })(); | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user