1
0
mirror of https://github.com/DataTables/DataTables.git synced 2025-01-19 12:52:11 +01:00

New - api methods: page(), page.info(), page.len(), draw()

- Paging control methods for the new API:
  - page() / page(n) - Get / set the current page
  - page.info() - Get information about the table's paging state
  - page.len() / page.len(n) - Get / set the page length

- Rewrite of core.page.js and core.length.js to be more space efficient.
  The functionality is identical to before, but now compresses much
  better (796 byte saving). The new paging API methods add only 614
  bytes (compressed), so overall a saving of 182 bytes, with the new
  functionality added by the new API.

- Start of draw methods for new API:
  - draw() - Draw the table. Need this to test the new paging methods
    since page() etc do not do a redraw themselves, you must call draw()
    when you are ready for the table to be redrawn now.
This commit is contained in:
Allan Jardine 2013-04-18 07:49:38 +01:00
parent fa7122392d
commit 15588d9e41
8 changed files with 619 additions and 354 deletions

View File

@ -2779,122 +2779,109 @@
} }
function _fnLengthChange ( settings, val )
{
var
start = settings._iDisplayStart,
records = settings.fnRecordsDisplay(),
end,
len = parseInt( val, 10 );
/* Redraw the table */
settings._iDisplayLength = len;
_fnCalculateEnd( settings );
end = settings.fnDisplayEnd();
/* If we have space to show extra rows (backing up from the end point - then do so */
if ( end === records )
{
start = end - len;
}
if ( len === -1 || start < 0 )
{
start = 0;
}
settings._iDisplayStart = start;
// Fire length change event
$(settings.oInstance).trigger( 'length', [settings, len] );
}
/** /**
* Generate the node required for user display length changing * Generate the node required for user display length changing
* @param {object} oSettings dataTables settings object * @param {object} settings dataTables settings object
* @returns {node} Display length feature node * @returns {node} Display length feature node
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnFeatureHtmlLength ( oSettings ) function _fnFeatureHtmlLength ( settings )
{
if ( oSettings.oScroll.bInfinite )
{ {
if ( settings.oScroll.bInfinite ) {
return null; return null;
} }
/* This can be overruled by not using the _MENU_ var/macro in the language variable */ var
var sName = 'name="'+oSettings.sTableId+'_length"'; tableId = settings.sTableId,
var sStdMenu = '<select size="1" '+sName+'>'; menu = settings.aLengthMenu,
var i, iLen; d2 = $.isArray( menu[0] ),
var aLengthMenu = oSettings.aLengthMenu; lengths = d2 ? menu[0] : menu,
language = d2 ? menu[1] : menu;
if ( aLengthMenu.length == 2 && typeof aLengthMenu[0] === 'object' && var select = $('<select/>', {
typeof aLengthMenu[1] === 'object' ) 'name': tableId+'_length',
{ 'aria-controls': tableId
for ( i=0, iLen=aLengthMenu[0].length ; i<iLen ; i++ )
{
sStdMenu += '<option value="'+aLengthMenu[0][i]+'">'+aLengthMenu[1][i]+'</option>';
}
}
else
{
for ( i=0, iLen=aLengthMenu.length ; i<iLen ; i++ )
{
sStdMenu += '<option value="'+aLengthMenu[i]+'">'+aLengthMenu[i]+'</option>';
}
}
sStdMenu += '</select>';
var nLength = document.createElement( 'div' );
if ( !oSettings.aanFeatures.l )
{
nLength.id = oSettings.sTableId+'_length';
}
nLength.className = oSettings.oClasses.sLength;
nLength.innerHTML = '<label>'+oSettings.oLanguage.sLengthMenu.replace( '_MENU_', sStdMenu )+'</label>';
/* Set the length to the current display length */
$('select', nLength).val( oSettings._iDisplayLength );
$('select', nLength).bind( 'change.DT', function(e) {
var iVal = $(this).val();
/* Update all other length options for the new display */
var n = oSettings.aanFeatures.l;
for ( i=0, iLen=n.length ; i<iLen ; i++ )
{
if ( n[i] != this.parentNode )
{
$('select', n[i]).val( iVal );
}
}
/* Redraw the table */
oSettings._iDisplayLength = parseInt(iVal, 10);
_fnCalculateEnd( oSettings );
/* If we have space to show extra rows (backing up from the end point - then do so */
if ( oSettings.fnDisplayEnd() == oSettings.fnRecordsDisplay() )
{
oSettings._iDisplayStart = oSettings.fnDisplayEnd() - oSettings._iDisplayLength;
if ( oSettings._iDisplayStart < 0 )
{
oSettings._iDisplayStart = 0;
}
}
if ( oSettings._iDisplayLength == -1 )
{
oSettings._iDisplayStart = 0;
}
_fnDraw( oSettings );
} ); } );
for ( var i=0, ien=lengths.length ; i<ien ; i++ ) {
select[0][ i ] = new Option( language[i], lengths[i] );
}
$('select', nLength).attr('aria-controls', oSettings.sTableId); var div = $('<div><label/></div>').addClass( settings.oClasses.sLength );
if ( ! settings.aanFeatures.l ) {
div[0].id = tableId+'_length';
}
return nLength; // This split doesn't matter where _MENU_ is, we get three items back from it
var a = settings.oLanguage.sLengthMenu.split(/(_MENU_)/);
div.children()
.append( a[0] )
.append( select )
.append( a[2] );
select
.val( settings._iDisplayLength )
.bind( 'change.DT', function(e) {
_fnLengthChange( settings, $(this).val() );
_fnDraw( settings );
} );
// Update node value whenever anything changes the table's length
$(settings.nTable).bind( 'length', function (e, s, len) {
select.val( len );
} );
return div[0];
} }
/** /**
* Recalculate the end point based on the start point * Recalculate the end point based on the start point
* @param {object} oSettings dataTables settings object * @param {object} settings dataTables settings object
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnCalculateEnd( oSettings ) function _fnCalculateEnd( settings )
{ {
if ( oSettings.oFeatures.bPaginate === false ) var
{ len = settings._iDisplayLength,
oSettings._iDisplayEnd = oSettings.aiDisplay.length; calc = settings._iDisplayStart + len,
} records = settings.aiDisplay.length;
else
{ settings._iDisplayEnd = ! settings.oFeatures.bPaginate || calc>records || len===-1 ?
/* Set the end point of the display - based on how many elements there are records :
* still to display calc;
*/
if ( oSettings._iDisplayStart + oSettings._iDisplayLength > oSettings.aiDisplay.length ||
oSettings._iDisplayLength == -1 )
{
oSettings._iDisplayEnd = oSettings.aiDisplay.length;
}
else
{
oSettings._iDisplayEnd = oSettings._iDisplayStart + oSettings._iDisplayLength;
}
}
} }
@ -2910,110 +2897,104 @@
* @returns {node} Pagination feature node * @returns {node} Pagination feature node
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnFeatureHtmlPaginate ( oSettings ) function _fnFeatureHtmlPaginate ( settings )
{ {
if ( oSettings.oScroll.bInfinite ) if ( settings.oScroll.bInfinite )
{ {
return null; return null;
} }
var nPaginate = document.createElement( 'div' ); var
nPaginate.className = oSettings.oClasses.sPaging+oSettings.sPaginationType; type = settings.sPaginationType,
plugin = DataTable.ext.oPagination[ type ],
redraw = function( settings ) {
_fnCalculateEnd( settings );
_fnDraw( settings );
},
node = $('<div/>').addClass( settings.oClasses.sPaging + type )[0];
DataTable.ext.oPagination[ oSettings.sPaginationType ].fnInit( oSettings, nPaginate, plugin.fnInit( settings, node, redraw );
function( oSettings ) {
_fnCalculateEnd( oSettings );
_fnDraw( oSettings );
}
);
/* Add a draw callback for the pagination on first instance, to update the paging display */ /* Add a draw callback for the pagination on first instance, to update the paging display */
if ( !oSettings.aanFeatures.p ) if ( ! settings.aanFeatures.p )
{ {
oSettings.aoDrawCallback.push( { settings.aoDrawCallback.push( {
"fn": function( oSettings ) { "fn": function( settings ) {
DataTable.ext.oPagination[ oSettings.sPaginationType ].fnUpdate( oSettings, function( oSettings ) { plugin.fnUpdate( settings, redraw );
_fnCalculateEnd( oSettings );
_fnDraw( oSettings );
} );
}, },
"sName": "pagination" "sName": "pagination"
} ); } );
} }
return nPaginate;
return node;
} }
/** /**
* Alter the display settings to change the page * Alter the display settings to change the page
* @param {object} oSettings dataTables settings object * @param {object} settings DataTables settings object
* @param {string|int} mAction Paging action to take: "first", "previous", "next" or "last" * @param {string|int} action Paging action to take: "first", "previous",
* or page number to jump to (integer) * "next" or "last" or page number to jump to (integer)
* @returns {bool} true page has changed, false - no change (no effect) eg 'first' on page 1 * @returns {bool} true page has changed, false - no change
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnPageChange ( oSettings, mAction ) function _fnPageChange ( settings, action )
{ {
var iOldStart = oSettings._iDisplayStart; var
start = settings._iDisplayStart,
len = settings._iDisplayLength,
records = settings.fnRecordsDisplay();
if ( typeof mAction === "number" ) if ( records === 0 || len === -1 )
{ {
oSettings._iDisplayStart = mAction * oSettings._iDisplayLength; start = 0;
if ( oSettings._iDisplayStart > oSettings.fnRecordsDisplay() ) }
else if ( typeof action === "number" )
{ {
oSettings._iDisplayStart = 0; start = action * len;
if ( start > records )
{
start = 0;
} }
} }
else if ( mAction == "first" ) else if ( action == "first" )
{ {
oSettings._iDisplayStart = 0; start = 0;
} }
else if ( mAction == "previous" ) else if ( action == "previous" )
{ {
oSettings._iDisplayStart = oSettings._iDisplayLength>=0 ? start = len >= 0 ?
oSettings._iDisplayStart - oSettings._iDisplayLength : start - len :
0; 0;
/* Correct for under-run */ if ( start < 0 )
if ( oSettings._iDisplayStart < 0 )
{ {
oSettings._iDisplayStart = 0; start = 0;
} }
} }
else if ( mAction == "next" ) else if ( action == "next" )
{ {
if ( oSettings._iDisplayLength >= 0 ) if ( start + len < records )
{ {
/* Make sure we are not over running the display array */ start += len;
if ( oSettings._iDisplayStart + oSettings._iDisplayLength < oSettings.fnRecordsDisplay() )
{
oSettings._iDisplayStart += oSettings._iDisplayLength;
} }
} }
else if ( action == "last" )
{
start = Math.floor( (records-1) / len) * len;
}
else else
{ {
oSettings._iDisplayStart = 0; _fnLog( settings, 0, "Unknown paging action: "+action );
} }
}
else if ( mAction == "last" )
{
if ( oSettings._iDisplayLength >= 0 )
{
var iPages = parseInt( (oSettings.fnRecordsDisplay()-1) / oSettings._iDisplayLength, 10 ) + 1;
oSettings._iDisplayStart = (iPages-1) * oSettings._iDisplayLength;
}
else
{
oSettings._iDisplayStart = 0;
}
}
else
{
_fnLog( oSettings, 0, "Unknown paging action: "+mAction );
}
$(oSettings.oInstance).trigger('page', oSettings);
return iOldStart != oSettings._iDisplayStart; var changed = settings._iDisplayStart === start;
settings._iDisplayStart = start;
$(settings.oInstance).trigger('page', settings);
return changed;
} }
@ -6787,7 +6768,7 @@
* @type object * @type object
* @ignore * @ignore
*/ */
var _api; var _Api;
/** /**
@ -6946,12 +6927,12 @@
* // Initialisation as a constructor * // Initialisation as a constructor
* var api = new $.fn.DataTable.Api( 'table.dataTable' ); * var api = new $.fn.DataTable.Api( 'table.dataTable' );
*/ */
DataTable.Api = _api = function ( context, data ) DataTable.Api = _Api = function ( context, data )
{ {
if ( ! this instanceof _api ) { if ( ! this instanceof _Api ) {
throw 'DT API must be constructed as a new object'; throw 'DT API must be constructed as a new object';
// or should it do the 'new' for the caller // or should it do the 'new' for the caller
// return new _api.apply( this, arguments ); // return new _Api.apply( this, arguments );
} }
var settings = []; var settings = [];
@ -6979,11 +6960,11 @@
this.push.apply( this, data ); this.push.apply( this, data );
} }
_api.extend( this, this, _apiStruct ); _Api.extend( this, this, _apiStruct );
}; };
_api.prototype = /** @lends DataTables.Api */{ _Api.prototype = /** @lends DataTables.Api */{
/** /**
* Return a new Api instance, comprised of the data held in the current * Return a new Api instance, comprised of the data held in the current
* instance, join with the other array(s) and/or value(s). * instance, join with the other array(s) and/or value(s).
@ -7036,7 +7017,7 @@
} }
} }
return new _api( this.context, a ); return new _Api( this.context, a );
}, },
@ -7075,7 +7056,7 @@
} }
} }
return new _api( this.context, a ); return new _Api( this.context, a );
}, },
@ -7165,7 +7146,7 @@
unique: function () unique: function ()
{ {
return new _api( this.context, _unique(this) ); return new _Api( this.context, _unique(this) );
}, },
@ -7175,9 +7156,9 @@
_api.extend = function ( scope, obj, ext ) _Api.extend = function ( scope, obj, ext )
{ {
if ( ! obj instanceof _api ) { if ( ! obj instanceof _Api ) {
return; return;
} }
@ -7190,7 +7171,7 @@
var ret = fn.apply( scope, arguments ); var ret = fn.apply( scope, arguments );
// Method extension // Method extension
_api.extend( ret, ret, struc.methodExt ); _Api.extend( ret, ret, struc.methodExt );
return ret; return ret;
}; };
}; };
@ -7207,12 +7188,12 @@
} }
// Property extension // Property extension
_api.extend( scope, obj[ struct.name ], struct.propExt ); _Api.extend( scope, obj[ struct.name ], struct.propExt );
} }
}; };
_api.register = function ( name, val ) _Api.register = function ( name, val )
{ {
var var
i, ien, i, ien,
@ -7257,7 +7238,7 @@
} }
// Rebuild the API with the new construct // Rebuild the API with the new construct
if ( _api.ready ) { if ( _Api.ready ) {
DataTable.api.build(); DataTable.api.build();
} }
}; };
@ -7269,7 +7250,7 @@
(/** @lends <global> */function() { (/** @lends <global> */function() {
var _api = DataTable.Api; var _Api = DataTable.Api;
/** /**
* Selector for HTML tables. Apply the given selector to the give array of * Selector for HTML tables. Apply the given selector to the give array of
@ -7322,7 +7303,7 @@
* information, if returned, is assigned to the API instance. Otherwise the * information, if returned, is assigned to the API instance. Otherwise the
* original API instance is returned for chaining. * original API instance is returned for chaining.
*/ */
_api.register( 'tables()', function ( selector, fn ) { _Api.register( 'tables()', function ( selector, fn ) {
// Argument shifting // Argument shifting
if ( typeof selector === 'function' ) { if ( typeof selector === 'function' ) {
fn = selector; fn = selector;
@ -7346,7 +7327,7 @@
// A new instance is created if there was a selector specified, or if // A new instance is created if there was a selector specified, or if
// data was returned from the callback // data was returned from the callback
var api = selector || a.length ? var api = selector || a.length ?
new _api( context, a ) : new _Api( context, a ) :
this; this;
return api; return api;
@ -7358,13 +7339,154 @@
* @return {DataTable.Api} New Api instance containing the DOM nodes for the * @return {DataTable.Api} New Api instance containing the DOM nodes for the
* tables. * tables.
*/ */
_api.register( 'tables().nodes()', function () { _Api.register( 'tables().nodes()', function () {
return this.tables( function ( settings, i ) { return this.tables( function ( settings, i ) {
return settings.nTable; return settings.nTable;
} ); } );
} ); } );
}());
(/** @lends <global> */function() {
var _api = DataTable.Api;
// draw()
// draw.standing()
/**
*
*/
_api.register( 'draw()', function ( full ) {
return this.tables( function ( settings ) {
if ( full ) {
_fnReDraw( settings );
}
else {
_fnCalculateEnd( settings );
_fnDraw( settings );
}
} );
} );
}());
(/** @lends <global> */function() {
var _Api = DataTable.Api;
/**
* Get the current page index.
*
* @return {integer} Current page index (zero based)
*//**
* Set the current page.
*
* Note that if you attempt to show a page which does not exist, DataTables will
* not throw an error, but rather reset the paging.
*
* @param {integer|string} action The paging action to take. This can be one of:
* * `integer` - The page index to jump to
* * `string` - An action to take:
* * `first` - Jump to first page.
* * `next` - Jump to the next page
* * `previous` - Jump to previous page
* * `last` - Jump to the last page.
* @returns {DataTables.Api} this
*/
_Api.register( 'page()', function ( action ) {
if ( action === undefined ) {
return this.page.info().page; // not an expensive call
}
// else, have an action to take on all tables
return this.tables( function ( settings ) {
_fnPageChange( settings, action );
_fnCalculateEnd( settings );
} );
} );
/**
* Paging information for the first table in the current context.
*
* If you require paging information for another table, use the `table()` method
* with a suitable selector.
*
* @return {object} Object with the following properties set:
* * `page` - Current page index (zero based - i.e. the first page is `0`)
* * `pages` - Total number of pages
* * `start` - Display index for the first record shown on the current page
* * `end` - Display index for the last record shown on the current page
* * `length` - Display length (number of records). Note that generally `start
* + length = end`, but this is not always true, for example if there are
* only 2 records to show on the final page, with a length of 10.
* * `recordsTotal` - Full data set length
* * `recordsDisplay` - Data set length once the current filtering criterion
* are applied.
*/
_Api.register( 'page.info()', function ( action ) {
if ( this.context.length === 0 ) {
return undefined;
}
var
settings = this.context[0],
start = settings._iDisplayStart,
len = settings._iDisplayLength,
visRecords = settings.fnRecordsDisplay(),
all = len === -1;
return {
"page": all ? 0 : Math.ceil( start / len ),
"pages": all ? 1 : Math.ceil( visRecords / len ),
"start": start,
"end": settings.fnDisplayEnd(),
"length": len,
"recordsTotal": settings.fnRecordsTotal(),
"recordsDisplay": visRecords
};
} );
/**
* Get the current page length.
*
* @return {integer} Current page length. Note `-1` indicates that all records
* are to be shown.
*//**
* Set the current page length.
*
* @param {integer} Page length to set. Use `-1` to show all records.
* @returns {DataTables.Api} this
*/
_Api.register( 'page.len()', function ( len ) {
// Note that we can't call this function 'length()' because `length`
// is a Javascript property of functions which defines how many arguments
// the function expects.
if ( len === undefined ) {
return this.context.length !== 0 ?
this.context[0]._iDisplayLength :
undefined;
}
// else, set the page length
return this.tables( function ( settings ) {
_fnLengthChange( settings, len );
_fnCalculateEnd( settings );
} );
} );
}()); }());
/** /**
@ -12897,6 +13019,16 @@
* @param {event} e jQuery event object * @param {event} e jQuery event object
* @param {object} o DataTables settings object {@link DataTable.models.oSettings} * @param {object} o DataTables settings object {@link DataTable.models.oSettings}
*/ */
/**
* Page length change event, fired when number of records to show on each
* page (the length) is changed.
* @name DataTable#length
* @event
* @param {event} e jQuery event object
* @param {object} o DataTables settings object {@link DataTable.models.oSettings}
* @param {integer} len New length
*/
})); }));
}(window, document)); }(window, document));

View File

@ -110,6 +110,8 @@
require('api.core.js'); require('api.core.js');
require('api.table.js'); require('api.table.js');
require('api.draw.js');
require('api.page.js');
require('api.static.js'); require('api.static.js');
/** /**
@ -309,6 +311,16 @@
* @param {event} e jQuery event object * @param {event} e jQuery event object
* @param {object} o DataTables settings object {@link DataTable.models.oSettings} * @param {object} o DataTables settings object {@link DataTable.models.oSettings}
*/ */
/**
* Page length change event, fired when number of records to show on each
* page (the length) is changed.
* @name DataTable#length
* @event
* @param {event} e jQuery event object
* @param {object} o DataTables settings object {@link DataTable.models.oSettings}
* @param {integer} len New length
*/
})); }));
}(window, document)); }(window, document));

View File

@ -48,7 +48,7 @@ var _apiStruct = [];
* @type object * @type object
* @ignore * @ignore
*/ */
var _api; var _Api;
/** /**
@ -207,12 +207,12 @@ var _unique = function ( src )
* // Initialisation as a constructor * // Initialisation as a constructor
* var api = new $.fn.DataTable.Api( 'table.dataTable' ); * var api = new $.fn.DataTable.Api( 'table.dataTable' );
*/ */
DataTable.Api = _api = function ( context, data ) DataTable.Api = _Api = function ( context, data )
{ {
if ( ! this instanceof _api ) { if ( ! this instanceof _Api ) {
throw 'DT API must be constructed as a new object'; throw 'DT API must be constructed as a new object';
// or should it do the 'new' for the caller // or should it do the 'new' for the caller
// return new _api.apply( this, arguments ); // return new _Api.apply( this, arguments );
} }
var settings = []; var settings = [];
@ -240,11 +240,11 @@ DataTable.Api = _api = function ( context, data )
this.push.apply( this, data ); this.push.apply( this, data );
} }
_api.extend( this, this, _apiStruct ); _Api.extend( this, this, _apiStruct );
}; };
_api.prototype = /** @lends DataTables.Api */{ _Api.prototype = /** @lends DataTables.Api */{
/** /**
* Return a new Api instance, comprised of the data held in the current * Return a new Api instance, comprised of the data held in the current
* instance, join with the other array(s) and/or value(s). * instance, join with the other array(s) and/or value(s).
@ -297,7 +297,7 @@ _api.prototype = /** @lends DataTables.Api */{
} }
} }
return new _api( this.context, a ); return new _Api( this.context, a );
}, },
@ -336,7 +336,7 @@ _api.prototype = /** @lends DataTables.Api */{
} }
} }
return new _api( this.context, a ); return new _Api( this.context, a );
}, },
@ -426,7 +426,7 @@ _api.prototype = /** @lends DataTables.Api */{
unique: function () unique: function ()
{ {
return new _api( this.context, _unique(this) ); return new _Api( this.context, _unique(this) );
}, },
@ -436,9 +436,9 @@ _api.prototype = /** @lends DataTables.Api */{
_api.extend = function ( scope, obj, ext ) _Api.extend = function ( scope, obj, ext )
{ {
if ( ! obj instanceof _api ) { if ( ! obj instanceof _Api ) {
return; return;
} }
@ -451,7 +451,7 @@ _api.prototype = /** @lends DataTables.Api */{
var ret = fn.apply( scope, arguments ); var ret = fn.apply( scope, arguments );
// Method extension // Method extension
_api.extend( ret, ret, struc.methodExt ); _Api.extend( ret, ret, struc.methodExt );
return ret; return ret;
}; };
}; };
@ -468,12 +468,12 @@ _api.prototype = /** @lends DataTables.Api */{
} }
// Property extension // Property extension
_api.extend( scope, obj[ struct.name ], struct.propExt ); _Api.extend( scope, obj[ struct.name ], struct.propExt );
} }
}; };
_api.register = function ( name, val ) _Api.register = function ( name, val )
{ {
var var
i, ien, i, ien,
@ -518,7 +518,7 @@ _api.register = function ( name, val )
} }
// Rebuild the API with the new construct // Rebuild the API with the new construct
if ( _api.ready ) { if ( _Api.ready ) {
DataTable.api.build(); DataTable.api.build();
} }
}; };

28
media/src/api/api.draw.js Normal file
View File

@ -0,0 +1,28 @@
(/** @lends <global> */function() {
var _api = DataTable.Api;
// draw()
// draw.standing()
/**
*
*/
_api.register( 'draw()', function ( full ) {
return this.tables( function ( settings ) {
if ( full ) {
_fnReDraw( settings );
}
else {
_fnCalculateEnd( settings );
_fnDraw( settings );
}
} );
} );
}());

112
media/src/api/api.page.js Normal file
View File

@ -0,0 +1,112 @@
(/** @lends <global> */function() {
var _Api = DataTable.Api;
/**
* Get the current page index.
*
* @return {integer} Current page index (zero based)
*//**
* Set the current page.
*
* Note that if you attempt to show a page which does not exist, DataTables will
* not throw an error, but rather reset the paging.
*
* @param {integer|string} action The paging action to take. This can be one of:
* * `integer` - The page index to jump to
* * `string` - An action to take:
* * `first` - Jump to first page.
* * `next` - Jump to the next page
* * `previous` - Jump to previous page
* * `last` - Jump to the last page.
* @returns {DataTables.Api} this
*/
_Api.register( 'page()', function ( action ) {
if ( action === undefined ) {
return this.page.info().page; // not an expensive call
}
// else, have an action to take on all tables
return this.tables( function ( settings ) {
_fnPageChange( settings, action );
_fnCalculateEnd( settings );
} );
} );
/**
* Paging information for the first table in the current context.
*
* If you require paging information for another table, use the `table()` method
* with a suitable selector.
*
* @return {object} Object with the following properties set:
* * `page` - Current page index (zero based - i.e. the first page is `0`)
* * `pages` - Total number of pages
* * `start` - Display index for the first record shown on the current page
* * `end` - Display index for the last record shown on the current page
* * `length` - Display length (number of records). Note that generally `start
* + length = end`, but this is not always true, for example if there are
* only 2 records to show on the final page, with a length of 10.
* * `recordsTotal` - Full data set length
* * `recordsDisplay` - Data set length once the current filtering criterion
* are applied.
*/
_Api.register( 'page.info()', function ( action ) {
if ( this.context.length === 0 ) {
return undefined;
}
var
settings = this.context[0],
start = settings._iDisplayStart,
len = settings._iDisplayLength,
visRecords = settings.fnRecordsDisplay(),
all = len === -1;
return {
"page": all ? 0 : Math.ceil( start / len ),
"pages": all ? 1 : Math.ceil( visRecords / len ),
"start": start,
"end": settings.fnDisplayEnd(),
"length": len,
"recordsTotal": settings.fnRecordsTotal(),
"recordsDisplay": visRecords
};
} );
/**
* Get the current page length.
*
* @return {integer} Current page length. Note `-1` indicates that all records
* are to be shown.
*//**
* Set the current page length.
*
* @param {integer} Page length to set. Use `-1` to show all records.
* @returns {DataTables.Api} this
*/
_Api.register( 'page.len()', function ( len ) {
// Note that we can't call this function 'length()' because `length`
// is a Javascript property of functions which defines how many arguments
// the function expects.
if ( len === undefined ) {
return this.context.length !== 0 ?
this.context[0]._iDisplayLength :
undefined;
}
// else, set the page length
return this.tables( function ( settings ) {
_fnLengthChange( settings, len );
_fnCalculateEnd( settings );
} );
} );
}());

View File

@ -2,7 +2,7 @@
(/** @lends <global> */function() { (/** @lends <global> */function() {
var _api = DataTable.Api; var _Api = DataTable.Api;
/** /**
* Selector for HTML tables. Apply the given selector to the give array of * Selector for HTML tables. Apply the given selector to the give array of
@ -55,7 +55,7 @@ var _table_selector = function ( selector, a )
* information, if returned, is assigned to the API instance. Otherwise the * information, if returned, is assigned to the API instance. Otherwise the
* original API instance is returned for chaining. * original API instance is returned for chaining.
*/ */
_api.register( 'tables()', function ( selector, fn ) { _Api.register( 'tables()', function ( selector, fn ) {
// Argument shifting // Argument shifting
if ( typeof selector === 'function' ) { if ( typeof selector === 'function' ) {
fn = selector; fn = selector;
@ -79,7 +79,7 @@ _api.register( 'tables()', function ( selector, fn ) {
// A new instance is created if there was a selector specified, or if // A new instance is created if there was a selector specified, or if
// data was returned from the callback // data was returned from the callback
var api = selector || a.length ? var api = selector || a.length ?
new _api( context, a ) : new _Api( context, a ) :
this; this;
return api; return api;
@ -91,7 +91,7 @@ _api.register( 'tables()', function ( selector, fn ) {
* @return {DataTable.Api} New Api instance containing the DOM nodes for the * @return {DataTable.Api} New Api instance containing the DOM nodes for the
* tables. * tables.
*/ */
_api.register( 'tables().nodes()', function () { _Api.register( 'tables().nodes()', function () {
return this.tables( function ( settings, i ) { return this.tables( function ( settings, i ) {
return settings.nTable; return settings.nTable;
} ); } );

View File

@ -1,119 +1,106 @@
function _fnLengthChange ( settings, val )
{
var
start = settings._iDisplayStart,
records = settings.fnRecordsDisplay(),
end,
len = parseInt( val, 10 );
/* Redraw the table */
settings._iDisplayLength = len;
_fnCalculateEnd( settings );
end = settings.fnDisplayEnd();
/* If we have space to show extra rows (backing up from the end point - then do so */
if ( end === records )
{
start = end - len;
}
if ( len === -1 || start < 0 )
{
start = 0;
}
settings._iDisplayStart = start;
// Fire length change event
$(settings.oInstance).trigger( 'length', [settings, len] );
}
/** /**
* Generate the node required for user display length changing * Generate the node required for user display length changing
* @param {object} oSettings dataTables settings object * @param {object} settings dataTables settings object
* @returns {node} Display length feature node * @returns {node} Display length feature node
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnFeatureHtmlLength ( oSettings ) function _fnFeatureHtmlLength ( settings )
{
if ( oSettings.oScroll.bInfinite )
{ {
if ( settings.oScroll.bInfinite ) {
return null; return null;
} }
/* This can be overruled by not using the _MENU_ var/macro in the language variable */ var
var sName = 'name="'+oSettings.sTableId+'_length"'; tableId = settings.sTableId,
var sStdMenu = '<select size="1" '+sName+'>'; menu = settings.aLengthMenu,
var i, iLen; d2 = $.isArray( menu[0] ),
var aLengthMenu = oSettings.aLengthMenu; lengths = d2 ? menu[0] : menu,
language = d2 ? menu[1] : menu;
if ( aLengthMenu.length == 2 && typeof aLengthMenu[0] === 'object' && var select = $('<select/>', {
typeof aLengthMenu[1] === 'object' ) 'name': tableId+'_length',
{ 'aria-controls': tableId
for ( i=0, iLen=aLengthMenu[0].length ; i<iLen ; i++ )
{
sStdMenu += '<option value="'+aLengthMenu[0][i]+'">'+aLengthMenu[1][i]+'</option>';
}
}
else
{
for ( i=0, iLen=aLengthMenu.length ; i<iLen ; i++ )
{
sStdMenu += '<option value="'+aLengthMenu[i]+'">'+aLengthMenu[i]+'</option>';
}
}
sStdMenu += '</select>';
var nLength = document.createElement( 'div' );
if ( !oSettings.aanFeatures.l )
{
nLength.id = oSettings.sTableId+'_length';
}
nLength.className = oSettings.oClasses.sLength;
nLength.innerHTML = '<label>'+oSettings.oLanguage.sLengthMenu.replace( '_MENU_', sStdMenu )+'</label>';
/* Set the length to the current display length */
$('select', nLength).val( oSettings._iDisplayLength );
$('select', nLength).bind( 'change.DT', function(e) {
var iVal = $(this).val();
/* Update all other length options for the new display */
var n = oSettings.aanFeatures.l;
for ( i=0, iLen=n.length ; i<iLen ; i++ )
{
if ( n[i] != this.parentNode )
{
$('select', n[i]).val( iVal );
}
}
/* Redraw the table */
oSettings._iDisplayLength = parseInt(iVal, 10);
_fnCalculateEnd( oSettings );
/* If we have space to show extra rows (backing up from the end point - then do so */
if ( oSettings.fnDisplayEnd() == oSettings.fnRecordsDisplay() )
{
oSettings._iDisplayStart = oSettings.fnDisplayEnd() - oSettings._iDisplayLength;
if ( oSettings._iDisplayStart < 0 )
{
oSettings._iDisplayStart = 0;
}
}
if ( oSettings._iDisplayLength == -1 )
{
oSettings._iDisplayStart = 0;
}
_fnDraw( oSettings );
} ); } );
for ( var i=0, ien=lengths.length ; i<ien ; i++ ) {
select[0][ i ] = new Option( language[i], lengths[i] );
}
$('select', nLength).attr('aria-controls', oSettings.sTableId); var div = $('<div><label/></div>').addClass( settings.oClasses.sLength );
if ( ! settings.aanFeatures.l ) {
div[0].id = tableId+'_length';
}
return nLength; // This split doesn't matter where _MENU_ is, we get three items back from it
var a = settings.oLanguage.sLengthMenu.split(/(_MENU_)/);
div.children()
.append( a[0] )
.append( select )
.append( a[2] );
select
.val( settings._iDisplayLength )
.bind( 'change.DT', function(e) {
_fnLengthChange( settings, $(this).val() );
_fnDraw( settings );
} );
// Update node value whenever anything changes the table's length
$(settings.nTable).bind( 'length', function (e, s, len) {
select.val( len );
} );
return div[0];
} }
/** /**
* Recalculate the end point based on the start point * Recalculate the end point based on the start point
* @param {object} oSettings dataTables settings object * @param {object} settings dataTables settings object
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnCalculateEnd( oSettings ) function _fnCalculateEnd( settings )
{ {
if ( oSettings.oFeatures.bPaginate === false ) var
{ len = settings._iDisplayLength,
oSettings._iDisplayEnd = oSettings.aiDisplay.length; calc = settings._iDisplayStart + len,
} records = settings.aiDisplay.length;
else
{ settings._iDisplayEnd = ! settings.oFeatures.bPaginate || calc>records || len===-1 ?
/* Set the end point of the display - based on how many elements there are records :
* still to display calc;
*/
if ( oSettings._iDisplayStart + oSettings._iDisplayLength > oSettings.aiDisplay.length ||
oSettings._iDisplayLength == -1 )
{
oSettings._iDisplayEnd = oSettings.aiDisplay.length;
}
else
{
oSettings._iDisplayEnd = oSettings._iDisplayStart + oSettings._iDisplayLength;
}
}
} }

View File

@ -11,109 +11,103 @@
* @returns {node} Pagination feature node * @returns {node} Pagination feature node
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnFeatureHtmlPaginate ( oSettings ) function _fnFeatureHtmlPaginate ( settings )
{ {
if ( oSettings.oScroll.bInfinite ) if ( settings.oScroll.bInfinite )
{ {
return null; return null;
} }
var nPaginate = document.createElement( 'div' ); var
nPaginate.className = oSettings.oClasses.sPaging+oSettings.sPaginationType; type = settings.sPaginationType,
plugin = DataTable.ext.oPagination[ type ],
redraw = function( settings ) {
_fnCalculateEnd( settings );
_fnDraw( settings );
},
node = $('<div/>').addClass( settings.oClasses.sPaging + type )[0];
DataTable.ext.oPagination[ oSettings.sPaginationType ].fnInit( oSettings, nPaginate, plugin.fnInit( settings, node, redraw );
function( oSettings ) {
_fnCalculateEnd( oSettings );
_fnDraw( oSettings );
}
);
/* Add a draw callback for the pagination on first instance, to update the paging display */ /* Add a draw callback for the pagination on first instance, to update the paging display */
if ( !oSettings.aanFeatures.p ) if ( ! settings.aanFeatures.p )
{ {
oSettings.aoDrawCallback.push( { settings.aoDrawCallback.push( {
"fn": function( oSettings ) { "fn": function( settings ) {
DataTable.ext.oPagination[ oSettings.sPaginationType ].fnUpdate( oSettings, function( oSettings ) { plugin.fnUpdate( settings, redraw );
_fnCalculateEnd( oSettings );
_fnDraw( oSettings );
} );
}, },
"sName": "pagination" "sName": "pagination"
} ); } );
} }
return nPaginate;
return node;
} }
/** /**
* Alter the display settings to change the page * Alter the display settings to change the page
* @param {object} oSettings dataTables settings object * @param {object} settings DataTables settings object
* @param {string|int} mAction Paging action to take: "first", "previous", "next" or "last" * @param {string|int} action Paging action to take: "first", "previous",
* or page number to jump to (integer) * "next" or "last" or page number to jump to (integer)
* @returns {bool} true page has changed, false - no change (no effect) eg 'first' on page 1 * @returns {bool} true page has changed, false - no change
* @memberof DataTable#oApi * @memberof DataTable#oApi
*/ */
function _fnPageChange ( oSettings, mAction ) function _fnPageChange ( settings, action )
{ {
var iOldStart = oSettings._iDisplayStart; var
start = settings._iDisplayStart,
len = settings._iDisplayLength,
records = settings.fnRecordsDisplay();
if ( typeof mAction === "number" ) if ( records === 0 || len === -1 )
{ {
oSettings._iDisplayStart = mAction * oSettings._iDisplayLength; start = 0;
if ( oSettings._iDisplayStart > oSettings.fnRecordsDisplay() ) }
else if ( typeof action === "number" )
{ {
oSettings._iDisplayStart = 0; start = action * len;
if ( start > records )
{
start = 0;
} }
} }
else if ( mAction == "first" ) else if ( action == "first" )
{ {
oSettings._iDisplayStart = 0; start = 0;
} }
else if ( mAction == "previous" ) else if ( action == "previous" )
{ {
oSettings._iDisplayStart = oSettings._iDisplayLength>=0 ? start = len >= 0 ?
oSettings._iDisplayStart - oSettings._iDisplayLength : start - len :
0; 0;
/* Correct for under-run */ if ( start < 0 )
if ( oSettings._iDisplayStart < 0 )
{ {
oSettings._iDisplayStart = 0; start = 0;
} }
} }
else if ( mAction == "next" ) else if ( action == "next" )
{ {
if ( oSettings._iDisplayLength >= 0 ) if ( start + len < records )
{ {
/* Make sure we are not over running the display array */ start += len;
if ( oSettings._iDisplayStart + oSettings._iDisplayLength < oSettings.fnRecordsDisplay() )
{
oSettings._iDisplayStart += oSettings._iDisplayLength;
} }
} }
else if ( action == "last" )
{
start = Math.floor( (records-1) / len) * len;
}
else else
{ {
oSettings._iDisplayStart = 0; _fnLog( settings, 0, "Unknown paging action: "+action );
} }
}
else if ( mAction == "last" ) var changed = settings._iDisplayStart === start;
{ settings._iDisplayStart = start;
if ( oSettings._iDisplayLength >= 0 )
{ $(settings.oInstance).trigger('page', settings);
var iPages = parseInt( (oSettings.fnRecordsDisplay()-1) / oSettings._iDisplayLength, 10 ) + 1;
oSettings._iDisplayStart = (iPages-1) * oSettings._iDisplayLength; return changed;
}
else
{
oSettings._iDisplayStart = 0;
}
}
else
{
_fnLog( oSettings, 0, "Unknown paging action: "+mAction );
}
$(oSettings.oInstance).trigger('page', oSettings);
return iOldStart != oSettings._iDisplayStart;
} }