uawdijnntqw1x1x1
IP : 216.73.216.110
Hostname : 6.87.74.97.host.secureserver.net
Kernel : Linux 6.87.74.97.host.secureserver.net 4.18.0-553.83.1.el8_10.x86_64 #1 SMP Mon Nov 10 04:22:44 EST 2025 x86_64
Disable Function : None :)
OS : Linux
PATH:
/
home
/
emeraadmin
/
.htpasswds
/
..
/
public_html
/
Trustmark
/
..
/
node_modules
/
..
/
4d695
/
datatables.net-dt.tar
/
/
package.json000064400000003573151676725350007063 0ustar00{ "_from": "datatables.net-dt", "_id": "datatables.net-dt@2.0.8", "_inBundle": false, "_integrity": "sha512-9SG5MWJXq2IQSJWuH+2DvK/9AXduZr0wI/lQbrtBBd18Ck5sO8z3EXxy5wYxxjTFZ9Z+wl0lHsO//qR8QYmWIA==", "_location": "/datatables.net-dt", "_phantomChildren": { "jquery": "3.3.1" }, "_requested": { "type": "tag", "registry": true, "raw": "datatables.net-dt", "name": "datatables.net-dt", "escapedName": "datatables.net-dt", "rawSpec": "", "saveSpec": null, "fetchSpec": "latest" }, "_requiredBy": [ "#USER", "/" ], "_resolved": "https://registry.npmjs.org/datatables.net-dt/-/datatables.net-dt-2.0.8.tgz", "_shasum": "b6be60a022160ec3d38f1ab162a4b1c50642686a", "_spec": "datatables.net-dt", "_where": "C:\\xampp\\htdocs\\emeraltd", "author": { "name": "SpryMedia Ltd", "url": "http://datatables.net" }, "bugs": { "url": "https://datatables.net/forums" }, "bundleDependencies": false, "dependencies": { "datatables.net": "2.0.8", "jquery": ">=1.7" }, "deprecated": false, "description": "DataTables for jQuery with styling for [DataTables](https://datatables.net/)", "files": [ "css/**/*.css", "js/**/*.js", "js/**/*.mjs", "types/**/*.d.ts" ], "homepage": "https://datatables.net", "ignore": [ "composer.json", "datatables.json", "package.json" ], "keywords": [ "Datatables", "jQuery", "table", "filter", "sort" ], "license": "MIT", "main": "js/dataTables.dataTables.js", "module": "js/dataTables.dataTables.mjs", "moduleType": [ "globals", "amd", "node" ], "name": "datatables.net-dt", "repository": { "type": "git", "url": "git+https://github.com/DataTables/Dist-DataTables-DataTables.git" }, "style": "css/dataTables.dataTables.css", "types": "./types/dataTables.dataTables.d.ts", "version": "2.0.8" } node_modules/datatables.net/package.json000064400000003312151676725350014420 0ustar00{ "_from": "datatables.net@2.0.8", "_id": "datatables.net@2.0.8", "_inBundle": false, "_integrity": "sha512-4/2dYx4vl975zQqZbyoVEm0huPe61qffjBRby7K7V+y9E+ORq4R8KavkgrNMmIgO6cl85Pg4AvCbVjvPCIT1Yg==", "_location": "/datatables.net-dt/datatables.net", "_phantomChildren": {}, "_requested": { "type": "version", "registry": true, "raw": "datatables.net@2.0.8", "name": "datatables.net", "escapedName": "datatables.net", "rawSpec": "2.0.8", "saveSpec": null, "fetchSpec": "2.0.8" }, "_requiredBy": [ "/datatables.net-dt" ], "_resolved": "https://registry.npmjs.org/datatables.net/-/datatables.net-2.0.8.tgz", "_shasum": "5199f0f2c26d8bba653af74cea035e9e07166768", "_spec": "datatables.net@2.0.8", "_where": "C:\\xampp\\htdocs\\emeraltd\\node_modules\\datatables.net-dt", "author": { "name": "SpryMedia Ltd", "url": "http://datatables.net" }, "bugs": { "url": "https://datatables.net/forums" }, "bundleDependencies": false, "dependencies": { "jquery": ">=1.7" }, "deprecated": false, "description": "jQuery DataTables", "files": [ "js/**/*.js", "js/**/*.mjs", "types/**/*.d.ts" ], "homepage": "https://datatables.net", "ignore": [ "composer.json", "datatables.json", "package.json" ], "keywords": [ "Datatables", "jQuery", "table", "filter", "sort" ], "license": "MIT", "main": "js/dataTables.js", "module": "js/dataTables.mjs", "moduleType": [ "globals", "amd", "node" ], "name": "datatables.net", "repository": { "type": "git", "url": "git+https://github.com/DataTables/Dist-DataTables.git" }, "types": "./types/types.d.ts", "version": "2.0.8" } node_modules/datatables.net/Readme.md000064400000004736151676725350013664 0ustar00# DataTables for jQuery This package contains distribution files for the [DataTables library](https://datatables.net) for [jQuery](http://jquery.com/). Only the core software for this library is contained in this package - to be correctly styled, a styling package for DataTables must also be included. Styling options include DataTable's native styling, [Bootstrap](http://getbootstrap.com) and [Foundation](http://foundation.zurb.com/). DataTables is a table enhancing library which adds features such as paging, ordering, search, scrolling and many more to a static HTML page. A comprehensive API is also available that can be used to manipulate the table. Please refer to the [DataTables web-site](//datatables.net) for a full range of documentation and examples. ## Installation ### Browser For inclusion of this library using a standard `<script>` tag, rather than using this package, it is recommended that you use the [DataTables download builder](//datatables.net/download) which can create CDN or locally hosted packages \for you, will all dependencies satisfied. ### npm ``` npm install datatables.net ``` ES3 Syntax ``` var $ = require( 'jquery' ); require( 'datatables.net' )( window, $ ); ``` ES6 Syntax ``` import 'datatables.net' ``` ### bower ``` bower install --save datatables.net ``` ## Documentation Full documentation of the DataTables options, API and plug-in interface are available on the [website](https://datatables.net/reference/index). The site also contains information on the wide variety of plug-ins that are available for DataTables, which can be used to enhance and customise your table even further. ## Bug / Support Support for DataTables is available through the [DataTables forums](//datatables.net/forums) and [commercial support options](//datatables.net/support) are available. ### Contributing If you are thinking of contributing code to DataTables, first of all, thank you! All fixes, patches and enhancements to DataTables are very warmly welcomed. This repository is a distribution repo, so patches and issues sent to this repo will not be accepted. Instead, please direct pull requests to the [DataTables/DataTablesSrc](http://github.com/DataTables/DataTablesSrc). For issues / bugs, please direct your questions to the [DataTables forums](//datatables.net/forums). ## License This software is released under the [MIT license](//datatables.net/license). You are free to use, modify and distribute this software, but all copyright information must remain. node_modules/datatables.net/License.txt000064400000002140151676725350014253 0ustar00The MIT License (MIT) Copyright SpryMedia Limited and other contributors http://datatables.net Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. 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. node_modules/datatables.net/types/types.d.ts000064400000301625151676725350015244 0ustar00// Type definitions for DataTables // // Project: https://datatables.net // Definitions by: // SpryMedia // Kiarash Ghiaseddin <https://github.com/Silver-Connection> // Omid Rad <https://github.com/omidkrad> // Armin Sander <https://github.com/pragmatrix> // Craig Boland <https://github.com/CNBoland> /// <reference types="jquery" /> /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Types */ /** * DataTable's internal settings object. DO NOT USE! The parameters * in this object are considered private. They can, do, and will * change! Always use the API methods for manipulating the table. */ export type InternalSettings = {[key: string]: any}; export type DomSelector = string | Node | JQuery; export type InstSelector = DomSelector | InternalSettings; export type RowIdx = number; export type RowSelector<T> = RowIdx | string | Node | JQuery | ((idx: RowIdx, data: T, node: Node | null) => boolean) | RowSelector<T>[]; export type ColumnIdx = number; export type ColumnSelector = ColumnIdx | string | Node | JQuery | ((idx:ColumnIdx, data: any, node: Node) => boolean) | ColumnSelector[]; export type CellIdx = { row: number; column: number; }; export type CellSelector = CellIdx | string | Node | JQuery | ((idx: CellIdx, data: any, node: Node | null) => boolean) | CellSelector[]; export type CellIdxWithVisible = { row: number; column: number; columnVisible: number; } export type SearchInput<T> = string | RegExp | ((data: string, rowData: T) => boolean); export type SearchInputColumn<T> = string | RegExp | ((data: string, rowData: T, column: number) => boolean); export type HeaderStructure = { cell: HTMLElement; colspan: number; rowspan: number; title: string; } /** * DataTables search options. * * @see https://datatables.net/reference/type/DataTables.SearchOptions */ export interface SearchOptions { /** Match from the start of words (ASCII) */ boundary?: boolean, /** Case insensitive search */ caseInsensitive?: boolean, /** Exact matching */ exact?: boolean, /** Treat the input as regex (true) or not (false) */ regex?: boolean, /** Use DataTables smart search */ smart?: boolean } export interface OrderIdx { idx: number; dir: 'asc' | 'desc'; } export interface OrderName { name: string; dir: 'asc' | 'desc'; } export type OrderArray = [number, 'asc' | 'desc']; export type OrderCombined = OrderIdx | OrderName | OrderArray; export type Order = OrderCombined | OrderCombined[]; export interface DataType { className?: string; detect?: ((data: any) => string); order?: { pre?: ((a: any, b: any) => number); asc?: ((a: any, b: any) => number); desc?: ((a: any, b: any) => number); } render?: ((data: any, type: string, row: any) => string | number | HTMLElement); search?: ((data: any) => string); } export interface Feature { /** Table information display */ info?: { /** Information display callback */ callback?: (settings: InternalSettings, start: number, end: number, max: number, total: number, pre: string) => string; /** Empty table text */ empty?: string; /** Information string postfix */ postfix?: string; /** Appended to the info string when searching is active */ search?: string; /** Table summary information display string */ text?: string; } /** Paging length control */ pageLength?: { /** Text for page length control */ menu?: Array<number | {label: string; value: number}>; /** Text for page length control */ text?: string; } /** Pagination buttons */ paging?: { /** Set the maximum number of paging number buttons */ buttons?: number; /** Paging button display options */ type?: 'numbers' | 'simple' | 'simple_numbers' | 'full' | 'full_numbers' | 'first_last_numbers'; /** * Set the maximum number of paging number buttons * @deprecated Use `buttons` instead. */ numbers?: number; /** Include the extreme page numbers, if separated by ellipsis, or not */ boundaryNumbers?: boolean; } /** Global search input */ search?: { /** Placeholder for the input element */ placeholder?: string; /** Text for search control */ text?: string; } } type LayoutNumber = '' | '1' | '2' | '3' | '4' | '5'; type LayoutSide = 'top' | 'bottom'; type LayoutEdge = 'Start' | 'End'; type LayoutKeys = `${LayoutSide}${LayoutNumber}${LayoutEdge}` | `${LayoutSide}${LayoutNumber}`; type Layout = Partial<Record<LayoutKeys, keyof Feature | Feature | Array<keyof Feature> | Feature[] | null>>; /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Main interfaces */ /** * DataTables class */ declare interface DataTables<T> extends DataTablesStatic { /** * Create a new DataTable * @param selector Selector to pick one or more `<table>` elements * @param opts Configuration settings */ new <T=any>( selector: InstSelector, opts?: Config ): Api<T> /** * CommonJS factory. This only applies to CommonJS environments, * in all others it would throw an error. */ (win?: Window, jQuery?: JQuery): DataTables<T>; } declare const DataTables: DataTables<any>; export default DataTables; // The `$().dataTable()` method adds a `.api()` method to the object // to allow access to the DataTables API interface JQueryDataTables extends JQuery { /** * Returns DataTables API instance * Usage: * $( selector ).dataTable().api(); */ api(): Api<any>; } // Extend the jQuery object with DataTables' construction methods declare global { interface JQueryDataTableApi extends DataTablesStatic { <T = any>(opts?: Config): Api<T>; } interface JQueryDataTableJq extends DataTablesStatic { (opts?: Config): JQueryDataTables; } interface JQuery { /** * Create a new DataTable, returning a DataTables API instance. * @param opts Configuration settings */ DataTable: JQueryDataTableApi; /** * Create a new DataTable, returning a jQuery object, extended * with an `api()` method which can be used to access the * DataTables API. * @param opts Configuration settings */ dataTable: JQueryDataTableJq; } } /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Options */ export interface Config { /** * Load data for the table's content from an Ajax source. */ ajax?: string | AjaxSettings | FunctionAjax; /** * Feature control DataTables' smart column width handling. */ autoWidth?: boolean; /** * Set a `caption` for the table. This can be used to describe the contents * of the table to the end user. A caption tag can also be read from HTML. */ caption?: string; /** * Data to use as the display data for the table. */ columns?: ConfigColumns[]; /** * Assign a column definition to one or more columns. */ columnDefs?: ConfigColumnDefs[]; /** * Data to use as the display data for the table. */ data?: any[]; /** * Delay the loading of server-side data until second draw */ deferLoading?: number | number[]; /** * Feature control deferred rendering for additional speed of initialisation. */ deferRender?: boolean; /** * Destroy any existing table matching the selector and replace with the new options. */ destroy?: boolean; /** * Initial paging start point. */ displayStart?: number; /** * Define the table control elements to appear on the page and in what order. * * @deprecated Use `layout` instead */ dom?: string; /** * Feature control table information display field. */ info?: boolean; /** * Language configuration object */ language?: ConfigLanguage; /** * */ layout?: Layout; /** * Feature control the end user's ability to change the paging display length of the table. */ lengthChange?: boolean; /** * Change the options in the page length select list. */ lengthMenu?: Array<(number | string)> | Array<Array<(number | string)>>; /** * Control which cell the order event handler will be applied to in a column. */ orderCellsTop?: boolean; /** * Highlight the columns being ordered in the table's body. */ orderClasses?: boolean; /** * Initial order (sort) to apply to the table. */ order?: Order | Order[]; /** * Ordering to always be applied to the table. */ orderFixed?: Order | Order[] | { pre?: Order | Order[], post: Order | Order[] }; /** * Feature control ordering (sorting) abilities in DataTables. */ ordering?: boolean; /** * Multiple column ordering ability control. */ orderMulti?: boolean; /** * Change the initial page length (number of rows per page). */ pageLength?: number; /** * Enable or disable table pagination. */ paging?: boolean; /** * Pagination button display options. Basic Types: numbers (1.10.8) simple, simple_numbers, full, full_numbers */ pagingType?: string; /** * Feature control the processing indicator. */ processing?: boolean; /** * Display component renderer types. */ renderer?: string | ConfigRenderer; /** * Retrieve an existing DataTables instance. */ retrieve?: boolean; /** * Data property name that DataTables will use to set <tr> element DOM IDs. Since: 1.10.8 */ rowId?: string; /** * Allow the table to reduce in height when a limited number of rows are shown. */ scrollCollapse?: boolean; /** * Horizontal scrolling. */ scrollX?: boolean; /** * Vertical scrolling. Since: 1.10 Exp: "200px" */ scrollY?: string; /** * Set an initial filter in DataTables and / or filtering options. */ search?: ConfigSearch | boolean; /** * Define an initial search for individual columns. */ searchCols?: ConfigSearch[]; /** * Set a throttle frequency for searching. */ searchDelay?: number; /** * Feature control search (filtering) abilities */ searching?: boolean; /** * Feature control DataTables' server-side processing mode. */ serverSide?: boolean; /** * Saved state validity duration. */ stateDuration?: number; /** * State saving - restore table state on page reload. */ stateSave?: boolean; /** * Set the zebra stripe class names for the rows in the table. */ stripeClasses?: string[]; /** * Tab index control for keyboard navigation. */ tabIndex?: number; /** * Callback for whenever a TR element is created for the table's body. */ createdRow?: FunctionCreateRow; /** * Function that is called every time DataTables performs a draw. */ drawCallback?: FunctionDrawCallback; /** * Footer display callback function. */ footerCallback?: FunctionFooterCallback; /** * Number formatting callback function. */ formatNumber?: FunctionFormatNumber; /** * Header display callback function. */ headerCallback?: FunctionHeaderCallback; /** * Table summary information display callback. */ infoCallback?: FunctionInfoCallback; /** * Initialisation complete callback. */ initComplete?: FunctionInitComplete; /** * Pre-draw callback. */ preDrawCallback?: FunctionPreDrawCallback; /** * Row draw callback.. */ rowCallback?: FunctionRowCallback; /** * Callback that defines where and how a saved state should be loaded. */ stateLoadCallback?: FunctionStateLoadCallback; /** * State loaded callback. */ stateLoaded?: FunctionStateLoaded; /** * State loaded - data manipulation callback. */ stateLoadParams?: FunctionStateLoadParams; /** * Callback that defines how the table state is stored and where. */ stateSaveCallback?: FunctionStateSaveCallback; /** * State save - data manipulation callback. */ stateSaveParams?: FunctionStateSaveParams; } export interface ConfigLanguage { emptyTable?: string; entries?: string | object; info?: string; infoEmpty?: string; infoFiltered?: string; infoPostFix?: string; decimal?: string; thousands?: string; lengthMenu?: string; loadingRecords?: string; processing?: string; search?: string; searchPlaceholder?: string; zeroRecords?: string; paginate?: { first?: string; last?: string; next?: string; previous?: string; }; aria?: { sortAscending?: string; sortDescending?: string; paginate?: { first?: string; last?: string; next?: string; previous?: string; }; }; url?: string; } export interface ConfigColumns { /** * Set the column's aria-label title. Since: 1.10.25 */ ariaTitle?: string; /** * Cell type to be created for a column. th/td */ cellType?: string; /** * Class to assign to each cell in the column. */ className?: string; /** * Add padding to the text content used when calculating the optimal with for a table. */ contentPadding?: string; /** * Cell created callback to allow DOM manipulation. */ createdCell?: FunctionColumnCreatedCell; /** * Class to assign to each cell in the column. */ data?: number | string | ObjectColumnData | FunctionColumnData | null; /** * Set default, static, content for a column. */ defaultContent?: string; /** * Text to display in the table's footer for this column. */ footer?: string; /** * Set a descriptive name for a column. */ name?: string; /** * Enable or disable ordering on this column. */ orderable?: boolean; /** * Define multiple column ordering as the default order for a column. */ orderData?: number | number[]; /** * Live DOM sorting type assignment. */ orderDataType?: string; /** * Ordering to always be applied to the table. Since 1.10 * * Array type is prefix ordering only and is a two-element array: * 0: Column index to order upon. * 1: Direction so order to apply ("asc" for ascending order or "desc" for descending order). */ orderFixed?: any[] | OrderFixed; /** * Order direction application sequence. */ orderSequence?: Array<'asc' | 'desc' | ''>; /** * Render (process) the data for use in the table. */ render?: number | string | ObjectColumnData | FunctionColumnRender | ObjectColumnRender; /** * Enable or disable filtering on the data in this column. */ searchable?: boolean; /** * Set the column title. */ title?: string; /** * Set the column type - used for filtering and sorting string processing. */ type?: string; /** * Enable or disable the display of this column. */ visible?: boolean; /** * Column width assignment. */ width?: string; } export type ConfigColumnDefs = ConfigColumnDefsMultiple | ConfigColumnDefsSingle; export interface ConfigColumnDefsMultiple extends ConfigColumns { /** * Target column(s). Either this or `target` must be specified. */ targets: string | number | Array<(number | string)>; } export interface ConfigColumnDefsSingle extends ConfigColumns { /** * Single column target. Either this or `targets` must be specified. Since: 1.12 */ target: string | number; } export interface ConfigRenderer { header?: string; pageButton?: string; } export interface ConfigSearch { /** * Control case-sensitive filtering option. */ caseInsensitive?: boolean; /** * Enable / disable escaping of regular expression characters in the search term. */ regex?: boolean; /** * Enable / disable DataTables' smart filtering. */ smart?: boolean; /** * Set an initial filtering condition on the table. */ search?: string; /** * Set a placeholder attribute for input type="text" tag elements. Since: 1.10.1 */ searchPlaceholder?: string; } /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * API */ export interface Api<T=any> { /** * API should be array-like */ [key: number]: any; /** * Returns DataTables API instance * * @param table Selector string for table */ (selector: string | Node | Node[] | JQuery): Api<T>; /** * Get jquery object * * @param selector jQuery selector to perform on the nodes inside the table's tbody tag. * @param modifier Option used to specify how the content's of the selected columns should be ordered, and if paging or filtering in the table should be taken into account. * @returns JQuery object with the matched elements in it's results set */ $(selector: string | Node | Node[] | JQuery, modifier?: ApiSelectorModifier): JQuery; /** * Ajax Methods */ ajax: ApiAjax; /** * Get a boolean value to indicate if there are any entries in the API instance's result set (i.e. any data, selected rows, etc). * @returns true if there there is one or more items in the result set, false otherwise. */ any(): boolean; /** * Get the contents of the `caption` element for the table. */ caption(): string; /** * Set the contents of the `-tag caption` element. If the table doesn't have * a `-tag caption` element, one will be created automatically. * * @param string The value to show in the table's `caption` tag. * @param side `top` or `bottom` to set where the table will be shown on the * table. If not given the previous value will be used (can also be set in * CSS). * @returns DataTables API instance for chaining. */ caption(set: string, side?: 'top' | 'bottom'): Api<T>; /** * Cell (single) selector and methods */ cell: ApiCell<T>; /** * Cells (multiple) selector and methods */ cells: ApiCells<T>; /** * Clear the table of all data. * * @returns DataTables Api instance. */ clear(): Api<T>; /** * Column Methods / object */ column: ApiColumn<T>; /** * Columns Methods / object */ columns: ApiColumns<T>; /** * Concatenate two or more API instances together * * @param a API instance to concatenate to the initial instance. * @param b Additional API instance(s) to concatenate to the initial instance. * @returns New API instance with the values from all passed in instances concatenated into its result set. */ concat(a: object, ...b: object[]): Api<any>; /** * The table setting objects that are manipulated by this API instance * * @private */ context: InternalSettings[], /** * Get the number of entries in an API instance's result set, regardless of multi-table grouping (e.g. any data, selected rows, etc). Since: 1.10.8 * * @returns The number of items in the API instance's result set */ count(): number; /** * Get the data for the whole table. * * @returns DataTables Api instance with the data for each row in the result set */ data(): Api<T>; /** * Destroy the DataTables in the current context. * * @param remove Completely remove the table from the DOM (true) or leave it in the DOM in its original plain un-enhanced HTML state (default, false). * @returns DataTables Api instance */ destroy(remove?: boolean): Api<T>; /** * Redraw the DataTables in the current context, optionally updating ordering, searching and paging as required. * * @param paging This parameter is used to determine what kind of draw DataTables will perform. * @returns DataTables Api instance */ draw(paging?: boolean | string): Api<T>; /** * Iterate over the contents of the API result set. * * @param fn Callback function which is called for each item in the API instance result set. The callback is called with three parameters * @returns Original API instance that was used. For chaining. */ each(fn: ((value: any, index: number, dt: Api<any>) => void)): Api<any>; /** * Reduce an Api instance to a single context and result set. * * @param idx Index to select * @returns New DataTables API instance with the context and result set containing the table and data for the index specified, or null if no matching index was available. */ eq(idx: number): Api<any>; /** * Show an error message to the end user / developer through the DataTables logging settings. * * @param msg Error message to show */ error(msg: string): Api<T>; /** * Iterate over the result set of an API instance and test each item, creating a new instance from those items which pass. * * @param fn Callback function which is called for each item in the API instance result set. The callback is called with three parameters. * @returns New API instance with the values from the result set which passed the test in the callback. */ filter(fn: ((value: any, index: number, dt: Api<any>) => boolean)): Api<Array<any>>; /** * Flatten a 2D array structured API instance to a 1D array structure. * * @returns New API instance with the 2D array values reduced to a 1D array. */ flatten(): Api<Array<any>>; /** * Look up a language token that was defined in the DataTables' language initialisation object. * * @param token The language token to lookup from the language object. * @param def The default value to use if the DataTables initialisation has not specified a value. This can be a string for simple cases, or an object for plurals. * @param numeric If handling numeric output, the number to be presented should be given in this parameter. * * @returns Resulting internationalised string. */ i18n(token: string, def: object | string, numeric?: number): string; /** * Find the first instance of a value in the API instance's result set. * * @param value Value to find in the instance's result set. * @returns The index of the item in the result set, or -1 if not found. */ indexOf(value: any): number; /** * Get the initialisation options used for the table. Since: DataTables 1.10.6 * * @returns Configuration object */ init(): Config; /** * Iterate over a result set of table, row, column or cell indexes * * @param type Iterator type * @param callback Callback function that is executed on each iteration. For the parameters passed to the function, please refer to the documentation above. As of this is executed in the scope of an API instance which has its context set to only the table in question. * @param returns Indicate if the callback function will return values or not. If set to true a new API instance will be returns with the return values from the callback function in its result set. If not set, or false the original instance will be returned for chaining, if no values are returned by the callback method. * @returns Original API instance if the callback returns no result (i.e. undefined) or a new API instance with the result set being the results from the callback, in order of execution. */ iterator(type: 'table', callback: InteratorTable, returns?: boolean): Api<any>; iterator(type: 'cell', callback: InteratorCell, returns?: boolean): Api<any>; iterator(type: 'column-rows', callback: InteratorColumnRows, returns?: boolean): Api<any>; iterator(type: 'column', callback: InteratorColumn, returns?: boolean): Api<any>; iterator(type: 'columns', callback: InteratorColumns, returns?: boolean): Api<any>; iterator(type: 'row', callback: InteratorRow, returns?: boolean): Api<any>; iterator(type: 'rows', callback: InteratorRows, returns?: boolean): Api<any>; /** * Iterate over a result set of table, row, column or cell indexes * * @param flatten If true the result set of the returned API instance will be a 1D array (i.e. flattened into a single array). If false (or not specified) each result will be concatenated to the instance's result set. Note that this is only relevant if you are returning arrays from the callback. * @param type Iterator type * @param callback Callback function that is executed on each iteration. For the parameters passed to the function, please refer to the documentation above. As of this is executed in the scope of an API instance which has its context set to only the table in question. * @param returns Indicate if the callback function will return values or not. If set to true a new API instance will be returns with the return values from the callback function in its result set. If not set, or false the original instance will be returned for chaining, if no values are returned by the callback method. * @returns Original API instance if the callback returns no result (i.e. undefined) or a new API instance with the result set being the results from the callback, in order of execution. */ iterator(flatten: boolean, type: 'table', callback: InteratorTable, returns?: boolean): Api<any>; iterator(flatten: boolean, type: 'cell', callback: InteratorCell, returns?: boolean): Api<any>; iterator(flatten: boolean, type: 'column-rows', callback: InteratorColumnRows, returns?: boolean): Api<any>; iterator(flatten: boolean, type: 'column', callback: InteratorColumn, returns?: boolean): Api<any>; iterator(flatten: boolean, type: 'columns', callback: InteratorColumns, returns?: boolean): Api<any>; iterator(flatten: boolean, type: 'row', callback: InteratorRow, returns?: boolean): Api<any>; iterator(flatten: boolean, type: 'rows', callback: InteratorRows, returns?: boolean): Api<any>; /** * Join the elements in the result set into a string. * * @param separator The string that will be used to separate each element of the result set. * @returns Contents of the instance's result set joined together as a single string. */ join(separator: string): string; /** * Find the last instance of a value in the API instance's result set. * * @param value Value to find in the instance's result set. * @returns The index of the item in the result set, or -1 if not found. */ lastIndexOf(value: any): number; /** * Number of elements in an API instance's result set. */ length: number; /** * Iterate over the result set of an API instance, creating a new API instance from the values returned by the callback. * * @param fn Callback function which is called for each item in the API instance result set. The callback is called with three parameters. * @returns New API instance with the values in the result set as those returned by the callback. */ map(fn: ((value: any, index: number, dt: Api<any>) => any)): Api<any>; /** * Remove event listeners that have previously been added with on(). * * @param event Event name to remove. * @param callback Specific callback function to remove if you want to unbind a single event listener. * @returns DataTables Api instance */ off(event: string, callback?: ((this: HTMLElement, e: Event, ...args: any[]) => void)): Api<T>; /** * Remove event handlers from selected elements * * @param event Event name to remove. * @param selector Element selector * @param callback Specific callback function to remove if you want to unbind a single event listener. * @returns DataTables Api instance */ off(event: string, selector: string, callback?: ((this: HTMLElement, e: Event, ...args: any[]) => void)): Api<T>; /** * Table events listener. * * @param event Event to listen for. * @param callback Event handler. * @returns DataTables Api instance */ on(event: string, callback: ((this: HTMLElement, e: Event, ...args: any[]) => void)): Api<T>; /** * Listen for events from selected elements * * @param event Event to listen for. * @param selector Element selector * @param callback Event handler. * @returns DataTables Api instance */ on(event: string, selector: string, callback: ((this: HTMLElement, e: Event, ...args: any[]) => void)): Api<T>; /** * Listen for a table event once and then remove the listener. * * @param event Event to listen for. * @param callback Event handler. * Listen for events from tables and fire a callback when they occur * @returns DataTables Api instance */ one(event: string, callback: ((this: HTMLElement, e: Event, ...args: any[]) => void)): Api<T>; /** * Listen for events from a selected element and trigger only once then remove the listener. * * @param event Event to listen for. * @param selector Element selector * @param callback Event handler. * @returns DataTables Api instance */ one(event: string, selector: string, callback: ((this: HTMLElement, e: Event, ...args: any[]) => void)): Api<T>; /** * Order Methods / object */ order: ApiOrder; /** * Page Methods / object */ page: ApiPage; /** * Iterate over the result set of an API instance, creating a new API instance from the values retrieved from the original elements. * * @param property object property name to use from the element in the original result set for the new result set. * @returns New API instance with the values in the result retrieved from the source object properties defined by the property being plucked. */ pluck(property: number | string): Api<any>; /** * Remove the last item from an API instance's result set. * * @returns Item removed form the result set (was previously the last item in the result set). */ pop(): any; /** * Add one or more items to the end of an API instance's result set. * * @param value_1 Item to add to the API instance's result set. * @returns The length of the modified API instance */ push(value_1: any, ...value_2: any[]): number; /** * Determine if the DataTable is ready or not */ ready(): boolean; /** * Execute a function when the DataTable becomes ready (or immediately if it already is) * * @param fn Function to execute */ ready(fn: ((this: Api<T>) => void)): Api<T>; /** * Apply a callback function against and accumulator and each element in the Api's result set (left-to-right). * * @param fn Callback function which is called for each item in the API instance result set. The callback is called with four parameters. * @param initialValue Value to use as the first argument of the first call to the fn callback. * @returns Result from the final call to the fn callback function. */ reduce(fn: ((current: number, value: any, index: number, dt: Api<any>) => number), initialValue?: any): any; /** * Apply a callback function against and accumulator and each element in the Api's result set (right-to-left). * * @param fn Callback function which is called for each item in the API instance result set. The callback is called with four parameters. * @param initialValue Value to use as the first argument of the first call to the fn callback. * @returns Result from the final call to the fn callback function. */ reduceRight(fn: ((current: number, value: any, index: number, dt: Api<any>) => number), initialValue?: any): any; /** * Reverse the result set of the API instance and return the original array. * * @returns The original API instance with the result set in reversed order. */ reverse(): Api<any>; /** * Row Methods / object */ row: ApiRow<T>; /** * Rows Methods / object */ rows: ApiRows<T>; /** * Search Methods / object */ search: ApiSearch<T>; /** * Obtain the table's settings object * * @returns DataTables API instance with the settings objects for the tables in the context in the result set */ settings(): Api<InternalSettings>; /** * Remove the first item from an API instance's result set. * * @returns Item removed form the result set (was previously the first item in the result set). */ shift(): any; /** * Create an independent copy of the API instance. * * @returns DataTables API instance */ slice(): Api<any>; /** * Sort the elements of the API instance's result set. * * @param fn This is a standard Javascript sort comparison function. It accepts two parameters. * @returns The original API instance with the result set sorted as defined by the sorting conditions used. */ sort(fn?: ((value1: any, value2: any) => number)): Api<Array<any>>; /** * Modify the contents of an Api instance's result set, adding or removing items from it as required. * * @param index Index at which to start modifying the Api instance's result set. * @param howMany Number of elements to remove from the result set. * @param value_1 Item to add to the result set at the index specified by the first parameter. * @returns An array of the items which were removed. If no elements were removed, an empty array is returned. */ splice(index: number, howMany: number, value_1?: any, ...value_2: any[]): any[]; /** * Page Methods / object */ state: ApiState<T>; /** * Select a table based on a selector from the API's context * * @param tableSelector Table selector. * @returns DataTables API instance with selected table in its context. */ table(tableSelector?: any): ApiTableMethods<T>; /** * Select tables based on the given selector * * @param tableSelector Table selector. * @returns DataTables API instance with all tables in the current context. */ tables(tableSelector?: any): ApiTablesMethods<T>; /** * Convert the API instance to a jQuery object, with the objects from the instance's result set in the jQuery result set. * * @returns jQuery object which contains the values from the API instance's result set. */ to$(): JQuery; /** * Create a native Javascript array object from an API instance. * * @returns Javascript array which contains the values from the API instance's result set. */ toArray(): any[]; /** * Convert the API instance to a jQuery object, with the objects from the instance's result set in the jQuery result set. * * @returns jQuery object which contains the values from the API instance's result set. */ toJQuery(): JQuery; /** * Trigger a DataTables related event. * * @param name The event name. * @param args An array of the arguments to send to the event. * @param bubbles Indicate if the event should bubble up the document in the * same way that DOM events usually do, or not. There is a performance * impact for bubbling events. * @returns Api instance for chaining */ trigger( name: string, args: any[], bubbles?: boolean): Api<T>; /** * Create a new API instance containing only the unique items from the elements in an instance's result set. * * @returns New Api instance which contains the unique items from the original instance's result set, in its own result set. */ unique(): Api<any>; /** * Add one or more items to the start of an API instance's result set. * * @param value_1 Item to add to the API instance's result set. * @returns The length of the modified API instance */ unshift(value_1: any, ...value_2: any[]): number; } export interface ApiSelectorModifier { /** * The order modifier provides the ability to control which order the rows are * processed in. Can be one of 'current', 'applied', 'index', 'original', or * the column index that you want the order to be applied from. */ order?: string | number; /** * The search modifier provides the ability to govern which rows are used by the selector using the search options that are applied to the table. * Values: 'none', 'applied', 'removed' */ search?: string; /** * The searchPlaceholder modifier provides the ability to provide informational text for an input control when it has no value. */ searchPlaceholder?: string; /** * The page modifier allows you to control if the selector should consider all data in the table, regardless of paging, or if only the rows in the currently disabled page should be used. * Values: 'all', 'current' */ page?: string; } export interface AjaxMethods extends Api<any> { /** * Reload the table data from the Ajax data source. * * @param callback Function which is executed when the data as been reloaded and the table fully redrawn. * @param resetPaging Reset (default action or true) or hold the current paging position (false). * @returns DataTables Api instance */ load(callback?: ((json: any) => void), resetPaging?: boolean): Api<any>; } export interface ApiSearch<T> { /** * Get current search * * @returns The currently applied global search. This may be an empty string if no search is applied. */ (): SearchInput<T>; /** * Set the global search to use on the table. Note this doesn't actually perform the search. * * @param input Search string to apply to the table. * @param regex Treat as a regular expression (true) or not (default, false). * @param smart Perform smart search. * @param caseInsen Do case-insensitive matching (default, true) or not (false). * @returns DataTables API instance */ (input: SearchInput<T>, regex?: boolean, smart?: boolean, caseInsen?: boolean): Api<any>; /** * Set the global search to use on the table. Note this doesn't actually perform the search. * * @param input Search string to apply to the table. * @param options Configuration options for how the search should be performed * @returns DataTables API instance */ (input: SearchInput<T>, options: SearchOptions): Api<any>; /** * Get a list of the names of searches applied to the table. * * @returns API instance containing the fixed search terms */ fixed(): Api<string>; /** * Get the search term used for the given name. * * @param name Fixed search term to get. * @returns The search term for the name given or undefined if not set. */ fixed( name: string ): SearchInput<T> | undefined; /** * Set a search term to apply to the table, using a name to uniquely identify it. * * @param name Name to give the fixed search term * @param search The search term to apply to the table or `null` to delete * an existing search term by the given name. * @returns API for chaining */ fixed( name: string, search: SearchInput<T> | null ): Api<T>; } export interface ApiPageInfo { page: number; pages: number; start: number; end: number; length: number; recordsTotal: number; recordsDisplay: number; serverSide: boolean; } export interface State { time: number; start: number; length: number; order: Array<Array<(string | number)>>; search: ConfigSearch; columns: Array<{ search: ConfigSearch; visible: boolean; }>; } /** * "table" - loop over the context's (i.e. the tables) for the instance * * @param settings Table settings object * @param counter Loop counter */ type InteratorTable = (settings: InternalSettings, counter: number) => any; /** * "cell" - loop over each table and cell in the result set * * @param settings Table settings object * @param rowIndex Row index * @param columnIndex Column index * @param tableCounter Table counter (outer) * @param cellCounter Cell counter (inner) */ type InteratorCell = (settings: InternalSettings, rowIndex: number, columnIndex: number, tableCounter: number, cellCounter: number) => any; /** * "columns" - loop over each item in the result set * * @param settings Table settings object * @param resultItem Result set item * @param counter Loop counter */ type InteratorColumns = (settings: InternalSettings, resultItem: any, counter: number) => any; /** * "column" - loop over each table and column in the result set * * @param settings Table settings object * @param columnIndex Column index * @param tableCounter Table counter (outer) * @param columnCounter Column counter (inner) */ type InteratorColumn = (settings: InternalSettings, columnIndex: number, tableCounter: number, columnCounter: number) => any; /** * "column-rows" - loop over each table, column and row in the result set applying selector-modifier. * * @param settings Table settings object * @param columnIndex Column index * @param tableCounter Table counter (outer) * @param columnCounter Column counter (inner) * @param rowIndexes Row indexes */ type InteratorColumnRows = (settings: InternalSettings, columnIndex: number, tableCounter: number, columnCounter: number, rowIndexes: number[]) => any; /** * "row" - loop over each table and row in the result set * * @param settings Table settings object * @param rowIndex Row index * @param tableCounter Table counter (outer) * @param rowCounter Row counter (inner) */ type InteratorRow = (settings: InternalSettings, rowIndex: number, tableCounter: number, rowCounter: number) => any; /** * "rows" - loop over each item in the result set * * @param settings Table settings object * @param resultItem Result set item * @param counter Loop counter */ type InteratorRows = (settings: InternalSettings, resultItem: any, counter: number) => any; export interface ApiAjax { /** * Get the latest JSON data obtained from the last Ajax request DataTables made * * @returns JSON object that was last loaded/ */ json(): object; /** * Get the data submitted by DataTables to the server in the last Ajax request * * @returns object containing the data submitted by DataTables */ params(): object; /** * Reload the table data from the Ajax data source. * * @param callback Function which is executed when the data as been reloaded and the table fully redrawn. * @param resetPaging Reset (default action or true) or hold the current paging position (false). * @returns DataTables Api */ reload(callback?: ((json: any) => void), resetPaging?: boolean): Api<any>; /** * Reload the table data from the Ajax data source * * @returns URL set as the Ajax data source for the table. */ url(): string; /** * Reload the table data from the Ajax data source * * @param url URL to set to be the Ajax data source for the table. * @returns DataTables Api instance for chaining or further ajax.url() methods */ url(url: string): AjaxMethods; } export interface ApiPage { /** * Get the current page of the table. * * @returns Currently displayed page number */ (): number; /** * Set the current page of the table. * * @param page Index or 'first', 'next', 'previous', 'last' * @returns DataTables API instance */ (page: number | string): Api<any>; /** * Get paging information about the table * * @returns Object with information about the table's paging state. */ info(): ApiPageInfo; /** * Get the table's page length. * * @returns Current page length. */ len(): number; /** * Set the table's page length. * * @param length Page length to set. use -1 to show all records. * @returns DataTables API instance. */ len(length: number): Api<any>; } export interface ApiOrder { /** * Get the ordering applied to the table. * * @returns Array of arrays containing information about the currently applied sort. This 2D array is the same format as the array used for setting the order to apply to the table */ (): OrderArray[]; /** * Set the ordering applied to the table. * * @param order Order Model * @returns DataTables Api instance */ (order?: Order | Order[]): Api<any>; (order: Order, ...args: Order[]): Api<any>; /** * Get the fixed ordering that is applied to the table. If there is more than one table in the API's context, * the ordering of the first table will be returned only (use table() if you require the ordering of a different table in the API's context). * @returns object describing the ordering that is applied to the table */ fixed(): OrderFixed; /** * Set the table's fixed ordering. Note this doesn't actually perform the order, but rather queues it up - use draw() to perform the ordering. * * @param order Used to indicate whether the ordering should be performed before or after the users own ordering. * @returns DataTables Api instance */ fixed(order: OrderFixed): Api<any>; /** * Add an ordering listener to an element, for a given column. * * @param node Selector * @param column Column index * @param callback Callback function * @returns DataTables API instance with the current order in the result set */ listener(node: string | Node | JQuery, column: number, callback: (() => void)): Api<any>; } export interface ApiState<T> { /** * Get the last saved state of the table * * @returns State saved object */ (): State; /** * Clear the saved state of the table. * * @returns The API instance that was used, available for chaining. */ clear(): Api<any>; /** * Get the table state that was loaded during initialisation. * * @returns State saved object. See state() for the object format. */ loaded(): State; /** * Trigger a state save. * * @returns The API instance that was used, available for chaining. */ save(): Api<T>; } export interface ApiCell<T> { /** * Select the cell found by a cell selector * * @param cellSelector Cell selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering * @returns DataTables API instance with selected cell */ (cellSelector: CellSelector, modifier?: ApiSelectorModifier): ApiCellMethods<T>; /** * Select the cell found by a cell selector * * @param rowSelector Row selector. * @param columnSelector Column selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering * @returns DataTables API instance with selected cell */ (rowSelector: RowSelector<T>, columnSelector: ColumnSelector, modifier?: ApiSelectorModifier): ApiCellMethods<T>; } export interface ApiCellMethods<T> extends Omit<Api<T>, 'render' | 'select'> { /** * Get the DataTables cached data for the selected cell * * @param type Specify which cache the data should be read from. Can take one of two values: search or order * @returns DataTables API instance with the cached data for each selected cell in the result set */ cache(type: string): Api<T>; /** * Get data for the selected cell * * @returns the data from the cell */ data(): any; /** * Get data for the selected cell * * @param data Value to assign to the data for the cell * @returns DataTables Api instance */ data(data: any): Api<T>; /** * Get index information about the selected cell * * @returns Object with index information for the selected cell. */ index(): CellIdxWithVisible; /** * Invalidate the data held in DataTables for the selected cell * * @param source Data source to read the new data from. * @returns DataTables API instance with selected cell references in the result set */ invalidate(source?: string): Api<T>; /** * Get the DOM element for the selected cell * * @returns The TD / TH cell the selector resolved to */ node(): Node; /** * Get data for the selected cell * * @param type Data type to get. This can be one of: 'display', 'filter', 'sort', 'type' * @returns Rendered data for the requested type */ render(type: string): any; } export interface ApiCells<T> { /** * Select all cells * * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering * @returns DataTables API instance with selected cells */ (modifier?: ApiSelectorModifier): ApiCellsMethods<T>; /** * Select cells found by a cell selector * * @param cellSelector Cell selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering * @returns DataTables API instance with selected cells */ (cellSelector: CellSelector, modifier?: ApiSelectorModifier): ApiCellsMethods<T>; /** * Select cells found by both row and column selectors * * @param rowSelector Row selector. * @param columnSelector Column selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering * @returns DataTables API instance with selected cells */ (rowSelector: RowSelector<T>, columnSelector: ColumnSelector, modifier?: ApiSelectorModifier): ApiCellsMethods<T>; } export interface ApiCellsMethods<T> extends Omit<Api<T>, 'data' | 'render' | 'select'> { /** * Get the DataTables cached data for the selected cells * * @param type Specify which cache the data should be read from. Can take one of two values: search or order * @returns DataTables API instance with the cached data for each selected cell in the result set */ cache(type: string): Api<T>; /** * Get data for the selected cells * * @returns DataTables API instance with data for each cell in the selected columns in the result set. This is a 1D array with each entry being the data for the cells from the selected column. */ data(): Api<Array<T>>; /** * Iterate over each selected cell, with the function context set to be the cell in question. Since: DataTables 1.10.6 * * @param fn Function to execute for every cell selected. */ every(fn: (this: ApiCellsMethods<T>, cellRowIdx: number, cellColIdx: number, tableLoop: number, cellLoop: number) => void): Api<any>; /** * Get index information about the selected cells */ indexes(): Api<CellIdxWithVisible>; /** * Invalidate the data held in DataTables for the selected cells * * @param source Data source to read the new data from. * @returns DataTables API instance with selected cell references in the result set */ invalidate(source?: string): Api<T>; /** * Get the DOM elements for the selected cells */ nodes(): Api<Node>; /** * Get data for the selected cell * * @param type Data type to get. This can be one of: 'display', 'filter', 'sort', 'type' * @returns Rendered data for the requested type */ render(type: string): any; /** * Get the title text for a column * * @param row Indicate which row in the header the title should be read from * when working with multi-row headers. * @return Column title */ title( row?: number ): string; /** * Set the title text for a column * * @param title Title to set * @param row Indicate which row in the header the title should be read from * when working with multi-row headers. * @return DataTables API instance for chaining */ title( title: string, row?: number ): Api<T>; /** * Get the column's data type (auto detected or configured). * * @return The column's data type. */ type(): string; /** * Compute the width of a column as it is shown. * * @return The width of the column in pixels or `null` if there is no data * in the table. */ width(): number | null; } export interface ApiColumn<T> { /** * Select the column found by a column selector * * @param columnSelector Column selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering in the table should be taken into account. */ (columnSelector: ColumnSelector, modifier?: ApiSelectorModifier): ApiColumnMethods<T>; /** * Convert from the input column index type to that required. * * @param type The type on conversion that should take place: 'fromVisible', 'toData', 'fromData', 'toVisible' * @param index The index to be converted * @returns Calculated column index */ index(type: string, index: number): number; } export interface ApiColumnMethods<T> extends Omit<Api<T>, 'init' | 'data' | 'order' | 'render' | 'search'> { /** * Get the DataTables cached data for the selected column(s) * * @param type Specify which cache the data should be read from. Can take one of two values: search or order * @return DataTables Api instance with an caches data for the selected column(s) */ cache(type: string): Api<any>; /** * Get the data for the cells in the selected column. * * @returns DataTables API instance with data for each cell in the selected columns in the result set. This is a 1D array with each entry being the data for the cells from the selected column. */ data(): Api<Array<any>>; /** * Get the data source property for the selected column. * * @returns the data source property */ dataSrc(): number | string | (() => string); /** * Get the footer th / td cell for the selected column. * * @param row Indicate which row in the footer the cell should be read from * when working with multi-row footers. * @returns HTML element for the footer of the column */ footer(row?: number): HTMLElement; /** * Get the header th / td cell for a column. * * @param row Indicate which row in the header the cell should be read from * when working with multi-row headers. * @returns HTML element for the header of the column */ header(row?: number): HTMLElement; /** * Get the column index of the selected column. * * @param type Specify if you want to get the column data index (default) or the visible index (visible). * @returns The column index for the selected column. */ index(type?: string): number; /** * Get the initialisation object used for the selected column. * * @returns Column configuration object */ init(): ConfigColumns; /** * Obtain the th / td nodes for the selected column * * @returns DataTables API instance with each cell's node from the selected columns in the result set. This is a 1D array with each entry being the node for the cells from the selected column. */ nodes(): Api<Array<Node>>; /** * Order the table, in the direction specified, by the column selected by the column() selector. * * @param direction Direction of sort to apply to the selected column - desc (descending) or asc (ascending). * @returns DataTables API instance */ order(direction: string): Api<any>; /** * Get the orderable state for the column (from `columns.orderable`). */ orderable(): boolean; /** * Get a list of the column ordering directions (from `columns.orderSequence`). */ orderable(directions: true): Api<string>; /** * Get rendered data for the selected column. * @param type Data type to get. Typically `display`, `filter`, `sort` or `type` * although can be anything that the rendering functions expect. */ render(type?: string): Api<T>; /** * Search methods and properties */ search: ApiColumnSearch<T>; /** * Get the title text for the selected column * * @param row Indicate which row in the header the title should be read from * when working with multi-row headers. * @return Column titles in API instance's data set */ title( row?: number ): string; /** * Set the title text for the selected column * * @param title Title to set * @param row Indicate which row in the header the title should be read from * when working with multi-row headers. * @return DataTables API instance for chaining */ title( title: string, row?: number ): Api<T>; /** * Get the data type for the selected column (auto detected or configured). * * @return DataTables API instance with column types in its data set */ type(): string; /** * Get the visibility of the selected column. * * @returns true if the column is visible, false if it is not. */ visible(): boolean; /** * Set the visibility of the selected column. * * @param show Specify if the column should be visible (true) or not (false). * @param redrawCalculations Indicate if DataTables should recalculate the column layout (true - default) or not (false). * @returns DataTables API instance with selected column in the result set. */ visible(show: boolean, redrawCalculations?: boolean): Api<any>; /** * Compute the width of the selected column as they are shown. * * @return Api instance with the width of each column in pixels or `null` if * there is no data in the table. */ width(): number | null; } export interface ApiColumnSearch<T> { /** * Get the currently applied column search. * * @returns the currently applied column search. */ (): string; /** * Set the search term for the matched column. * * @param input Search apply. * @param regex Treat as a regular expression (true) or not (default, false). * @param smart Perform smart search. * @param caseInsen Do case-insensitive matching (default, true) or not (false). * @returns DataTables API instance */ (input: SearchInputColumn<T>, regex?: boolean, smart?: boolean, caseInsen?: boolean): Api<any>; /** * Set the search term for the matched column. * * @param input Search to apply. * @param Search Search configuration options * @returns DataTables API instance */ (input: SearchInputColumn<T>, options: SearchOptions): Api<any>; /** * Get a list of the names of searches applied to the column * * @returns API instance containing the column's fixed search terms */ fixed(): Api<string>; /** * Get the search term for the column used for the given name. * * @param name Fixed search term to get. * @returns The search term for the name given or undefined if not set. */ fixed( name: string ): SearchInputColumn<T> | undefined; /** * Set a search term to apply to the column, using a name to uniquely identify it. * * @param name Name to give the fixed search term * @param search The search term to apply to the column or `null` to delete * an existing search term by the given name. * @returns API for chaining */ fixed( name: string, search: SearchInputColumn<T> | null ): Api<T>; } export interface ApiColumns<T> { /** * Select all columns * * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering in the table should be taken into account. * @returns DataTables API instance with selected columns in the result set. */ (modifier?: ApiSelectorModifier): ApiColumnsMethods<T>; /** * Select columns found by a cell selector * * @param columnSelector Column selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering in the table should be taken into account. * @returns DataTables API instance with selected columns */ (columnSelector: ColumnSelector, modifier?: ApiSelectorModifier): ApiColumnsMethods<T>; /** * Recalculate the column widths for layout. * * @returns DataTables API instance. */ adjust(): Api<T>; } export interface ApiColumnsMethods<T> extends Omit<Api<T>, 'init' | 'data' | 'order' | 'render' | 'search'> { /** * Get the DataTables cached data for the selected columna * * @param type Specify which cache the data should be read from. Can take one of two values: search or order * @return DataTables Api instance with an caches data for the selected columna */ cache(type: string): Api<any>; /** * Obtain the data for the columns from the selector * * @returns DataTables API instance with data for each cell in the selected columns in the result set. This is a 2D array with the top level array entries for each column matched by the columns() selector. */ data(): Api<Array<Array<any>>>; /** * Get the data source property for the selected columns. * * @returns API instance with the result set containing the data source parameters for the selected columns as configured by */ dataSrc(): Api<any>; /** * Iterate over each selected column, with the function context set to be the column in question. Since: DataTables 1.10.6 * * @param fn Function to execute for every column selected. * @returns DataTables API instance of the selected columns. */ every(fn: (this: ApiColumnMethods<T>, colIdx: number, tableLoop: number, colLoop: number) => void): Api<any>; /** * Get the footer th / td cell for the selected columns. * * @param row Indicate which row in the footer the cell should be read from * when working with multi-row footers. * @returns HTML element for the footer of the columns */ footer(row?: number): Api<HTMLElement>; /** * Get the header th / td cell for a columns. * * @param row Indicate which row in the header the cell should be read from * when working with multi-row headers. * @returns HTML element for the header of the columns */ header(row?: number): Api<HTMLElement>; /** * Get the column indexes of the selected columns. * * @param type Specify if you want to get the column data index (default) or the visible index (visible). * @returns DataTables API instance with selected columns' indexes in the result set. */ indexes(type?: string): Api<Array<number>>; /** * Get the initialisation objects used for the selected columns. * * @returns Api instance of column configuration objects */ init(): Api<ConfigColumns>; /** * Obtain the th / td nodes for the selected columns * * @returns DataTables API instance with each cell's node from the selected columns in the result set. This is a 2D array with the top level array entries for each column matched by the columns() selector. */ nodes(): Api<Array<Array<Node>>>; /** * Order the table, in the direction specified, by the columns selected by the column() selector. * * @param direction Direction of sort to apply to the selected columna - desc (descending) or asc (ascending). * @returns DataTables API instance */ order(direction: string): Api<any>; /** * Get the orderable state for the selected columns (from `columns.orderable`). */ orderable(): Api<boolean>; /** * Get a list of the column ordering directions (from `columns.orderSequence`). */ orderable(directions: true): Api<Array<string>>; /** * Get rendered data for the selected columns. * @param type Data type to get. Typically `display`, `filter`, `sort` or `type` * although can be anything that the rendering functions expect. */ render(type?: string): Api<Array<T>>; /** * Search methods and properties */ search: ApiColumnsSearch<T>; /** * Get the title text for the selected columns * * @param row Indicate which row in the header the title should be read from * when working with multi-row headers. * @return Column titles in API instance's data set */ titles( row?: number ): Api<string>; /** * Set the title text for the selected columns * * @param title Title to set * @param row Indicate which row in the header the title should be read from * when working with multi-row headers. * @return DataTables API instance for chaining */ titles( title: string, row?: number ): Api<T>; /** * Get the data type for the selected columns (auto detected or configured). * * @return DataTables API instance with column types in its data set */ types(): Api<string>; /** * Get the visibility of the selected columns. * * @returns true if the columns is visible, false if it is not. */ visible(): boolean; /** * Set the visibility of the selected columns. * * @param show Specify if the columns should be visible (true) or not (false). * @param redrawCalculations Indicate if DataTables should recalculate the columns layout (true - default) or not (false). * @returns DataTables API instance with selected columns in the result set. */ visible(show: boolean, redrawCalculations?: boolean): Api<any>; /** * Compute the width of the selected columns as they are shown. * * @return Api instance with the width of each column in pixels or `null` if * there is no data in the table. */ widths(): Api<number | null>; } export interface ApiColumnsSearch<T> { /** * Get the currently applied columns search. * * @returns the currently applied columns search. */ (): Api<SearchInputColumn<T>[]>; /** * Set the search term for the columns from the selector. Note this doesn't actually perform the search. * * @param input Search to apply to the selected columns. * @param regex Treat as a regular expression (true) or not (default, false). * @param smart Perform smart search (default, true) or not (false). * @param caseInsen Do case-insensitive matching (default, true) or not (false). * @returns DataTables Api instance. */ (input: SearchInputColumn<T>, regex?: boolean, smart?: boolean, caseInsen?: boolean): Api<any>; /** * Set the search term for the matched columns. * * @param input Search to apply. * @param Search Search configuration options * @returns DataTables API instance */ (input: SearchInputColumn<T>, options: SearchOptions): Api<any>; /** * Get a list of the names of searches applied to the matched columns * * @returns API instance containing the column's fixed search terms */ fixed(): Api<string[]>; /** * Get the search term for the matched columns used for the given name. * * @param name Fixed search term to get. * @returns The search term for the name given or undefined if not set. */ fixed( name: string ): Api<SearchInputColumn<T> | undefined>; /** * Set a search term to apply to the matched columns, using a name to * uniquely identify it. * * @param name Name to give the fixed search term * @param search The search term to apply to the column or `null` to delete * an existing search term by the given name. * @returns API for chaining */ fixed( name: string, search: SearchInputColumn<T> | null ): Api<T>; } export interface ApiRowChildMethods<T> { /** * Get the child row(s) that have been set for a parent row * * @returns Query object with the child rows for the parent row in its result set, or undefined if there are no child rows set for the parent yet. */ (): JQuery; /** * Get the child row(s) that have been set for a parent row * * @param showRemove This parameter can be given as true or false * @returns DataTables Api instance. */ (showRemove: boolean): RowChildMethods<T>; /** * Set the data to show in the child row(s). Note that calling this method will replace any child rows which are already attached to the parent row. * * @param data The data to be shown in the child row can be given in multiple different ways. * @param className Class name that is added to the td cell node(s) of the child row(s). As of 1.10.1 it is also added to the tr row node of the child row(s). * @returns DataTables Api instance */ (data: (string | Node | JQuery) | Array<(string | number | JQuery)>, className?: string): RowChildMethods<T>; /** * Hide the child row(s) of a parent row * * @returns DataTables API instance. */ hide(): Api<any>; /** * Check if the child rows of a parent row are visible * * @returns boolean indicating whether the child rows are visible. */ isShown(): boolean; /** * Remove child row(s) from display and release any allocated memory * * @returns DataTables API instance. */ remove(): Api<any>; /** * Show the child row(s) of a parent row * * @returns DataTables API instance. */ show(): Api<any>; } export interface RowChildMethods<T> extends Api<T> { /** * Hide the child row(s) of a parent row * * @returns DataTables API instance. */ hide(): Api<any>; /** * Remove child row(s) from display and release any allocated memory * * @returns DataTables API instance. */ remove(): Api<any>; /** * Make newly defined child rows visible * * @returns DataTables API instance. */ show(): Api<any>; } export interface ApiRow<T> { /** * Select a row found by a row selector * * @param rowSelector Row selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering in the table should be taken into account. * @returns DataTables API instance with selected row in the result set */ (rowSelector: RowSelector<T>, modifier?: ApiSelectorModifier): ApiRowMethods<T>; /** * Add a new row to the table using the given data * * @param data Data to use for the new row. This may be an array, object or Javascript object instance, but must be in the same format as the other data in the table+ * @returns DataTables API instance with the newly added row in its result set. */ add(data: any[] | object): ApiRowMethods<T>; } export interface ApiRowMethods<T> extends Omit<Api<T>, 'data' | 'select'> { /** * Get the DataTables cached data for the selected row(s) * * @param type Specify which cache the data should be read from. Can take one of two values: search or order * @returns DataTables API instance with data for each cell in the selected row in the result set. This is a 1D array with each entry being the data for the cells from the selected row. */ cache(type: string): Api<Array<any>> | Api <Array<Array<any>>>; /** * Order Methods / object */ child: ApiRowChildMethods<T>; /** * Get the data for the selected row * * @returns Data source object for the data source of the row. */ data(): T; /** * Set the data for the selected row * * @param d Data to use for the row. * @returns DataTables API instance with the row retrieved by the selector in the result set. */ data(d: any[] | object): Api<T>; /** * Get the id of the selected row. Since: 1.10.8 * * @param hash true - Append a hash (#) to the start of the row id. This can be useful for then using the id as a selector * false - Do not modify the id value. * @returns Row id. If the row does not have an id available 'undefined' will be returned. */ id(hash?: boolean): string; /** * Get the row index of the row column. * * @returns Row index */ index(): number; /** * Obtain the th / td nodes for the selected row(s) * * @param source Data source to read the new data from. Values: 'auto', 'data', 'dom' */ invalidate(source?: string): Api<Array<any>>; /** * Obtain the tr node for the selected row * * @returns tr element of the selected row or null if the element is not yet available */ node(): Node; /** * Delete the selected row from the DataTable. * * @returns DataTables API instance with removed row reference in the result set */ remove(): Api<Node>; } export interface ApiRows<T> { /** * Select all rows * * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering in the table should be taken into account. * @returns DataTables API instance with selected rows */ (modifier?: ApiSelectorModifier): ApiRowsMethods<T>; /** * Select rows found by a row selector * * @param rowSelector Row selector. * @param modifier Option used to specify how the cells should be ordered, and if paging or filtering in the table should be taken into account. * @returns DataTables API instance with selected rows in the result set */ (rowSelector: RowSelector<T>, modifier?: ApiSelectorModifier): ApiRowsMethods<T>; /** * Add new rows to the table using the data given * * @param data Array of data elements, with each one describing a new row to be added to the table * @returns DataTables API instance with the newly added rows in its result set. */ add(data: T[]): ApiRowsMethods<T>; } export interface ApiRowsMethods<T> extends Omit<Api<T>, 'select'> { /** * Get the DataTables cached data for the selected row(s) * * @param type Specify which cache the data should be read from. Can take one of two values: search or order * @returns DataTables API instance with data for each cell in the selected row in the result set. This is a 1D array with each entry being the data for the cells from the selected row. */ cache(type: string): Api<Array<any>> | Api <Array<Array<any>>>; /** * Get the data for the selected rows * * @returns DataTables API instance with data for each row from the selector in the result set. */ data(): Api<T>; /** * Iterate over each selected row, with the function context set to be the row in question. Since: DataTables 1.10.6 * * @param fn Function to execute for every row selected. * @returns DataTables API instance of the selected rows. */ every(fn: (this: ApiRowMethods<T>, rowIdx: number, tableLoop: number, rowLoop: number) => void): Api<any>; /** * Get the ids of the selected rows. Since: 1.10.8 * * @param hash true - Append a hash (#) to the start of each row id. This can be useful for then using the ids as selectors * false - Do not modify the id value. * @returns Api instance with the selected rows in its result set. If a row does not have an id available 'undefined' will be returned as the value. */ ids(hash?: boolean): Api<Array<any>>; /** * Get the row indexes of the selected rows. * * @returns DataTables API instance with selected row indexes in the result set. */ indexes(): Api<Array<number>>; /** * Obtain the th / td nodes for the selected row(s) * * @param source Data source to read the new data from. Values: 'auto', 'data', 'dom' */ invalidate(source?: string): Api<Array<any>>; /** * Obtain the tr nodes for the selected rows * * @returns DataTables API instance with each row's node from the selected rows in the result set. */ nodes(): Api<Array<Node>>; /** * Delete the selected rows from the DataTable. * * @returns DataTables API instance with references for the removed rows in the result set */ remove(): Api<Array<any>>; } export interface ApiTableMethods<T> extends Api<T> { footer: { /** * Get the tfoot node for the table in the API's context * * @returns HTML tbody node. */ (): Node; /** * Get an array that represents the structure of the footer * * @param selector Column selector */ structure(selector?: string): HeaderStructure[][]; } header: { /** * Get the thead node for the table in the API's context * * @returns HTML thead node. */ (): Node; /** * Get an array that represents the structure of the header * * @param selector Column selector */ structure(selector?: string): HeaderStructure[][]; } /** * Get the tbody node for the table in the API's context * * @returns HTML tfoot node. */ body(): Node; /** * Get the div container node for the table in the API's context * * @returns HTML div node. */ container(): Node; /** * Get the table node for the table in the API's context * * @returns HTML table node for the selected table. */ node(): Node; } export interface ApiTablesMethods<T> extends Api<T> { /** * Get the tfoot nodes for the tables in the API's context * * @returns Array of HTML tfoot nodes for all table in the API's context */ footer(): Api<Array<Node>>; /** * Get the thead nodes for the tables in the API's context * * @returns Array of HTML thead nodes for all table in the API's context */ header(): Api<Array<Node>>; /** * Get the tbody nodes for the tables in the API's context * * @returns Array of HTML tbody nodes for all table in the API's context */ body(): Api<Array<Node>>; /** * Get the div container nodes for the tables in the API's context * * @returns Array of HTML div nodes for all table in the API's context */ containers(): Api<Array<Node>>; /** * Iterate over each selected table, with the function context set to be the table in question. Since: DataTables 1.10.6 * * @param fn Function to execute for every table selected. */ every(fn: (this: ApiTableMethods<T>, tableIndex: number) => void): Api<any>; /** * Get the table nodes for the tables in the API's context * * @returns Array of HTML table nodes for all table in the API's context */ nodes(): Api<Array<Node>>; } export interface DataTablesStatic { /** * Get DataTable API instance * * @param table Selector string for table */ Api: ApiStatic; /** * Default Settings */ defaults: Config; /** * Default Settings */ ext: DataTablesStaticExt; /** Feature control namespace */ feature: { /** * Create a new feature that can be used for layout * * @param name The name of the new feature. * @param construct A function that will create the elements and event listeners for the feature being added. */ register(name: string, construct: (dt: InternalSettings, options: any) => HTMLElement | JQuery): void; } /** * Check if a table node is a DataTable already or not. * * @param table The table to check. * @returns true the given table is a DataTable, false otherwise */ isDataTable(table: string | Node | JQuery | Api<any>): boolean; /** * Helpers for `columns.render`. * * The options defined here can be used with the `columns.render` initialisation * option to provide a display renderer. */ render: DataTablesStaticRender; /** * Get all DataTable tables that have been initialised - optionally you can select to get only currently visible tables and / or retrieve the tables as API instances. * * @param visible As a boolean value this options is used to indicate if you want all tables on the page should be returned (false), or visible tables only (true). * Since 1.10.8 this option can also be given as an object. * @returns Array or DataTables API instance containing all matching DataTables */ tables(visible?: { /** * Get only visible tables (true) or all tables regardless of visibility (false). */ visible: boolean; /** * Return a DataTables API instance for the selected tables (true) or an array (false). */ api: boolean; } | boolean): Array<Api<any>>| Api<any>; /** * Get the data type definition object for a specific registered data type. * * @param name Data type name to get the definition of */ type(name: string): DataType; /** * Set one or more properties for a specific data type. * * @param name Data type name to set values for * @param definition Object containing the values to set */ type(name: string, definition: DataType): void; /** * Set an individual property value for a a given data type. * * @param name Data type name to set a property value for * @param property Name of the data type property to set * @param val The value to set (typically a function or string) */ type(name: string, property: keyof(DataType), val: any): void; /** * Get the names of the registered data types DataTables can work with. */ types(): string[]; /** * Utils */ util: DataTablesStaticUtil; /** * Version number compatibility check function * * Usage: * $.fn.dataTable.versionCheck("1.10.0"); * @param version Version string * @returns true if this version of DataTables is greater or equal to the required version, or false if this version of DataTales is not suitable */ versionCheck(version: string): boolean; } export type ApiStaticRegisterFn<T> = (this: Api<T>, ...args: any[]) => any; export interface ApiStatic { /** * Create a new API instance to an existing DataTable. Note that this * does not create a new DataTable. */ new (selector: string | Node | Node[] | JQuery | InternalSettings): Api<any>; register<T=any>(name: string, fn: ApiStaticRegisterFn<T>): void; registerPlural<T=any>(pluralName: string, singleName: string, fn: ApiStaticRegisterFn<T>): void; } export interface OrderFixed { /** * Two-element array: * 0: Column index to order upon. * 1: Direction so order to apply ("asc" for ascending order or "desc" for descending order). */ pre?: any[]; /** * Two-element array: * 0: Column index to order upon. * 1: Direction so order to apply ("asc" for ascending order or "desc" for descending order). */ post?: any[]; } export interface DataTablesStaticRender { /** * Format an ISO8061 date in auto locale detected format */ date(): ObjectColumnRender; /** * Format an ISO8061 date value using the specified format * @param to Display format * @param locale Locale * @param def Default value if empty */ date(to: string, locale?: string): ObjectColumnRender; /** * Format a date value * @param from Data format * @param to Display format * @param locale Locale * @param def Default value if empty */ date(from?: string, to?: string, locale?: string, def?: string): ObjectColumnRender; /** * Format an ISO8061 datetime in auto locale detected format */ datetime(): ObjectColumnRender; /** * Format an ISO8061 datetime value using the specified format * @param to Display format * @param locale Locale * @param def Default value if empty */ datetime(to: string, locale?: string): ObjectColumnRender; /** * Format a datetime value * @param from Data format * @param to Display format * @param locale Locale * @param def Default value if empty */ datetime(from?: string, to?: string, locale?: string, def?: string): ObjectColumnRender; /** * Render a number with auto-detected locale thousands and decimal */ number(): ObjectColumnRender; /** * Will format numeric data (defined by `columns.data`) for display, retaining the original unformatted data for sorting and filtering. * * @param thousands Thousands grouping separator. `null` for auto locale * @param decimal Decimal point indicator. `null` for auto locale * @param precision Integer number of decimal points to show. * @param prefix Prefix (optional). * @param postfix Postfix (/suffix) (optional). */ number(thousands: string | null, decimal: string | null, precision: number, prefix?: string, postfix?: string): ObjectColumnRender; /** * Escape HTML to help prevent XSS attacks. It has no optional parameters. */ text(): ObjectColumnRender; /** * Format an ISO8061 date in auto locale detected format */ time(): ObjectColumnRender; /** * Format an ISO8061 time value using the specified format * @param to Display format * @param locale Locale * @param def Default value if empty */ time(to: string, locale?: string): ObjectColumnRender; /** * Format a time value * @param from Data format * @param to Display format * @param locale Locale * @param def Default value if empty */ time(from?: string, to?: string, locale?: string, def?: string): ObjectColumnRender; } export interface DataTablesStaticUtil { /** * Normalise diacritic characters in a string. * * @param str String to have diacritic characters replaced * @param appendOriginal Append the original string to the result * (`true`) or not (`false`) * @returns Updated string */ diacritics(str: string, appendOriginal?: boolean): string; /** * Set the diacritic removal function * * @param replacement Removal function */ diacritics(replacement: (str: string, appendOriginal: boolean) => string): void; /** * Escape special characters in a regular expression string. Since: 1.10.4 * * @param str String to escape * @returns Escaped string */ escapeRegex(str: string): string; /** * Escape entities in a string. * * @param str String to have HTML entities escaped * @returns Sanitized string */ escapeHtml(str: string): string; /** * Set the HTML escaping function. * * @param escapeFunction Function to use for HTML escaping in DataTables. */ escapeHtml(escapeFunction: (str: string) => string): void; /** * Create a read function from a descriptor. Since 1.11 * @param source A descriptor that is used to define how to read the data from the source object. */ get<T=any, D=any>(source: string | number | object | Function | null): ((data: D, type?: string, val?: T, meta?: CellMetaSettings) => T); /** * Create a write function from a descriptor. Since 1.11 * @param source A descriptor that is used to define how to write data to a source object */ set<T=any, D=any>(source: string | number | object | Function | null): ((data: D, val: T, meta?: CellMetaSettings) => void); /** * Remove mark up from a string * * @param str String to have HTML tags stripped from. * @returns Stripped string */ stripHtml(str: string): string; /** * Set the HTML stripping function to be used by DataTables. * * @param stripFunction Function to use for HTML stripping in DataTables. */ stripHtml(stripFunction: (str: string) => string): void; /** * Throttle the calls to a method to reduce call frequency. Since: 1.10.3 * * @param fn Function * @param period ms * @returns Wrapper function that can be called and will automatically throttle calls to the passed in function to the given period. */ throttle(fn: (data: any) => void, period?: number): (() => void); /** * Get unique values from an array. * * @returns Array with unique values */ unique<T=any>(input: Array<T>): Array<T>; } export interface AjaxData { draw: number; start: number; length: number; data: any; order: AjaxDataOrder[]; columns: AjaxDataColumn[]; search: AjaxDataSearch; } export interface AjaxDataSearch { value: string; regex: boolean; } export interface AjaxDataOrder { column: number; dir: string; } export interface AjaxDataColumn { data: string | number; name: string; searchable: boolean; orderable: boolean; search: AjaxDataSearch; } export interface AjaxResponse { draw?: number; recordsTotal?: number; recordsFiltered?: number; data: any; error?: string; } export type AjaxDataSrc = string | ((data: any) => any[]); export interface AjaxSettings extends JQueryAjaxSettings { /** * Add or modify data submitted to the server upon an Ajax request. */ data?: object | FunctionAjaxData; /** * Data property or manipulation method for table data. */ dataSrc?: AjaxDataSrc | { /** Mapping for `data` property */ data: AjaxDataSrc; /** Mapping for `draw` property */ draw: AjaxDataSrc; /** Mapping for `recordsTotal` property */ recordsTotal: AjaxDataSrc; /** Mapping for `recordsFiltered` property */ recordsFiltered: AjaxDataSrc; }; } interface FunctionColumnData { (row: any, type: 'set', s: any, meta: CellMetaSettings): void; (row: any, type: 'display' | 'sort' | 'filter' | 'type', s: undefined, meta: CellMetaSettings): any; } interface ObjectColumnData { _: string | number | FunctionColumnData; filter?: string | number | FunctionColumnData; display?: string | number | FunctionColumnData; type?: string | number | FunctionColumnData; sort?: string | number | FunctionColumnData; } interface ObjectColumnRender { _?: string | number | FunctionColumnRender; filter?: string | number | FunctionColumnRender; display?: string | number | FunctionColumnRender; type?: string | number | FunctionColumnRender; sort?: string | number | FunctionColumnRender; } export interface CellMetaSettings { row: number; col: number; settings: InternalSettings; } export interface DataTablesStaticExt { builder: string; buttons: DataTablesStaticExtButtons; classes: ExtClassesSettings; errMode: string; feature: any[]; iApiIndex: number; internal: object; legacy: object; oApi: object; order: object; oSort: object; pager: object; renderer: object; search: any[]; selector: object; /** * Type based plug-ins. */ type: ExtTypeSettings; } export interface DataTablesStaticExtButtons { // Intentionally empty, completed in Buttons extension } /** * Classes used by DataTables. Used for styling integration. Note * that these all use legacy Hungarian notation. */ export interface ExtClassesSettings { /** Class to apply to the container for all DataTables controlled elements */ container: string; /** Empty row classes */ empty: { /** Empty row class */ row: string; }; /** Info feature classes */ info: { /** Container `<div>` class */ container: string; }; /** Length feature classes */ length: { /** Container `<div>` class */ container: string; /** Applied to the `<select>` element */ select: string; }; /** Table header cell ordering classes */ order: { /** Applied if a column can sort asc */ canAsc: string; /** Applied if a column can sort desc */ canDesc: string; /** Applied if a column is sorting asc */ isAsc: string; /** Applied if a column is sorting desc */ isDesc: string; /** Applied if there is no sorting available on a column */ none: string; /** Position class */ position: string; }; /** Processing indicator classes */ processing: { /** Container `<div>` class */ container: string; }; /** Classes for scrolling tables (`scrollX` / `scrollY`) */ scrolling: { /** Body table wrapper `<div>` */ body: string; /** Applied to the wrapper for all scrolling elements */ container: string; /** Scrolling footer classes */ footer: { /** Applied to the footer container `<div>` */ self: string; /** Applied to the inner footer `<div>` */ inner: string; }; /** Scrolling header classes */ header: { /** Applied to the header container `<div>` */ self: string; /** Applied to the inner header `<div>` */ inner: string; } }; /** Search feature classes */ search: { /** Applied to the search container `<div>` */ container: string; /** Class to add to `<input>` element */ input: string; }; /** Class to add to the `<table>` when DataTables is initialised on it */ table: string; /** Table body classes */ tbody: { /** Applied to all cells in the table body */ cell: string; /** Applied to all `tr` element in the table body */ row: string; }; /** Table header classes */ thead: { /** Applied to `td`/`th` elements in the table header */ cell: string; /** Applied to `tr` elements in the table header */ row: string; }; /** Table footer classes */ tfoot: { /** Applied to `td`/`th` elements in the footer */ cell: string; /** Applied to `tr` elements in the footer */ row: string; }; /** Paging feature classes */ paging: { /** Button shows the current page */ active: string; /** Applied to all buttons */ button: string; /** Container class */ container: string; /** Button unavailable class */ disabled: string; }; } export interface ExtTypeSettings { /** * Type detection functions for plug-in development. * * @see https://datatables.net/manual/plug-ins/type-detection */ detect: FunctionExtTypeSettingsDetect[]; /** * Type based ordering functions for plug-in development. * * @see https://datatables.net/manual/plug-ins/sorting * @default {} */ order: any; /** * Type based search formatting for plug-in development. * * @default {} * @example * $.fn.dataTable.ext.type.search['title-numeric'] = function ( d ) { * return d.replace(/\n/g," ").replace( /<.*?>/g, "" ); * } */ search: any; } /** * @param data Data from the column cell to be analysed. * @param DataTables settings object. */ type FunctionExtTypeSettingsDetect = (data: any, settings: InternalSettings) => (string | null); type FunctionAjax = (data: object, callback: ((data: any) => void), settings: InternalSettings) => void; type FunctionAjaxData = (data: AjaxData, settings: InternalSettings) => string | object; type FunctionColumnRender = (data: any, type: any, row: any, meta: CellMetaSettings) => any; type FunctionColumnCreatedCell = (cell: Node, cellData: any, rowData: any, row: number, col: number) => void; type FunctionCreateRow = (row: Node, data: any[] | object, dataIndex: number, cells: Node[]) => void; type FunctionDrawCallback = (settings: InternalSettings) => void; type FunctionFooterCallback = (tfoot: Node, data: any[], start: number, end: number, display: any[]) => void; type FunctionFormatNumber = (formatNumber: number) => void; type FunctionHeaderCallback = (thead: Node, data: any[], start: number, end: number, display: any[]) => void; type FunctionInfoCallback = (settings: InternalSettings, start: number, end: number, mnax: number, total: number, pre: string) => void; type FunctionInitComplete = (settings: InternalSettings, json: object) => void; type FunctionPreDrawCallback = (settings: InternalSettings) => void; type FunctionRowCallback = (row: Node, data: any[] | object, index: number) => void; type FunctionStateLoadCallback = (settings: InternalSettings) => void; type FunctionStateLoaded = (settings: InternalSettings, data: object) => void; type FunctionStateLoadParams = (settings: InternalSettings, data: object) => void; type FunctionStateSaveCallback = (settings: InternalSettings, data: object) => void; type FunctionStateSaveParams = (settings: InternalSettings, data: object) => void; node_modules/datatables.net/js/dataTables.min.js000064400000260432151676725350015742 0ustar00/*! DataTables 2.0.8 * © SpryMedia Ltd - datatables.net/license */ !function(n){"use strict";var a;"function"==typeof define&&define.amd?define(["jquery"],function(t){return n(t,window,document)}):"object"==typeof exports?(a=require("jquery"),"undefined"==typeof window?module.exports=function(t,e){return t=t||window,e=e||a(t),n(e,t,t.document)}:module.exports=n(a,window,window.document)):window.DataTable=n(jQuery,window,document)}(function(V,q,_){"use strict";function g(t){var e=parseInt(t,10);return!isNaN(e)&&isFinite(t)?e:null}function o(t,e,n){var a=typeof t,r="string"==a;return"number"==a||"bigint"==a||!!y(t)||(e&&r&&(t=R(t,e)),n&&r&&(t=t.replace(P,"")),!isNaN(parseFloat(t))&&isFinite(t))}function l(t,e,n){var a;return!!y(t)||("string"!=typeof t||!t.match(/<(input|select)/i))&&(y(a=t)||"string"==typeof a)&&!!o(I(t),e,n)||null}function v(t,e,n,a){var r=[],o=0,i=e.length;if(void 0!==a)for(;o<i;o++)t[e[o]][n]&&r.push(t[e[o]][n][a]);else for(;o<i;o++)t[e[o]]&&r.push(t[e[o]][n]);return r}function h(t,e){var n,a=[];void 0===e?(e=0,n=t):(n=e,e=t);for(var r=e;r<n;r++)a.push(r);return a}function b(t){for(var e=[],n=0,a=t.length;n<a;n++)t[n]&&e.push(t[n]);return e}var C,U,e,t,$=function(t,H){var W,X,B;return $.factory(t,H)?$:this instanceof $?V(t).DataTable(H):(X=void 0===(H=t),B=(W=this).length,X&&(H={}),this.api=function(){return new U(this)},this.each(function(){var n=1<B?Zt({},H,!0):H,a=0,t=this.getAttribute("id"),r=!1,e=$.defaults,o=V(this);if("table"!=this.nodeName.toLowerCase())Z(null,0,"Non-table node initialisation ("+this.nodeName+")",2);else{V(this).trigger("options.dt",n),nt(e),at(e.column),z(e,e,!0),z(e.column,e.column,!0),z(e,V.extend(n,o.data()),!0);for(var i=$.settings,a=0,l=i.length;a<l;a++){var s=i[a];if(s.nTable==this||s.nTHead&&s.nTHead.parentNode==this||s.nTFoot&&s.nTFoot.parentNode==this){var E=(void 0!==n.bRetrieve?n:e).bRetrieve,k=(void 0!==n.bDestroy?n:e).bDestroy;if(X||E)return s.oInstance;if(k){new $.Api(s).destroy();break}return void Z(s,0,"Cannot reinitialise DataTable",3)}if(s.sTableId==this.id){i.splice(a,1);break}}null!==t&&""!==t||(t="DataTables_Table_"+$.ext._unique++,this.id=t);var u=V.extend(!0,{},$.models.oSettings,{sDestroyWidth:o[0].style.width,sInstance:t,sTableId:t,colgroup:V("<colgroup>").prependTo(this),fastData:function(t,e,n){return G(u,t,e,n)}}),t=(u.nTable=this,u.oInit=n,i.push(u),u.api=new U(u),u.oInstance=1===W.length?W:o.dataTable(),nt(n),n.aLengthMenu&&!n.iDisplayLength&&(n.iDisplayLength=Array.isArray(n.aLengthMenu[0])?n.aLengthMenu[0][0]:V.isPlainObject(n.aLengthMenu[0])?n.aLengthMenu[0].value:n.aLengthMenu[0]),n=Zt(V.extend(!0,{},e),n),Q(u.oFeatures,n,["bPaginate","bLengthChange","bFilter","bSort","bSortMulti","bInfo","bProcessing","bAutoWidth","bSortClasses","bServerSide","bDeferRender"]),Q(u,n,["ajax","fnFormatNumber","sServerMethod","aaSorting","aaSortingFixed","aLengthMenu","sPaginationType","iStateDuration","bSortCellsTop","iTabIndex","sDom","fnStateLoadCallback","fnStateSaveCallback","renderer","searchDelay","rowId","caption","layout",["iCookieDuration","iStateDuration"],["oSearch","oPreviousSearch"],["aoSearchCols","aoPreSearchCols"],["iDisplayLength","_iDisplayLength"]]),Q(u.oScroll,n,[["sScrollX","sX"],["sScrollXInner","sXInner"],["sScrollY","sY"],["bScrollCollapse","bCollapse"]]),Q(u.oLanguage,n,"fnInfoCallback"),K(u,"aoDrawCallback",n.fnDrawCallback),K(u,"aoStateSaveParams",n.fnStateSaveParams),K(u,"aoStateLoadParams",n.fnStateLoadParams),K(u,"aoStateLoaded",n.fnStateLoaded),K(u,"aoRowCallback",n.fnRowCallback),K(u,"aoRowCreatedCallback",n.fnCreatedRow),K(u,"aoHeaderCallback",n.fnHeaderCallback),K(u,"aoFooterCallback",n.fnFooterCallback),K(u,"aoInitComplete",n.fnInitComplete),K(u,"aoPreDrawCallback",n.fnPreDrawCallback),u.rowIdFn=J(n.rowId),u),c=($.__browser||(P={},$.__browser=P,j=V("<div/>").css({position:"fixed",top:0,left:-1*q.pageXOffset,height:1,width:1,overflow:"hidden"}).append(V("<div/>").css({position:"absolute",top:1,left:1,width:100,overflow:"scroll"}).append(V("<div/>").css({width:"100%",height:10}))).appendTo("body"),p=j.children(),O=p.children(),P.barWidth=p[0].offsetWidth-p[0].clientWidth,P.bScrollbarLeft=1!==Math.round(O.offset().left),j.remove()),V.extend(t.oBrowser,$.__browser),t.oScroll.iBarWidth=$.__browser.barWidth,u.oClasses),d=(V.extend(c,$.ext.classes,n.oClasses),o.addClass(c.table),u.oFeatures.bPaginate||(n.iDisplayStart=0),void 0===u.iInitDisplayStart&&(u.iInitDisplayStart=n.iDisplayStart,u._iDisplayStart=n.iDisplayStart),u.oLanguage),f=(V.extend(!0,d,n.oLanguage),d.sUrl?(V.ajax({dataType:"json",url:d.sUrl,success:function(t){z(e.oLanguage,t),V.extend(!0,d,t,u.oInit.oLanguage),tt(u,null,"i18n",[u],!0),Et(u)},error:function(){Z(u,0,"i18n file loading error",21),Et(u)}}),r=!0):tt(u,null,"i18n",[u]),[]),h=this.getElementsByTagName("thead"),p=It(u,h[0]);if(n.aoColumns)f=n.aoColumns;else if(p.length)for(l=p[a=0].length;a<l;a++)f.push(null);for(a=0,l=f.length;a<l;a++)rt(u);var g,m,v,b,y,D,x,S=u,T=n.aoColumnDefs,w=f,M=p,_=function(t,e){ot(u,t,e)},C=S.aoColumns;if(w)for(g=0,m=w.length;g<m;g++)w[g]&&w[g].name&&(C[g].sName=w[g].name);if(T)for(g=T.length-1;0<=g;g--){var I=void 0!==(x=T[g]).target?x.target:void 0!==x.targets?x.targets:x.aTargets;for(Array.isArray(I)||(I=[I]),v=0,b=I.length;v<b;v++){var A=I[v];if("number"==typeof A&&0<=A){for(;C.length<=A;)rt(S);_(A,x)}else if("number"==typeof A&&A<0)_(C.length+A,x);else if("string"==typeof A)for(y=0,D=C.length;y<D;y++)"_all"===A?_(y,x):-1!==A.indexOf(":name")?C[y].sName===A.replace(":name","")&&_(y,x):M.forEach(function(t){t[y]&&(t=V(t[y].cell),A.match(/^[a-z][\w-]*$/i)&&(A="."+A),t.is(A))&&_(y,x)})}}if(w)for(g=0,m=w.length;g<m;g++)_(g,w[g]);var L,F,N,j,P=o.children("tbody").find("tr").eq(0),R=(P.length&&(L=function(t,e){return null!==t.getAttribute("data-"+e)?e:null},V(P[0]).children("th, td").each(function(t,e){var n,a=u.aoColumns[t];a||Z(u,0,"Incorrect column count",18),a.mData===t&&(n=L(e,"sort")||L(e,"order"),e=L(e,"filter")||L(e,"search"),null===n&&null===e||(a.mData={_:t+".display",sort:null!==n?t+".@data-"+n:void 0,type:null!==n?t+".@data-"+n:void 0,filter:null!==e?t+".@data-"+e:void 0},a._isArrayHost=!0,ot(u,t)))})),u.oFeatures),O=function(){if(void 0===n.aaSorting){var t=u.aaSorting;for(a=0,l=t.length;a<l;a++)t[a][1]=u.aoColumns[a].asSorting[0]}Yt(u),K(u,"aoDrawCallback",function(){(u.bSorted||"ssp"===et(u)||R.bDeferRender)&&Yt(u)});var e=o.children("caption"),e=(u.caption&&(e=0===e.length?V("<caption/>").appendTo(o):e).html(u.caption),e.length&&(e[0]._captionSide=e.css("caption-side"),u.captionNode=e[0]),0===h.length&&(h=V("<thead/>").appendTo(o)),u.nTHead=h[0],V("tr",h).addClass(c.thead.row),o.children("tbody")),e=(0===e.length&&(e=V("<tbody/>").insertAfter(h)),u.nTBody=e[0],o.children("tfoot"));if(0===e.length&&(e=V("<tfoot/>").appendTo(o)),u.nTFoot=e[0],V("tr",e).addClass(c.tfoot.row),n.aaData)for(a=0;a<n.aaData.length;a++)Y(u,n.aaData[a]);else"dom"==et(u)&&ut(u,V(u.nTBody).children("tr"));u.aiDisplay=u.aiDisplayMaster.slice(),!(u.bInitialised=!0)===r&&Et(u)};K(u,"aoDrawCallback",Gt),n.bStateSave?(R.bStateSave=!0,N=O,(F=u).oFeatures.bStateSave?void 0!==(j=F.fnStateLoadCallback.call(F.oInstance,F,function(t){Jt(F,t,N)}))&&Jt(F,j,N):N()):O()}}),W=null,this)},c=($.ext=C={buttons:{},classes:{},builder:"-source-",errMode:"alert",feature:[],features:{},search:[],selector:{cell:[],column:[],row:[]},legacy:{ajax:null},pager:{},renderer:{pageButton:{},header:{}},order:{},type:{className:{},detect:[],render:{},search:{},order:{}},_unique:0,fnVersionCheck:$.fnVersionCheck,iApiIndex:0,sVersion:$.version},V.extend(C,{afnFiltering:C.search,aTypes:C.type.detect,ofnSearch:C.type.search,oSort:C.type.order,afnSortData:C.order,aoFeatures:C.feature,oStdClasses:C.classes,oPagination:C.pager}),V.extend($.ext.classes,{container:"dt-container",empty:{row:"dt-empty"},info:{container:"dt-info"},length:{container:"dt-length",select:"dt-input"},order:{canAsc:"dt-orderable-asc",canDesc:"dt-orderable-desc",isAsc:"dt-ordering-asc",isDesc:"dt-ordering-desc",none:"dt-orderable-none",position:"sorting_"},processing:{container:"dt-processing"},scrolling:{body:"dt-scroll-body",container:"dt-scroll",footer:{self:"dt-scroll-foot",inner:"dt-scroll-footInner"},header:{self:"dt-scroll-head",inner:"dt-scroll-headInner"}},search:{container:"dt-search",input:"dt-input"},table:"dataTable",tbody:{cell:"",row:""},thead:{cell:"",row:""},tfoot:{cell:"",row:""},paging:{active:"current",button:"dt-paging-button",container:"dt-paging",disabled:"disabled"}}),{}),d=/[\r\n\u2028]/g,L=/<([^>]*>)/g,F=Math.pow(2,28),N=/^\d{2,4}[./-]\d{1,2}[./-]\d{1,2}([T ]{1}\d{1,2}[:.]\d{2}([.:]\d{2})?)?$/,j=new RegExp("(\\"+["/",".","*","+","?","|","(",")","[","]","{","}","\\","$","^","-"].join("|\\")+")","g"),P=/['\u00A0,$£€¥%\u2009\u202F\u20BD\u20a9\u20BArfkɃΞ]/gi,y=function(t){return!t||!0===t||"-"===t},R=function(t,e){return c[e]||(c[e]=new RegExp(Pt(e),"g")),"string"==typeof t&&"."!==e?t.replace(/\./g,"").replace(c[e],"."):t},f=function(t,e,n){var a=[],r=0,o=t.length;if(void 0!==n)for(;r<o;r++)t[r]&&t[r][e]&&a.push(t[r][e][n]);else for(;r<o;r++)t[r]&&a.push(t[r][e]);return a},I=function(t){if(t.length>F)throw new Error("Exceeded max str len");var e;for(t=t.replace(L,"");(t=(e=t).replace(/<script/i,""))!==e;);return e},u=function(t){return"string"==typeof(t=Array.isArray(t)?t.join(","):t)?t.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,"""):t},O=function(t,e){var n;return"string"!=typeof t?t:(n=t.normalize("NFD")).length!==t.length?(!0===e?t+" ":"")+n.replace(/[\u0300-\u036f]/g,""):n},x=function(t){if(Array.from&&Set)return Array.from(new Set(t));if(function(t){if(!(t.length<2))for(var e=t.slice().sort(),n=e[0],a=1,r=e.length;a<r;a++){if(e[a]===n)return!1;n=e[a]}return!0}(t))return t.slice();var e,n,a,r=[],o=t.length,i=0;t:for(n=0;n<o;n++){for(e=t[n],a=0;a<i;a++)if(r[a]===e)continue t;r.push(e),i++}return r},E=function(t,e){if(Array.isArray(e))for(var n=0;n<e.length;n++)E(t,e[n]);else t.push(e);return t};function D(e,t){t&&t.split(" ").forEach(function(t){t&&e.classList.add(t)})}function k(e){var n,a,r={};V.each(e,function(t){(n=t.match(/^([^A-Z]+?)([A-Z])/))&&-1!=="a aa ai ao as b fn i m o s ".indexOf(n[1]+" ")&&(a=t.replace(n[0],n[2].toLowerCase()),r[a]=t,"o"===n[1])&&k(e[t])}),e._hungarianMap=r}function z(e,n,a){var r;e._hungarianMap||k(e),V.each(n,function(t){void 0===(r=e._hungarianMap[t])||!a&&void 0!==n[r]||("o"===r.charAt(0)?(n[r]||(n[r]={}),V.extend(!0,n[r],n[t]),z(e[r],n[r],a)):n[r]=n[t])})}$.util={diacritics:function(t,e){if("function"!=typeof t)return O(t,e);O=t},debounce:function(n,a){var r;return function(){var t=this,e=arguments;clearTimeout(r),r=setTimeout(function(){n.apply(t,e)},a||250)}},throttle:function(a,t){var r,o,i=void 0!==t?t:200;return function(){var t=this,e=+new Date,n=arguments;r&&e<r+i?(clearTimeout(o),o=setTimeout(function(){r=void 0,a.apply(t,n)},i)):(r=e,a.apply(t,n))}},escapeRegex:function(t){return t.replace(j,"\\$1")},set:function(a){var f;return V.isPlainObject(a)?$.util.set(a._):null===a?function(){}:"function"==typeof a?function(t,e,n){a(t,"set",e,n)}:"string"!=typeof a||-1===a.indexOf(".")&&-1===a.indexOf("[")&&-1===a.indexOf("(")?function(t,e){t[a]=e}:(f=function(t,e,n){for(var a,r,o,i,l=ft(n),n=l[l.length-1],s=0,u=l.length-1;s<u;s++){if("__proto__"===l[s]||"constructor"===l[s])throw new Error("Cannot set prototype values");if(a=l[s].match(dt),r=l[s].match(p),a){if(l[s]=l[s].replace(dt,""),t[l[s]]=[],(a=l.slice()).splice(0,s+1),i=a.join("."),Array.isArray(e))for(var c=0,d=e.length;c<d;c++)f(o={},e[c],i),t[l[s]].push(o);else t[l[s]]=e;return}r&&(l[s]=l[s].replace(p,""),t=t[l[s]](e)),null!==t[l[s]]&&void 0!==t[l[s]]||(t[l[s]]={}),t=t[l[s]]}n.match(p)?t[n.replace(p,"")](e):t[n.replace(dt,"")]=e},function(t,e){return f(t,e,a)})},get:function(r){var o,f;return V.isPlainObject(r)?(o={},V.each(r,function(t,e){e&&(o[t]=$.util.get(e))}),function(t,e,n,a){var r=o[e]||o._;return void 0!==r?r(t,e,n,a):t}):null===r?function(t){return t}:"function"==typeof r?function(t,e,n,a){return r(t,e,n,a)}:"string"!=typeof r||-1===r.indexOf(".")&&-1===r.indexOf("[")&&-1===r.indexOf("(")?function(t){return t[r]}:(f=function(t,e,n){var a,r,o;if(""!==n)for(var i=ft(n),l=0,s=i.length;l<s;l++){if(d=i[l].match(dt),a=i[l].match(p),d){if(i[l]=i[l].replace(dt,""),""!==i[l]&&(t=t[i[l]]),r=[],i.splice(0,l+1),o=i.join("."),Array.isArray(t))for(var u=0,c=t.length;u<c;u++)r.push(f(t[u],e,o));var d=d[0].substring(1,d[0].length-1);t=""===d?r:r.join(d);break}if(a)i[l]=i[l].replace(p,""),t=t[i[l]]();else{if(null===t||null===t[i[l]])return null;if(void 0===t||void 0===t[i[l]])return;t=t[i[l]]}}return t},function(t,e){return f(t,e,r)})},stripHtml:function(t){var e=typeof t;if("function"!=e)return"string"==e?I(t):t;I=t},escapeHtml:function(t){var e=typeof t;if("function"!=e)return"string"==e||Array.isArray(t)?u(t):t;u=t},unique:x};var r=function(t,e,n){void 0!==t[e]&&(t[n]=t[e])};function nt(t){r(t,"ordering","bSort"),r(t,"orderMulti","bSortMulti"),r(t,"orderClasses","bSortClasses"),r(t,"orderCellsTop","bSortCellsTop"),r(t,"order","aaSorting"),r(t,"orderFixed","aaSortingFixed"),r(t,"paging","bPaginate"),r(t,"pagingType","sPaginationType"),r(t,"pageLength","iDisplayLength"),r(t,"searching","bFilter"),"boolean"==typeof t.sScrollX&&(t.sScrollX=t.sScrollX?"100%":""),"boolean"==typeof t.scrollX&&(t.scrollX=t.scrollX?"100%":"");var e=t.aoSearchCols;if(e)for(var n=0,a=e.length;n<a;n++)e[n]&&z($.models.oSearch,e[n]);t.serverSide&&!t.searchDelay&&(t.searchDelay=400)}function at(t){r(t,"orderable","bSortable"),r(t,"orderData","aDataSort"),r(t,"orderSequence","asSorting"),r(t,"orderDataType","sortDataType");var e=t.aDataSort;"number"!=typeof e||Array.isArray(e)||(t.aDataSort=[e])}function rt(t){var e=$.defaults.column,n=t.aoColumns.length,e=V.extend({},$.models.oColumn,e,{aDataSort:e.aDataSort||[n],mData:e.mData||n,idx:n,searchFixed:{},colEl:V("<col>").attr("data-dt-column",n)}),e=(t.aoColumns.push(e),t.aoPreSearchCols);e[n]=V.extend({},$.models.oSearch,e[n])}function ot(t,e,n){function a(t){return"string"==typeof t&&-1!==t.indexOf("@")}var r=t.aoColumns[e],o=(null!=n&&(at(n),z($.defaults.column,n,!0),void 0===n.mDataProp||n.mData||(n.mData=n.mDataProp),n.sType&&(r._sManualType=n.sType),n.className&&!n.sClass&&(n.sClass=n.className),e=r.sClass,V.extend(r,n),Q(r,n,"sWidth","sWidthOrig"),e!==r.sClass&&(r.sClass=e+" "+r.sClass),void 0!==n.iDataSort&&(r.aDataSort=[n.iDataSort]),Q(r,n,"aDataSort")),r.mData),i=J(o);r.mRender&&Array.isArray(r.mRender)&&(n=(e=r.mRender.slice()).shift(),r.mRender=$.render[n].apply(q,e)),r._render=r.mRender?J(r.mRender):null;r._bAttrSrc=V.isPlainObject(o)&&(a(o.sort)||a(o.type)||a(o.filter)),r._setter=null,r.fnGetData=function(t,e,n){var a=i(t,e,void 0,n);return r._render&&e?r._render(a,e,t,n):a},r.fnSetData=function(t,e,n){return m(o)(t,e,n)},"number"==typeof o||r._isArrayHost||(t._rowReadObject=!0),t.oFeatures.bSort||(r.bSortable=!1)}function M(t){var e=t;if(e.oFeatures.bAutoWidth){var n,a,r=e.nTable,o=e.aoColumns,i=e.oScroll,l=i.sY,s=i.sX,i=i.sXInner,u=X(e,"bVisible"),c=r.getAttribute("width"),d=r.parentNode,f=r.style.width,f=(f&&-1!==f.indexOf("%")&&(c=f),tt(e,null,"column-calc",{visible:u},!1),V(r.cloneNode()).css("visibility","hidden").removeAttr("id")),h=(f.append("<tbody>"),V("<tr/>").appendTo(f.find("tbody")));for(f.append(V(e.nTHead).clone()).append(V(e.nTFoot).clone()),f.find("tfoot th, tfoot td").css("width",""),f.find("thead th, thead td").each(function(){var t=lt(e,this,!0,!1);t?(this.style.width=t,s&&V(this).append(V("<div/>").css({width:t,margin:0,padding:0,border:0,height:1}))):this.style.width=""}),n=0;n<u.length;n++){p=u[n],a=o[p];var p=function(t,e){var n=t.aoColumns[e];if(!n.maxLenString){for(var a,r="",o=-1,i=0,l=t.aiDisplayMaster.length;i<l;i++){var s=t.aiDisplayMaster[i],s=vt(t,s)[e],s=s&&"object"==typeof s&&s.nodeType?s.innerHTML:s+"";s=s.replace(/id=".*?"/g,"").replace(/name=".*?"/g,""),(a=I(s).replace(/ /g," ")).length>o&&(r=s,o=a.length)}n.maxLenString=r}return n.maxLenString}(e,p),g=C.type.className[a.sType],m=p+a.sContentPadding,p=-1===p.indexOf("<")?_.createTextNode(m):m;V("<td/>").addClass(g).addClass(a.sClass).append(p).appendTo(h)}V("[name]",f).removeAttr("name");var v=V("<div/>").css(s||l?{position:"absolute",top:0,left:0,height:1,right:0,overflow:"hidden"}:{}).append(f).appendTo(d),b=(s&&i?f.width(i):s?(f.css("width","auto"),f.removeAttr("width"),f.width()<d.clientWidth&&c&&f.width(d.clientWidth)):l?f.width(d.clientWidth):c&&f.width(c),0),y=f.find("tbody tr").eq(0).children();for(n=0;n<u.length;n++){var D=y[n].getBoundingClientRect().width;b+=D,o[u[n]].sWidth=A(D)}r.style.width=A(b),v.remove(),c&&(r.style.width=A(c)),!c&&!s||e._reszEvt||(V(q).on("resize.DT-"+e.sInstance,$.util.throttle(function(){e.bDestroying||M(e)})),e._reszEvt=!0)}for(var x=t,S=x.aoColumns,T=0;T<S.length;T++){var w=lt(x,[T],!1,!1);S[T].colEl.css("width",w)}i=t.oScroll;""===i.sY&&""===i.sX||Xt(t),tt(t,null,"column-sizing",[t])}function H(t,e){t=X(t,"bVisible");return"number"==typeof t[e]?t[e]:null}function T(t,e){t=X(t,"bVisible").indexOf(e);return-1!==t?t:null}function W(t){var e=t.aoHeader,n=t.aoColumns,a=0;if(e.length)for(var r=0,o=e[0].length;r<o;r++)n[r].bVisible&&"none"!==V(e[0][r].cell).css("display")&&a++;return a}function X(t,n){var a=[];return t.aoColumns.map(function(t,e){t[n]&&a.push(e)}),a}function B(t){for(var e,n,a,r,o,i,l,s=t.aoColumns,u=t.aoData,c=$.ext.type.detect,d=0,f=s.length;d<f;d++){if(l=[],!(o=s[d]).sType&&o._sManualType)o.sType=o._sManualType;else if(!o.sType){for(e=0,n=c.length;e<n;e++){for(a=0,r=u.length;a<r;a++)if(u[a]){if(void 0===l[a]&&(l[a]=G(t,a,d,"type")),!(i=c[e](l[a],t))&&e!==c.length-2)break;if("html"===i&&!y(l[a]))break}if(i){o.sType=i;break}}o.sType||(o.sType="string")}var h=C.type.className[o.sType],h=(h&&(it(t.aoHeader,d,h),it(t.aoFooter,d,h)),C.type.render[o.sType]);if(h&&!o._render){o._render=$.util.get(h),p=b=v=m=g=void 0;for(var p,g=t,m=d,v=g.aoData,b=0;b<v.length;b++)v[b].nTr&&(p=G(g,b,m,"display"),v[b].displayData[m]=p,ct(v[b].anCells[m],p))}}}function it(t,e,n){t.forEach(function(t){t[e]&&t[e].unique&&D(t[e].cell,n)})}function lt(t,e,n,a){Array.isArray(e)||(e=st(e));for(var r,o=0,i=t.aoColumns,l=0,s=e.length;l<s;l++){var u=i[e[l]],c=n?u.sWidthOrig:u.sWidth;if(a||!1!==u.bVisible){if(null==c)return null;"number"==typeof c?(r="px",o+=c):(u=c.match(/([\d\.]+)([^\d]*)/))&&(o+=+u[1],r=3===u.length?u[2]:"px")}}return o+r}function st(t){t=V(t).closest("[data-dt-column]").attr("data-dt-column");return t?t.split(",").map(function(t){return+t}):[]}function Y(t,e,n,a){for(var r=t.aoData.length,o=V.extend(!0,{},$.models.oRow,{src:n?"dom":"data",idx:r}),i=(o._aData=e,t.aoData.push(o),t.aoColumns),l=0,s=i.length;l<s;l++)i[l].sType=null;t.aiDisplayMaster.push(r);e=t.rowIdFn(e);return void 0!==e&&(t.aIds[e]=o),!n&&t.oFeatures.bDeferRender||bt(t,r,n,a),r}function ut(n,t){var a;return(t=t instanceof V?t:V(t)).map(function(t,e){return a=mt(n,e),Y(n,a.data,e,a.cells)})}function G(t,e,n,a){"search"===a?a="filter":"order"===a&&(a="sort");var r=t.aoData[e];if(r){var o=t.iDraw,i=t.aoColumns[n],r=r._aData,l=i.sDefaultContent,s=i.fnGetData(r,a,{settings:t,row:e,col:n});if(void 0===(s="display"!==a&&s&&"object"==typeof s&&s.nodeName?s.innerHTML:s))return t.iDrawError!=o&&null===l&&(Z(t,0,"Requested unknown parameter "+("function"==typeof i.mData?"{function}":"'"+i.mData+"'")+" for row "+e+", column "+n,4),t.iDrawError=o),l;if(s!==r&&null!==s||null===l||void 0===a){if("function"==typeof s)return s.call(r)}else s=l;return null===s&&"display"===a?"":s="filter"===a&&(e=$.ext.type.search)[i.sType]?e[i.sType](s):s}}function ct(t,e){e&&"object"==typeof e&&e.nodeName?V(t).empty().append(e):t.innerHTML=e}var dt=/\[.*?\]$/,p=/\(\)$/;function ft(t){return(t.match(/(\\.|[^.])+/g)||[""]).map(function(t){return t.replace(/\\\./g,".")})}var J=$.util.get,m=$.util.set;function ht(t){return f(t.aoData,"_aData")}function pt(t){t.aoData.length=0,t.aiDisplayMaster.length=0,t.aiDisplay.length=0,t.aIds={}}function gt(t,e,n,a){var r,o,i=t.aoData[e];if(i._aSortData=null,i._aFilterData=null,i.displayData=null,"dom"!==n&&(n&&"auto"!==n||"dom"!==i.src)){var l=i.anCells,s=vt(t,e);if(l)if(void 0!==a)ct(l[a],s[a]);else for(r=0,o=l.length;r<o;r++)ct(l[r],s[r])}else i._aData=mt(t,i,a,void 0===a?void 0:i._aData).data;var u=t.aoColumns;if(void 0!==a)u[a].sType=null,u[a].maxLenString=null;else{for(r=0,o=u.length;r<o;r++)u[r].sType=null,u[r].maxLenString=null;yt(t,i)}}function mt(t,e,n,a){function r(t,e){var n;"string"==typeof t&&-1!==(n=t.indexOf("@"))&&(n=t.substring(n+1),m(t)(a,e.getAttribute(n)))}function o(t){void 0!==n&&n!==d||(l=f[d],s=t.innerHTML.trim(),l&&l._bAttrSrc?(m(l.mData._)(a,s),r(l.mData.sort,t),r(l.mData.type,t),r(l.mData.filter,t)):h?(l._setter||(l._setter=m(l.mData)),l._setter(a,s)):a[d]=s),d++}var i,l,s,u=[],c=e.firstChild,d=0,f=t.aoColumns,h=t._rowReadObject;a=void 0!==a?a:h?{}:[];if(c)for(;c;)"TD"!=(i=c.nodeName.toUpperCase())&&"TH"!=i||(o(c),u.push(c)),c=c.nextSibling;else for(var p=0,g=(u=e.anCells).length;p<g;p++)o(u[p]);var e=e.firstChild?e:e.nTr;return e&&(e=e.getAttribute("id"))&&m(t.rowId)(a,e),{data:a,cells:u}}function vt(t,e){var n=t.aoData[e],a=t.aoColumns;if(!n.displayData){n.displayData=[];for(var r=0,o=a.length;r<o;r++)n.displayData.push(G(t,e,r,"display"))}return n.displayData}function bt(t,e,n,a){var r,o,i,l,s,u,c=t.aoData[e],d=c._aData,f=[],h=t.oClasses.tbody.row;if(null===c.nTr){for(r=n||_.createElement("tr"),c.nTr=r,c.anCells=f,D(r,h),r._DT_RowIndex=e,yt(t,c),l=0,s=t.aoColumns.length;l<s;l++){i=t.aoColumns[l],(o=(u=!n||!a[l])?_.createElement(i.sCellType):a[l])||Z(t,0,"Incorrect column count",18),o._DT_CellIndex={row:e,column:l},f.push(o);var p=vt(t,e);!u&&(!i.mRender&&i.mData===l||V.isPlainObject(i.mData)&&i.mData._===l+".display")||ct(o,p[l]),i.bVisible&&u?r.appendChild(o):i.bVisible||u||o.parentNode.removeChild(o),i.fnCreatedCell&&i.fnCreatedCell.call(t.oInstance,o,G(t,e,l),d,e,l)}tt(t,"aoRowCreatedCallback","row-created",[r,d,e,f])}else D(c.nTr,h)}function yt(t,e){var n=e.nTr,a=e._aData;n&&((t=t.rowIdFn(a))&&(n.id=t),a.DT_RowClass&&(t=a.DT_RowClass.split(" "),e.__rowc=e.__rowc?x(e.__rowc.concat(t)):t,V(n).removeClass(e.__rowc.join(" ")).addClass(a.DT_RowClass)),a.DT_RowAttr&&V(n).attr(a.DT_RowAttr),a.DT_RowData)&&V(n).data(a.DT_RowData)}function Dt(t,e){var n,a=t.oClasses,r=t.aoColumns,o="header"===e?t.nTHead:t.nTFoot,i="header"===e?"sTitle":e;if(o){if(("header"===e||f(t.aoColumns,i).join(""))&&1===(n=(n=V("tr",o)).length?n:V("<tr/>").appendTo(o)).length)for(var l=V("td, th",n).length,s=r.length;l<s;l++)V("<th/>").html(r[l][i]||"").appendTo(n);var u=It(t,o,!0);"header"===e?t.aoHeader=u:t.aoFooter=u,V(o).children("tr").attr("role","row"),V(o).children("tr").children("th, td").each(function(){te(t,e)(t,V(this),a)})}}function xt(t,e,n){var a,r,o,i,l,s=[],u=[],c=t.aoColumns,t=c.length;if(e){for(n=n||h(t).filter(function(t){return c[t].bVisible}),a=0;a<e.length;a++)s[a]=e[a].slice().filter(function(t,e){return n.includes(e)}),u.push([]);for(a=0;a<s.length;a++)for(r=0;r<s[a].length;r++)if(l=i=1,void 0===u[a][r]){for(o=s[a][r].cell;void 0!==s[a+i]&&s[a][r].cell==s[a+i][r].cell;)u[a+i][r]=null,i++;for(;void 0!==s[a][r+l]&&s[a][r].cell==s[a][r+l].cell;){for(var d=0;d<i;d++)u[a+d][r+l]=null;l++}var f=V("span.dt-column-title",o);u[a][r]={cell:o,colspan:l,rowspan:i,title:(f.length?f:V(o)).html()}}return u}}function St(t,e){for(var n,a,r=xt(t,e),o=0;o<e.length;o++){if(n=e[o].row)for(;a=n.firstChild;)n.removeChild(a);for(var i=0;i<r[o].length;i++){var l=r[o][i];l&&V(l.cell).appendTo(n).attr("rowspan",l.rowspan).attr("colspan",l.colspan)}}}function S(t,e){if(r="ssp"==et(s=t),void 0!==(i=s.iInitDisplayStart)&&-1!==i&&(s._iDisplayStart=!r&&i>=s.fnRecordsDisplay()?0:i,s.iInitDisplayStart=-1),-1!==tt(t,"aoPreDrawCallback","preDraw",[t]).indexOf(!1))w(t,!1);else{var l,n=[],a=0,r="ssp"==et(t),o=t.aiDisplay,i=t._iDisplayStart,s=t.fnDisplayEnd(),u=t.aoColumns,c=V(t.nTBody);if(t.bDrawing=!0,r){if(!t.bDestroying&&!e)return 0===t.iDraw&&c.empty().append(Tt(t)),(l=t).iDraw++,w(l,!0),void At(l,function(e){function n(t,e){return"function"==typeof a[t][e]?"function":a[t][e]}var a=e.aoColumns,t=e.oFeatures,r=e.oPreviousSearch,o=e.aoPreSearchCols;return{draw:e.iDraw,columns:a.map(function(e,t){return{data:n(t,"mData"),name:e.sName,searchable:e.bSearchable,orderable:e.bSortable,search:{value:o[t].search,regex:o[t].regex,fixed:Object.keys(e.searchFixed).map(function(t){return{name:t,term:e.searchFixed[t].toString()}})}}}),order:$t(e).map(function(t){return{column:t.col,dir:t.dir,name:n(t.col,"sName")}}),start:e._iDisplayStart,length:t.bPaginate?e._iDisplayLength:-1,search:{value:r.search,regex:r.regex,fixed:Object.keys(e.searchFixed).map(function(t){return{name:t,term:e.searchFixed[t].toString()}})}}}(l),function(t){var e=l,n=Lt(e,t=t),a=Ft(e,"draw",t),r=Ft(e,"recordsTotal",t),t=Ft(e,"recordsFiltered",t);if(void 0!==a){if(+a<e.iDraw)return;e.iDraw=+a}n=n||[],pt(e),e._iRecordsTotal=parseInt(r,10),e._iRecordsDisplay=parseInt(t,10);for(var o=0,i=n.length;o<i;o++)Y(e,n[o]);e.aiDisplay=e.aiDisplayMaster.slice(),S(e,!0),kt(e),w(e,!1)})}else t.iDraw++;if(0!==o.length)for(var d=r?t.aoData.length:s,f=r?0:i;f<d;f++){for(var h=o[f],p=t.aoData[h],g=(null===p.nTr&&bt(t,h),p.nTr),m=0;m<u.length;m++){var v=u[m],b=p.anCells[m];D(b,C.type.className[v.sType]),D(b,v.sClass),D(b,t.oClasses.tbody.cell)}tt(t,"aoRowCallback",null,[g,p._aData,a,f,h]),n.push(g),a++}else n[0]=Tt(t);tt(t,"aoHeaderCallback","header",[V(t.nTHead).children("tr")[0],ht(t),i,s,o]),tt(t,"aoFooterCallback","footer",[V(t.nTFoot).children("tr")[0],ht(t),i,s,o]),c[0].replaceChildren?c[0].replaceChildren.apply(c[0],n):(c.children().detach(),c.append(V(n))),V(t.nTableWrapper).toggleClass("dt-empty-footer",0===V("tr",t.nTFoot).length),tt(t,"aoDrawCallback","draw",[t],!0),t.bSorted=!1,t.bFiltered=!1,t.bDrawing=!1}}function s(t,e,n){var a=t.oFeatures,r=a.bSort,a=a.bFilter;void 0!==n&&!0!==n||(r&&zt(t),a?Nt(t,t.oPreviousSearch):t.aiDisplay=t.aiDisplayMaster.slice()),!0!==e&&(t._iDisplayStart=0),t._drawHold=e,S(t),t._drawHold=!1}function Tt(t){var e=t.oLanguage,n=e.sZeroRecords,a=et(t);return t.iDraw<1&&"ssp"===a||t.iDraw<=1&&"ajax"===a?n=e.sLoadingRecords:e.sEmptyTable&&0===t.fnRecordsTotal()&&(n=e.sEmptyTable),V("<tr/>").append(V("<td />",{colSpan:W(t),class:t.oClasses.empty.row}).html(n))[0]}function wt(t,e,n){for(var i={},a=(V.each(e,function(t,e){if(null!==e){var t=t.replace(/([A-Z])/g," $1").split(" "),n=(i[t[0]]||(i[t[0]]={}),1===t.length?"full":t[1].toLowerCase()),a=i[t[0]],r=function(e,n){V.isPlainObject(n)?Object.keys(n).map(function(t){e.push({feature:t,opts:n[t]})}):e.push(n)};if(a[n]&&a[n].contents||(a[n]={contents:[]}),Array.isArray(e))for(var o=0;o<e.length;o++)r(a[n].contents,e[o]);else r(a[n].contents,e);Array.isArray(a[n].contents)||(a[n].contents=[a[n].contents])}}),Object.keys(i).map(function(t){return 0!==t.indexOf(n)?null:{name:t,val:i[t]}}).filter(function(t){return null!==t})),r=(a.sort(function(t,e){t=+t.name.replace(/[^0-9]/g,"");return+e.name.replace(/[^0-9]/g,"")-t}),"bottom"===n&&a.reverse(),[]),o=0,l=a.length;o<l;o++)a[o].val.full&&(r.push({full:a[o].val.full}),_t(t,r[r.length-1]),delete a[o].val.full),Object.keys(a[o].val).length&&(r.push(a[o].val),_t(t,r[r.length-1]));return r}function _t(o,i){function l(t,e){return C.features[t]||Z(o,0,"Unknown feature: "+t),C.features[t].apply(this,[o,e])}V.each(i,function(t){for(var e,n=i[t].contents,a=0,r=n.length;a<r;a++)n[a]&&("string"==typeof n[a]?n[a]=l(n[a],null):V.isPlainObject(n[a])?n[a]=l(n[a].feature,n[a].opts):"function"==typeof n[a].node?n[a]=n[a].node(o):"function"==typeof n[a]&&(e=n[a](o),n[a]="function"==typeof e.node?e.node():e))})}function Ct(e){var a,t=e.oClasses,n=V(e.nTable),r=V("<div/>").attr({id:e.sTableId+"_wrapper",class:t.container}).insertBefore(n);if(e.nTableWrapper=r[0],e.sDom)for(var o,i,l,s,u,c,d=e,t=e.sDom,f=r,h=t.match(/(".*?")|('.*?')|./g),p=0;p<h.length;p++)o=null,"<"==(i=h[p])?(l=V("<div/>"),"'"!=(s=h[p+1])[0]&&'"'!=s[0]||(s=s.replace(/['"]/g,""),u="",-1!=s.indexOf(".")?(c=s.split("."),u=c[0],c=c[1]):"#"==s[0]?u=s:c=s,l.attr("id",u.substring(1)).addClass(c),p++),f.append(l),f=l):">"==i?f=f.parent():"t"==i?o=Wt(d):$.ext.feature.forEach(function(t){i==t.cFeature&&(o=t.fnInit(d))}),o&&f.append(o);else{var n=wt(e,e.layout,"top"),t=wt(e,e.layout,"bottom"),g=te(e,"layout");n.forEach(function(t){g(e,r,t)}),g(e,r,{full:{table:!0,contents:[Wt(e)]}}),t.forEach(function(t){g(e,r,t)})}var n=e,t=n.nTable,m=""!==n.oScroll.sX||""!==n.oScroll.sY;n.oFeatures.bProcessing&&(a=V("<div/>",{id:n.sTableId+"_processing",class:n.oClasses.processing.container,role:"status"}).html(n.oLanguage.sProcessing).append("<div><div></div><div></div><div></div><div></div></div>"),m?a.prependTo(V("div.dt-scroll",n.nTableWrapper)):a.insertBefore(t),V(t).on("processing.dt.DT",function(t,e,n){a.css("display",n?"block":"none")}))}function It(t,e,n){for(var a,r,o,i,l,s,u=t.aoColumns,c=V(e).children("tr"),d=e&&"thead"===e.nodeName.toLowerCase(),f=[],h=0,p=c.length;h<p;h++)f.push([]);for(h=0,p=c.length;h<p;h++)for(r=(a=c[h]).firstChild;r;){if("TD"==r.nodeName.toUpperCase()||"TH"==r.nodeName.toUpperCase()){var g,m,v,b,y,D=[];for(b=(b=+r.getAttribute("colspan"))&&0!=b&&1!=b?b:1,y=(y=+r.getAttribute("rowspan"))&&0!=y&&1!=y?y:1,l=function(t,e,n){for(var a=t[e];a[n];)n++;return n}(f,h,0),s=1==b,n&&(s&&(ot(t,l,V(r).data()),g=u[l],m=r.getAttribute("width")||null,(v=r.style.width.match(/width:\s*(\d+[pxem%]+)/))&&(m=v[1]),g.sWidthOrig=g.sWidth||m,d?(null===g.sTitle||g.autoTitle||(r.innerHTML=g.sTitle),!g.sTitle&&s&&(g.sTitle=I(r.innerHTML),g.autoTitle=!0)):g.footer&&(r.innerHTML=g.footer),g.ariaTitle||(g.ariaTitle=V(r).attr("aria-label")||g.sTitle),g.className)&&V(r).addClass(g.className),0===V("span.dt-column-title",r).length&&V("<span>").addClass("dt-column-title").append(r.childNodes).appendTo(r),d)&&0===V("span.dt-column-order",r).length&&V("<span>").addClass("dt-column-order").appendTo(r),i=0;i<b;i++){for(o=0;o<y;o++)f[h+o][l+i]={cell:r,unique:s},f[h+o].row=a;D.push(l+i)}r.setAttribute("data-dt-column",x(D).join(","))}r=r.nextSibling}return f}function At(n,t,a){function e(t){var e=n.jqXHR?n.jqXHR.status:null;(null===t||"number"==typeof e&&204==e)&&Lt(n,t={},[]),(e=t.error||t.sError)&&Z(n,0,e),n.json=t,tt(n,null,"xhr",[n,t,n.jqXHR],!0),a(t)}var r,o=n.ajax,i=n.oInstance,l=(V.isPlainObject(o)&&o.data&&(l="function"==typeof(r=o.data)?r(t,n):r,t="function"==typeof r&&l?l:V.extend(!0,t,l),delete o.data),{url:"string"==typeof o?o:"",data:t,success:e,dataType:"json",cache:!1,type:n.sServerMethod,error:function(t,e){-1===tt(n,null,"xhr",[n,null,n.jqXHR],!0).indexOf(!0)&&("parsererror"==e?Z(n,0,"Invalid JSON response",1):4===t.readyState&&Z(n,0,"Ajax error",7)),w(n,!1)}});V.isPlainObject(o)&&V.extend(l,o),n.oAjaxData=t,tt(n,null,"preXhr",[n,t,l],!0),"function"==typeof o?n.jqXHR=o.call(i,t,e,n):""===o.url?(i={},$.util.set(o.dataSrc)(i,[]),e(i)):(n.jqXHR=V.ajax(l),r&&(o.data=r))}function Lt(t,e,n){var a="data";if(V.isPlainObject(t.ajax)&&void 0!==t.ajax.dataSrc&&("string"==typeof(t=t.ajax.dataSrc)||"function"==typeof t?a=t:void 0!==t.data&&(a=t.data)),!n)return"data"===a?e.aaData||e[a]:""!==a?J(a)(e):e;m(a)(e,n)}function Ft(t,e,n){var t=V.isPlainObject(t.ajax)?t.ajax.dataSrc:null;return t&&t[e]?J(t[e])(n):(t="","draw"===e?t="sEcho":"recordsTotal"===e?t="iTotalRecords":"recordsFiltered"===e&&(t="iTotalDisplayRecords"),void 0!==n[t]?n[t]:n[e])}function Nt(n,t){var e=n.aoPreSearchCols;if(B(n),"ssp"!=et(n)){for(var a,r,o,i,l,s=n,u=s.aoColumns,c=s.aoData,d=0;d<c.length;d++)if(c[d]&&!(l=c[d])._aFilterData){for(o=[],a=0,r=u.length;a<r;a++)u[a].bSearchable?"string"!=typeof(i=null===(i=G(s,d,a,"filter"))?"":i)&&i.toString&&(i=i.toString()):i="",i.indexOf&&-1!==i.indexOf("&")&&(Rt.innerHTML=i,i=Ot?Rt.textContent:Rt.innerText),i.replace&&(i=i.replace(/[\r\n\u2028]/g,"")),o.push(i);l._aFilterData=o,l._sFilterRow=o.join(" "),0}n.aiDisplay=n.aiDisplayMaster.slice(),jt(n.aiDisplay,n,t.search,t),V.each(n.searchFixed,function(t,e){jt(n.aiDisplay,n,e,{})});for(var f=0;f<e.length;f++){var h=e[f];jt(n.aiDisplay,n,h.search,h,f),V.each(n.aoColumns[f].searchFixed,function(t,e){jt(n.aiDisplay,n,e,{},f)})}for(var p,g,m=n,v=$.ext.search,b=m.aiDisplay,y=0,D=v.length;y<D;y++){for(var x=[],S=0,T=b.length;S<T;S++)g=b[S],p=m.aoData[g],v[y](m,p._aFilterData,g,p._aData,S)&&x.push(g);b.length=0,b.push.apply(b,x)}}n.bFiltered=!0,tt(n,null,"search",[n])}function jt(t,e,n,a,r){if(""!==n){for(var o=0,i=[],l="function"==typeof n?n:null,s=n instanceof RegExp?n:l?null:function(t,e){var a=[],e=V.extend({},{boundary:!1,caseInsensitive:!0,exact:!1,regex:!1,smart:!0},e);"string"!=typeof t&&(t=t.toString());if(t=O(t),e.exact)return new RegExp("^"+Pt(t)+"$",e.caseInsensitive?"i":"");{var n,r,o;t=e.regex?t:Pt(t),e.smart&&(n=(t.match(/!?["\u201C][^"\u201D]+["\u201D]|[^ ]+/g)||[""]).map(function(t){var e,n=!1;return"!"===t.charAt(0)&&(n=!0,t=t.substring(1)),'"'===t.charAt(0)?t=(e=t.match(/^"(.*)"$/))?e[1]:t:"“"===t.charAt(0)&&(t=(e=t.match(/^\u201C(.*)\u201D$/))?e[1]:t),n&&(1<t.length&&a.push("(?!"+t+")"),t=""),t.replace(/"/g,"")}),r=a.length?a.join(""):"",o=e.boundary?"\\b":"",t="^(?=.*?"+o+n.join(")(?=.*?"+o)+")("+r+".)*$")}return new RegExp(t,e.caseInsensitive?"i":"")}(n,a),o=0;o<t.length;o++){var u=e.aoData[t[o]],c=void 0===r?u._sFilterRow:u._aFilterData[r];(l&&l(c,u._aData,t[o],r)||s&&s.test(c))&&i.push(t[o])}for(t.length=i.length,o=0;o<i.length;o++)t[o]=i[o]}}var Pt=$.util.escapeRegex,Rt=V("<div>")[0],Ot=void 0!==Rt.textContent;function Et(n){var a,t,e,r,o,i,l=n.iInitDisplayStart;n.bInitialised?(Dt(n,"header"),Dt(n,"footer"),St(n,n.aoHeader),St(n,n.aoFooter),Ct(n),e=(t=n).nTHead,i=e.querySelectorAll("tr"),r=t.bSortCellsTop,o=':not([data-dt-order="disable"]):not([data-dt-order="icon-only"])',!0===r?e=i[0]:!1===r&&(e=i[i.length-1]),Vt(t,e,e===t.nTHead?"tr"+o+" th"+o+", tr"+o+" td"+o:"th"+o+", td"+o),Ut(t,r=[],t.aaSorting),t.aaSorting=r,Bt(n),w(n,!0),tt(n,null,"preInit",[n],!0),s(n),"ssp"!=(i=et(n))&&("ajax"==i?At(n,{},function(t){var e=Lt(n,t);for(a=0;a<e.length;a++)Y(n,e[a]);n.iInitDisplayStart=l,s(n),w(n,!1),kt(n)}):(kt(n),w(n,!1)))):setTimeout(function(){Et(n)},200)}function kt(t){var e;t._bInitComplete||(e=[t,t.json],t._bInitComplete=!0,M(t),tt(t,null,"plugin-init",e,!0),tt(t,"aoInitComplete","init",e,!0))}function Mt(t,e){e=parseInt(e,10);t._iDisplayLength=e,Kt(t),tt(t,null,"length",[t,e])}function Ht(t,e,n){var a=t._iDisplayStart,r=t._iDisplayLength,o=t.fnRecordsDisplay();if(0===o||-1===r)a=0;else if("number"==typeof e)o<(a=e*r)&&(a=0);else if("first"==e)a=0;else if("previous"==e)(a=0<=r?a-r:0)<0&&(a=0);else if("next"==e)a+r<o&&(a+=r);else if("last"==e)a=Math.floor((o-1)/r)*r;else{if("ellipsis"===e)return;Z(t,0,"Unknown paging action: "+e,5)}o=t._iDisplayStart!==a;t._iDisplayStart=a,tt(t,null,o?"page":"page-nc",[t]),o&&n&&S(t)}function w(t,e){tt(t,null,"processing",[t,e])}function Wt(t){var e,n,a,r,o,i,l,s,u,c,d,f,h,p=V(t.nTable),g=t.oScroll;return""===g.sX&&""===g.sY?t.nTable:(e=g.sX,n=g.sY,a=t.oClasses.scrolling,o=(r=t.captionNode)?r._captionSide:null,u=V(p[0].cloneNode(!1)),i=V(p[0].cloneNode(!1)),c=function(t){return t?A(t):null},(l=p.children("tfoot")).length||(l=null),u=V(s="<div/>",{class:a.container}).append(V(s,{class:a.header.self}).css({overflow:"hidden",position:"relative",border:0,width:e?c(e):"100%"}).append(V(s,{class:a.header.inner}).css({"box-sizing":"content-box",width:g.sXInner||"100%"}).append(u.removeAttr("id").css("margin-left",0).append("top"===o?r:null).append(p.children("thead"))))).append(V(s,{class:a.body}).css({position:"relative",overflow:"auto",width:c(e)}).append(p)),l&&u.append(V(s,{class:a.footer.self}).css({overflow:"hidden",border:0,width:e?c(e):"100%"}).append(V(s,{class:a.footer.inner}).append(i.removeAttr("id").css("margin-left",0).append("bottom"===o?r:null).append(p.children("tfoot"))))),c=u.children(),d=c[0],f=c[1],h=l?c[2]:null,V(f).on("scroll.DT",function(){var t=this.scrollLeft;d.scrollLeft=t,l&&(h.scrollLeft=t)}),V("th, td",d).on("focus",function(){var t=d.scrollLeft;f.scrollLeft=t,l&&(f.scrollLeft=t)}),V(f).css("max-height",n),g.bCollapse||V(f).css("height",n),t.nScrollHead=d,t.nScrollBody=f,t.nScrollFoot=h,t.aoDrawCallback.push(Xt),u[0])}function Xt(e){var t=e.oScroll.iBarWidth,n=V(e.nScrollHead).children("div"),a=n.children("table"),r=e.nScrollBody,o=V(r),i=V(e.nScrollFoot).children("div"),l=i.children("table"),s=V(e.nTHead),u=V(e.nTable),c=e.nTFoot&&V("th, td",e.nTFoot).length?V(e.nTFoot):null,d=e.oBrowser,f=r.scrollHeight>r.clientHeight;if(e.scrollBarVis!==f&&void 0!==e.scrollBarVis)e.scrollBarVis=f,M(e);else{if(e.scrollBarVis=f,u.children("thead, tfoot").remove(),(f=s.clone().prependTo(u)).find("th, td").removeAttr("tabindex"),f.find("[id]").removeAttr("id"),c&&(m=c.clone().prependTo(u)).find("[id]").removeAttr("id"),e.aiDisplay.length)for(var h=u.children("tbody").eq(0).children("tr").eq(0).children("th, td").map(function(t){return{idx:H(e,t),width:V(this).outerWidth()}}),p=0;p<h.length;p++){var g=e.aoColumns[h[p].idx].colEl[0];g.style.width.replace("px","")!==h[p].width&&(g.style.width=h[p].width+"px")}a.find("colgroup").remove(),a.append(e.colgroup.clone()),c&&(l.find("colgroup").remove(),l.append(e.colgroup.clone())),V("th, td",f).each(function(){V(this.childNodes).wrapAll('<div class="dt-scroll-sizing">')}),c&&V("th, td",m).each(function(){V(this.childNodes).wrapAll('<div class="dt-scroll-sizing">')});var s=Math.floor(u.height())>r.clientHeight||"scroll"==o.css("overflow-y"),f="padding"+(d.bScrollbarLeft?"Left":"Right"),m=u.outerWidth();a.css("width",A(m)),n.css("width",A(m)).css(f,s?t+"px":"0px"),c&&(l.css("width",A(m)),i.css("width",A(m)).css(f,s?t+"px":"0px")),u.children("colgroup").prependTo(u),o.trigger("scroll"),!e.bSorted&&!e.bFiltered||e._drawHold||(r.scrollTop=0)}}function A(t){return null===t?"0px":"number"==typeof t?t<0?"0px":t+"px":t.match(/\d$/)?t+"px":t}function Bt(t){var e=t.aoColumns;for(t.colgroup.empty(),a=0;a<e.length;a++)e[a].bVisible&&t.colgroup.append(e[a].colEl)}function Vt(o,t,e,i,l){Qt(t,e,function(t){var e=!1,n=void 0===i?st(t.target):[i];if(n.length){for(var a=0,r=n.length;a<r;a++)if(!1!==function(t,e,n,a){function r(t,e){var n=t._idx;return(n=void 0===n?s.indexOf(t[1]):n)+1<s.length?n+1:e?null:0}var o,i=t.aoColumns[e],l=t.aaSorting,s=i.asSorting;if(!i.bSortable)return!1;"number"==typeof l[0]&&(l=t.aaSorting=[l]);(a||n)&&t.oFeatures.bSortMulti?-1!==(i=f(l,"0").indexOf(e))?null===(o=null===(o=r(l[i],!0))&&1===l.length?0:o)?l.splice(i,1):(l[i][1]=s[o],l[i]._idx=o):(a?l.push([e,s[0],0]):l.push([e,l[0][1],0]),l[l.length-1]._idx=0):l.length&&l[0][0]==e?(o=r(l[0]),l.length=1,l[0][1]=s[o],l[0]._idx=o):(l.length=0,l.push([e,s[0]]),l[0]._idx=0)}(o,n[a],a,t.shiftKey)&&(e=!0),1===o.aaSorting.length&&""===o.aaSorting[0][1])break;e&&(w(o,!0),setTimeout(function(){zt(o),qt(o,o.aiDisplay),w(o,!1),s(o,!1,!1),l&&l()},0))}})}function qt(t,e){if(!(e.length<2)){for(var n=t.aiDisplayMaster,a={},r={},o=0;o<n.length;o++)a[n[o]]=o;for(o=0;o<e.length;o++)r[e[o]]=a[e[o]];e.sort(function(t,e){return r[t]-r[e]})}}function Ut(n,a,t){function e(t){var e;V.isPlainObject(t)?void 0!==t.idx?a.push([t.idx,t.dir]):t.name&&-1!==(e=f(n.aoColumns,"sName").indexOf(t.name))&&a.push([e,t.dir]):a.push(t)}if(V.isPlainObject(t))e(t);else if(t.length&&"number"==typeof t[0])e(t);else if(t.length)for(var r=0;r<t.length;r++)e(t[r])}function $t(t){var e,n,a,r,o,i,l,s=[],u=$.ext.type.order,c=t.aoColumns,d=t.aaSortingFixed,f=V.isPlainObject(d),h=[];if(t.oFeatures.bSort)for(Array.isArray(d)&&Ut(t,h,d),f&&d.pre&&Ut(t,h,d.pre),Ut(t,h,t.aaSorting),f&&d.post&&Ut(t,h,d.post),e=0;e<h.length;e++)if(c[l=h[e][0]])for(n=0,a=(r=c[l].aDataSort).length;n<a;n++)i=c[o=r[n]].sType||"string",void 0===h[e]._idx&&(h[e]._idx=c[o].asSorting.indexOf(h[e][1])),h[e][1]&&s.push({src:l,col:o,dir:h[e][1],index:h[e]._idx,type:i,formatter:u[i+"-pre"],sorter:u[i+"-"+h[e][1]]});return s}function zt(t,e,n){var a,r,o,i,l,c,d=[],s=$.ext.type.order,f=t.aoData,u=t.aiDisplayMaster;for(B(t),void 0!==e?(l=t.aoColumns[e],c=[{src:e,col:e,dir:n,index:0,type:l.sType,formatter:s[l.sType+"-pre"],sorter:s[l.sType+"-"+n]}],u=u.slice()):c=$t(t),a=0,r=c.length;a<r;a++){i=c[a],S=x=D=g=p=h=y=b=v=m=void 0;var h,p,g,m=t,v=i.col,b=m.aoColumns[v],y=$.ext.order[b.sSortDataType];y&&(h=y.call(m.oInstance,m,v,T(m,v)));for(var D=$.ext.type.order[b.sType+"-pre"],x=m.aoData,S=0;S<x.length;S++)x[S]&&((p=x[S])._aSortData||(p._aSortData=[]),p._aSortData[v]&&!y||(g=y?h[S]:G(m,S,v,"sort"),p._aSortData[v]=D?D(g,m):g))}if("ssp"!=et(t)&&0!==c.length){for(a=0,o=u.length;a<o;a++)d[a]=a;c.length&&"desc"===c[0].dir&&d.reverse(),u.sort(function(t,e){for(var n,a,r,o,i=c.length,l=f[t]._aSortData,s=f[e]._aSortData,u=0;u<i;u++)if(n=l[(o=c[u]).col],a=s[o.col],o.sorter){if(0!==(r=o.sorter(n,a)))return r}else if(0!==(r=n<a?-1:a<n?1:0))return"asc"===o.dir?r:-r;return(n=d[t])<(a=d[e])?-1:a<n?1:0})}else 0===c.length&&u.sort(function(t,e){return t<e?-1:e<t?1:0});return void 0===e&&(t.bSorted=!0,tt(t,null,"order",[t,c])),u}function Yt(t){var e,n,a,r=t.aLastSort,o=t.oClasses.order.position,i=$t(t),l=t.oFeatures;if(l.bSort&&l.bSortClasses){for(e=0,n=r.length;e<n;e++)a=r[e].src,V(f(t.aoData,"anCells",a)).removeClass(o+(e<2?e+1:3));for(e=0,n=i.length;e<n;e++)a=i[e].src,V(f(t.aoData,"anCells",a)).addClass(o+(e<2?e+1:3))}t.aLastSort=i}function Gt(n){var t;n._bLoadingState||(t={time:+new Date,start:n._iDisplayStart,length:n._iDisplayLength,order:V.extend(!0,[],n.aaSorting),search:V.extend({},n.oPreviousSearch),columns:n.aoColumns.map(function(t,e){return{visible:t.bVisible,search:V.extend({},n.aoPreSearchCols[e])}})},n.oSavedState=t,tt(n,"aoStateSaveParams","stateSaveParams",[n,t]),n.oFeatures.bStateSave&&!n.bDestroying&&n.fnStateSaveCallback.call(n.oInstance,n,t))}function Jt(n,t,e){var a,r,o=n.aoColumns,i=(n._bLoadingState=!0,n._bInitComplete?new $.Api(n):null);if(t&&t.time){var l=n.iStateDuration;if(0<l&&t.time<+new Date-1e3*l)n._bLoadingState=!1;else if(-1!==tt(n,"aoStateLoadParams","stateLoadParams",[n,t]).indexOf(!1))n._bLoadingState=!1;else if(t.columns&&o.length!==t.columns.length)n._bLoadingState=!1;else{if(n.oLoadedState=V.extend(!0,{},t),tt(n,null,"stateLoadInit",[n,t],!0),void 0!==t.length&&(i?i.page.len(t.length):n._iDisplayLength=t.length),void 0!==t.start&&(null===i?(n._iDisplayStart=t.start,n.iInitDisplayStart=t.start):Ht(n,t.start/n._iDisplayLength)),void 0!==t.order&&(n.aaSorting=[],V.each(t.order,function(t,e){n.aaSorting.push(e[0]>=o.length?[0,e[1]]:e)})),void 0!==t.search&&V.extend(n.oPreviousSearch,t.search),t.columns){for(a=0,r=t.columns.length;a<r;a++){var s=t.columns[a];void 0!==s.visible&&(i?i.column(a).visible(s.visible,!1):o[a].bVisible=s.visible),void 0!==s.search&&V.extend(n.aoPreSearchCols[a],s.search)}i&&i.columns.adjust()}n._bLoadingState=!1,tt(n,"aoStateLoaded","stateLoaded",[n,t])}}else n._bLoadingState=!1;e()}function Z(t,e,n,a){if(n="DataTables warning: "+(t?"table id="+t.sTableId+" - ":"")+n,a&&(n+=". For more information about this error, please see https://datatables.net/tn/"+a),e)q.console&&console.log&&console.log(n);else{e=$.ext,e=e.sErrMode||e.errMode;if(t&&tt(t,null,"dt-error",[t,a,n],!0),"alert"==e)alert(n);else{if("throw"==e)throw new Error(n);"function"==typeof e&&e(t,a,n)}}}function Q(n,a,t,e){Array.isArray(t)?V.each(t,function(t,e){Array.isArray(e)?Q(n,a,e[0],e[1]):Q(n,a,e)}):(void 0===e&&(e=t),void 0!==a[t]&&(n[e]=a[t]))}function Zt(t,e,n){var a,r;for(r in e)Object.prototype.hasOwnProperty.call(e,r)&&(a=e[r],V.isPlainObject(a)?(V.isPlainObject(t[r])||(t[r]={}),V.extend(!0,t[r],a)):n&&"data"!==r&&"aaData"!==r&&Array.isArray(a)?t[r]=a.slice():t[r]=a);return t}function Qt(t,e,n){V(t).on("click.DT",e,function(t){n(t)}).on("keypress.DT",e,function(t){13===t.which&&(t.preventDefault(),n(t))}).on("selectstart.DT",e,function(){return!1})}function K(t,e,n){n&&t[e].push(n)}function tt(e,t,n,a,r){var o=[];return t&&(o=e[t].slice().reverse().map(function(t){return t.apply(e.oInstance,a)})),null!==n&&(t=V.Event(n+".dt"),n=V(e.nTable),t.dt=e.api,n[r?"trigger":"triggerHandler"](t,a),r&&0===n.parents("body").length&&V("body").trigger(t,a),o.push(t.result)),o}function Kt(t){var e=t._iDisplayStart,n=t.fnDisplayEnd(),a=t._iDisplayLength;n<=e&&(e=n-a),e-=e%a,t._iDisplayStart=e=-1===a||e<0?0:e}function te(t,e){var t=t.renderer,n=$.ext.renderer[e];return V.isPlainObject(t)&&t[e]?n[t[e]]||n._:"string"==typeof t&&n[t]||n._}function et(t){return t.oFeatures.bServerSide?"ssp":t.ajax?"ajax":"dom"}function ee(t,e,n){var a=t.fnFormatNumber,r=t._iDisplayStart+1,o=t._iDisplayLength,i=t.fnRecordsDisplay(),l=t.fnRecordsTotal(),s=-1===o;return e.replace(/_START_/g,a.call(t,r)).replace(/_END_/g,a.call(t,t.fnDisplayEnd())).replace(/_MAX_/g,a.call(t,l)).replace(/_TOTAL_/g,a.call(t,i)).replace(/_PAGE_/g,a.call(t,s?1:Math.ceil(r/o))).replace(/_PAGES_/g,a.call(t,s?1:Math.ceil(i/o))).replace(/_ENTRIES_/g,t.api.i18n("entries","",n)).replace(/_ENTRIES-MAX_/g,t.api.i18n("entries","",l)).replace(/_ENTRIES-TOTAL_/g,t.api.i18n("entries","",i))}var ne=[],n=Array.prototype;U=function(t,e){if(!(this instanceof U))return new U(t,e);function n(t){t=t,e=$.settings,a=f(e,"nTable");var n,e,a,r=t?t.nTable&&t.oFeatures?[t]:t.nodeName&&"table"===t.nodeName.toLowerCase()?-1!==(r=a.indexOf(t))?[e[r]]:null:t&&"function"==typeof t.settings?t.settings().toArray():("string"==typeof t?n=V(t).get():t instanceof V&&(n=t.get()),n?e.filter(function(t,e){return n.includes(a[e])}):void 0):[];r&&o.push.apply(o,r)}var o=[];if(Array.isArray(t))for(var a=0,r=t.length;a<r;a++)n(t[a]);else n(t);this.context=1<o.length?x(o):o,e&&this.push.apply(this,e),this.selector={rows:null,cols:null,opts:null},U.extend(this,this,ne)},$.Api=U,V.extend(U.prototype,{any:function(){return 0!==this.count()},context:[],count:function(){return this.flatten().length},each:function(t){for(var e=0,n=this.length;e<n;e++)t.call(this,this[e],e,this);return this},eq:function(t){var e=this.context;return e.length>t?new U(e[t],this[t]):null},filter:function(t){t=n.filter.call(this,t,this);return new U(this.context,t)},flatten:function(){var t=[];return new U(this.context,t.concat.apply(t,this.toArray()))},get:function(t){return this[t]},join:n.join,includes:function(t){return-1!==this.indexOf(t)},indexOf:n.indexOf,iterator:function(t,e,n,a){var r,o,i,l,s,u,c,d,f=[],h=this.context,p=this.selector;for("string"==typeof t&&(a=n,n=e,e=t,t=!1),o=0,i=h.length;o<i;o++){var g=new U(h[o]);if("table"===e)void 0!==(r=n.call(g,h[o],o))&&f.push(r);else if("columns"===e||"rows"===e)void 0!==(r=n.call(g,h[o],this[o],o))&&f.push(r);else if("every"===e||"column"===e||"column-rows"===e||"row"===e||"cell"===e)for(c=this[o],"column-rows"===e&&(u=he(h[o],p.opts)),l=0,s=c.length;l<s;l++)d=c[l],void 0!==(r="cell"===e?n.call(g,h[o],d.row,d.column,o,l):n.call(g,h[o],d,o,l,u))&&f.push(r)}return f.length||a?((t=(a=new U(h,t?f.concat.apply([],f):f)).selector).rows=p.rows,t.cols=p.cols,t.opts=p.opts,a):this},lastIndexOf:n.lastIndexOf,length:0,map:function(t){t=n.map.call(this,t,this);return new U(this.context,t)},pluck:function(t){var e=$.util.get(t);return this.map(function(t){return e(t)})},pop:n.pop,push:n.push,reduce:n.reduce,reduceRight:n.reduceRight,reverse:n.reverse,selector:null,shift:n.shift,slice:function(){return new U(this.context,this)},sort:n.sort,splice:n.splice,toArray:function(){return n.slice.call(this)},to$:function(){return V(this)},toJQuery:function(){return V(this)},unique:function(){return new U(this.context,x(this.toArray()))},unshift:n.unshift}),q.__apiStruct=ne,U.extend=function(t,e,n){if(n.length&&e&&(e instanceof U||e.__dt_wrapper))for(var a,r=0,o=n.length;r<o;r++)"__proto__"!==(a=n[r]).name&&(e[a.name]="function"===a.type?function(e,n,a){return function(){var t=n.apply(e||this,arguments);return U.extend(t,t,a.methodExt),t}}(t,a.val,a):"object"===a.type?{}:a.val,e[a.name].__dt_wrapper=!0,U.extend(t,e[a.name],a.propExt))},U.register=e=function(t,e){if(Array.isArray(t))for(var n=0,a=t.length;n<a;n++)U.register(t[n],e);else for(var r=t.split("."),o=ne,i=0,l=r.length;i<l;i++){var s,u,c=function(t,e){for(var n=0,a=t.length;n<a;n++)if(t[n].name===e)return t[n];return null}(o,u=(s=-1!==r[i].indexOf("()"))?r[i].replace("()",""):r[i]);c||o.push(c={name:u,val:{},methodExt:[],propExt:[],type:"object"}),i===l-1?(c.val=e,c.type="function"==typeof e?"function":V.isPlainObject(e)?"object":"other"):o=s?c.methodExt:c.propExt}},U.registerPlural=t=function(t,e,n){U.register(t,n),U.register(e,function(){var t=n.apply(this,arguments);return t===this?this:t instanceof U?t.length?Array.isArray(t[0])?new U(t.context,t[0]):t[0]:void 0:t})};function ae(t,e){var n,a;return Array.isArray(t)?(n=[],t.forEach(function(t){t=ae(t,e);n.push.apply(n,t)}),n.filter(function(t){return t})):"number"==typeof t?[e[t]]:(a=e.map(function(t){return t.nTable}),V(a).filter(t).map(function(){var t=a.indexOf(this);return e[t]}).toArray())}function re(r,o,t){var e,n;t&&(e=new U(r)).one("draw",function(){t(e.ajax.json())}),"ssp"==et(r)?s(r,o):(w(r,!0),(n=r.jqXHR)&&4!==n.readyState&&n.abort(),At(r,{},function(t){pt(r);for(var e=Lt(r,t),n=0,a=e.length;n<a;n++)Y(r,e[n]);s(r,o),kt(r),w(r,!1)}))}function oe(t,e,n,a,r){for(var o,i,l,s,u=[],c=typeof e,d=0,f=(e=e&&"string"!=c&&"function"!=c&&void 0!==e.length?e:[e]).length;d<f;d++)for(l=0,s=(i=e[d]&&e[d].split&&!e[d].match(/[[(:]/)?e[d].split(","):[e[d]]).length;l<s;l++)(o=(o=n("string"==typeof i[l]?i[l].trim():i[l])).filter(function(t){return null!=t}))&&o.length&&(u=u.concat(o));var h=C.selector[t];if(h.length)for(d=0,f=h.length;d<f;d++)u=h[d](a,r,u);return x(u)}function ie(t){return(t=t||{}).filter&&void 0===t.search&&(t.search=t.filter),V.extend({search:"none",order:"current",page:"all"},t)}function le(t){var e=new U(t.context[0]);return t.length&&e.push(t[0]),e.selector=t.selector,e.length&&1<e[0].length&&e[0].splice(1),e}e("tables()",function(t){return null!=t?new U(ae(t,this.context)):this}),e("table()",function(t){var t=this.tables(t),e=t.context;return e.length?new U(e[0]):t}),[["nodes","node","nTable"],["body","body","nTBody"],["header","header","nTHead"],["footer","footer","nTFoot"]].forEach(function(e){t("tables()."+e[0]+"()","table()."+e[1]+"()",function(){return this.iterator("table",function(t){return t[e[2]]},1)})}),[["header","aoHeader"],["footer","aoFooter"]].forEach(function(n){e("table()."+n[0]+".structure()",function(t){var t=this.columns(t).indexes().flatten(),e=this.context[0];return xt(e,e[n[1]],t)})}),t("tables().containers()","table().container()",function(){return this.iterator("table",function(t){return t.nTableWrapper},1)}),e("tables().every()",function(n){var a=this;return this.iterator("table",function(t,e){n.call(a.table(e),e)})}),e("caption()",function(r,o){var t,e=this.context;return void 0===r?(t=e[0].captionNode)&&e.length?t.innerHTML:null:this.iterator("table",function(t){var e=V(t.nTable),n=V(t.captionNode),a=V(t.nTableWrapper);n.length||(n=V("<caption/>").html(r),t.captionNode=n[0],o)||(e.prepend(n),o=n.css("caption-side")),n.html(r),o&&(n.css("caption-side",o),n[0]._captionSide=o),(a.find("div.dataTables_scroll").length?(t="top"===o?"Head":"Foot",a.find("div.dataTables_scroll"+t+" table")):e).prepend(n)},1)}),e("caption.node()",function(){var t=this.context;return t.length?t[0].captionNode:null}),e("draw()",function(e){return this.iterator("table",function(t){"page"===e?S(t):s(t,!1===(e="string"==typeof e?"full-hold"!==e:e))})}),e("page()",function(e){return void 0===e?this.page.info().page:this.iterator("table",function(t){Ht(t,e)})}),e("page.info()",function(){var t,e,n,a,r;if(0!==this.context.length)return e=(t=this.context[0])._iDisplayStart,n=t.oFeatures.bPaginate?t._iDisplayLength:-1,a=t.fnRecordsDisplay(),{page:(r=-1===n)?0:Math.floor(e/n),pages:r?1:Math.ceil(a/n),start:e,end:t.fnDisplayEnd(),length:n,recordsTotal:t.fnRecordsTotal(),recordsDisplay:a,serverSide:"ssp"===et(t)}}),e("page.len()",function(e){return void 0===e?0!==this.context.length?this.context[0]._iDisplayLength:void 0:this.iterator("table",function(t){Mt(t,e)})}),e("ajax.json()",function(){var t=this.context;if(0<t.length)return t[0].json}),e("ajax.params()",function(){var t=this.context;if(0<t.length)return t[0].oAjaxData}),e("ajax.reload()",function(e,n){return this.iterator("table",function(t){re(t,!1===n,e)})}),e("ajax.url()",function(e){var t=this.context;return void 0===e?0===t.length?void 0:(t=t[0],V.isPlainObject(t.ajax)?t.ajax.url:t.ajax):this.iterator("table",function(t){V.isPlainObject(t.ajax)?t.ajax.url=e:t.ajax=e})}),e("ajax.url().load()",function(e,n){return this.iterator("table",function(t){re(t,!1===n,e)})});function se(o,i,t,e){function l(t,e){var n;if(Array.isArray(t)||t instanceof V)for(var a=0,r=t.length;a<r;a++)l(t[a],e);else t.nodeName&&"tr"===t.nodeName.toLowerCase()?(t.setAttribute("data-dt-row",i.idx),s.push(t)):(n=V("<tr><td></td></tr>").attr("data-dt-row",i.idx).addClass(e),V("td",n).addClass(e).html(t)[0].colSpan=W(o),s.push(n[0]))}var s=[];l(t,e),i._details&&i._details.detach(),i._details=V(s),i._detailsShow&&i._details.insertAfter(i.nTr)}function ue(t,e){var n=t.context;if(n.length&&t.length){var a=n[0].aoData[t[0]];if(a._details){(a._detailsShow=e)?(a._details.insertAfter(a.nTr),V(a.nTr).addClass("dt-hasChild")):(a._details.detach(),V(a.nTr).removeClass("dt-hasChild")),tt(n[0],null,"childRow",[e,t.row(t[0])]);var i=n[0],r=new U(i),a=".dt.DT_details",e="draw"+a,t="column-sizing"+a,a="destroy"+a,l=i.aoData;if(r.off(e+" "+t+" "+a),f(l,"_details").length>0){r.on(e,function(t,e){if(i!==e)return;r.rows({page:"current"}).eq(0).each(function(t){var e=l[t];if(e._detailsShow)e._details.insertAfter(e.nTr)})});r.on(t,function(t,e){if(i!==e)return;var n,a=W(e);for(var r=0,o=l.length;r<o;r++){n=l[r];if(n&&n._details)n._details.each(function(){var t=V(this).children("td");if(t.length==1)t.attr("colspan",a)})}});r.on(a,function(t,e){if(i!==e)return;for(var n=0,a=l.length;n<a;n++)if(l[n]&&l[n]._details)me(r,n)})}ge(n)}}}function ce(t,e,n,a,r,o){for(var i=[],l=0,s=r.length;l<s;l++)i.push(G(t,r[l],e,o));return i}function de(t,e,n){var a=t.aoHeader;return a[void 0!==n?n:t.bSortCellsTop?0:a.length-1][e].cell}function fe(e,n){return function(t){return y(t)||"string"!=typeof t||(t=t.replace(d," "),e&&(t=I(t)),n&&(t=O(t,!1))),t}}var he=function(t,e){var n,a=[],r=t.aiDisplay,o=t.aiDisplayMaster,i=e.search,l=e.order,e=e.page;if("ssp"==et(t))return"removed"===i?[]:h(0,o.length);if("current"==e)for(u=t._iDisplayStart,c=t.fnDisplayEnd();u<c;u++)a.push(r[u]);else if("current"==l||"applied"==l){if("none"==i)a=o.slice();else if("applied"==i)a=r.slice();else if("removed"==i){for(var s={},u=0,c=r.length;u<c;u++)s[r[u]]=null;o.forEach(function(t){Object.prototype.hasOwnProperty.call(s,t)||a.push(t)})}}else if("index"==l||"original"==l)for(u=0,c=t.aoData.length;u<c;u++)t.aoData[u]&&("none"==i||-1===(n=r.indexOf(u))&&"removed"==i||0<=n&&"applied"==i)&&a.push(u);else if("number"==typeof l){var d=zt(t,l,"asc");if("none"===i)a=d;else for(u=0;u<d.length;u++)(-1===(n=r.indexOf(d[u]))&&"removed"==i||0<=n&&"applied"==i)&&a.push(d[u])}return a},pe=(e("rows()",function(n,a){void 0===n?n="":V.isPlainObject(n)&&(a=n,n=""),a=ie(a);var t=this.iterator("table",function(t){return e=oe("row",e=n,function(n){var t=g(n),a=r.aoData;if(null!==t&&!o)return[t];if(i=i||he(r,o),null!==t&&-1!==i.indexOf(t))return[t];if(null==n||""===n)return i;if("function"==typeof n)return i.map(function(t){var e=a[t];return n(t,e._aData,e.nTr)?t:null});if(n.nodeName)return t=n._DT_RowIndex,e=n._DT_CellIndex,void 0!==t?a[t]&&a[t].nTr===n?[t]:[]:e?a[e.row]&&a[e.row].nTr===n.parentNode?[e.row]:[]:(t=V(n).closest("*[data-dt-row]")).length?[t.data("dt-row")]:[];if("string"==typeof n&&"#"===n.charAt(0)){var e=r.aIds[n.replace(/^#/,"")];if(void 0!==e)return[e.idx]}t=b(v(r.aoData,i,"nTr"));return V(t).filter(n).map(function(){return this._DT_RowIndex}).toArray()},r=t,o=a),"current"!==o.order&&"applied"!==o.order||qt(r,e),e;var r,e,o,i},1);return t.selector.rows=n,t.selector.opts=a,t}),e("rows().nodes()",function(){return this.iterator("row",function(t,e){return t.aoData[e].nTr||void 0},1)}),e("rows().data()",function(){return this.iterator(!0,"rows",function(t,e){return v(t.aoData,e,"_aData")},1)}),t("rows().cache()","row().cache()",function(n){return this.iterator("row",function(t,e){t=t.aoData[e];return"search"===n?t._aFilterData:t._aSortData},1)}),t("rows().invalidate()","row().invalidate()",function(n){return this.iterator("row",function(t,e){gt(t,e,n)})}),t("rows().indexes()","row().index()",function(){return this.iterator("row",function(t,e){return e},1)}),t("rows().ids()","row().id()",function(t){for(var e=[],n=this.context,a=0,r=n.length;a<r;a++)for(var o=0,i=this[a].length;o<i;o++){var l=n[a].rowIdFn(n[a].aoData[this[a][o]]._aData);e.push((!0===t?"#":"")+l)}return new U(n,e)}),t("rows().remove()","row().remove()",function(){return this.iterator("row",function(t,e){var n=t.aoData,a=n[e],r=t.aiDisplayMaster.indexOf(e),r=(-1!==r&&t.aiDisplayMaster.splice(r,1),0<t._iRecordsDisplay&&t._iRecordsDisplay--,Kt(t),t.rowIdFn(a._aData));void 0!==r&&delete t.aIds[r],n[e]=null}),this}),e("rows.add()",function(o){var t=this.iterator("table",function(t){for(var e,n=[],a=0,r=o.length;a<r;a++)(e=o[a]).nodeName&&"TR"===e.nodeName.toUpperCase()?n.push(ut(t,e)[0]):n.push(Y(t,e));return n},1),e=this.rows(-1);return e.pop(),e.push.apply(e,t),e}),e("row()",function(t,e){return le(this.rows(t,e))}),e("row().data()",function(t){var e,n=this.context;return void 0===t?n.length&&this.length&&this[0].length?n[0].aoData[this[0]]._aData:void 0:((e=n[0].aoData[this[0]])._aData=t,Array.isArray(t)&&e.nTr&&e.nTr.id&&m(n[0].rowId)(t,e.nTr.id),gt(n[0],this[0],"data"),this)}),e("row().node()",function(){var t=this.context;if(t.length&&this.length&&this[0].length){t=t[0].aoData[this[0]];if(t&&t.nTr)return t.nTr}return null}),e("row.add()",function(e){e instanceof V&&e.length&&(e=e[0]);var t=this.iterator("table",function(t){return e.nodeName&&"TR"===e.nodeName.toUpperCase()?ut(t,e)[0]:Y(t,e)});return this.row(t[0])}),V(_).on("plugin-init.dt",function(t,e){var a=new U(e);a.on("stateSaveParams.DT",function(t,e,n){for(var a=e.rowIdFn,r=e.aiDisplayMaster,o=[],i=0;i<r.length;i++){var l=r[i],l=e.aoData[l];l._detailsShow&&o.push("#"+a(l._aData))}n.childRows=o}),a.on("stateLoaded.DT",function(t,e,n){pe(a,n)}),pe(a,a.state.loaded())}),function(t,e){e&&e.childRows&&t.rows(e.childRows.map(function(t){return t.replace(/([^:\\]*(?:\\.[^:\\]*)*):/g,"$1\\:")})).every(function(){tt(t.settings()[0],null,"requestChild",[this])})}),ge=$.util.throttle(function(t){Gt(t[0])},500),me=function(t,e){var n=t.context;n.length&&(e=n[0].aoData[void 0!==e?e:t[0]])&&e._details&&(e._details.remove(),e._detailsShow=void 0,e._details=void 0,V(e.nTr).removeClass("dt-hasChild"),ge(n))},ve="row().child",be=ve+"()",ye=(e(be,function(t,e){var n=this.context;return void 0===t?n.length&&this.length&&n[0].aoData[this[0]]?n[0].aoData[this[0]]._details:void 0:(!0===t?this.child.show():!1===t?me(this):n.length&&this.length&&se(n[0],n[0].aoData[this[0]],t,e),this)}),e([ve+".show()",be+".show()"],function(){return ue(this,!0),this}),e([ve+".hide()",be+".hide()"],function(){return ue(this,!1),this}),e([ve+".remove()",be+".remove()"],function(){return me(this),this}),e(ve+".isShown()",function(){var t=this.context;return t.length&&this.length&&t[0].aoData[this[0]]&&t[0].aoData[this[0]]._detailsShow||!1}),/^([^:]+)?:(name|title|visIdx|visible)$/),be=(e("columns()",function(n,a){void 0===n?n="":V.isPlainObject(n)&&(a=n,n=""),a=ie(a);var t=this.iterator("table",function(t){return e=n,l=a,s=(i=t).aoColumns,u=f(s,"sName"),c=f(s,"sTitle"),t=$.util.get("[].[].cell")(i.aoHeader),d=x(E([],t)),oe("column",e,function(n){var a,t=g(n);if(""===n)return h(s.length);if(null!==t)return[0<=t?t:s.length+t];if("function"==typeof n)return a=he(i,l),s.map(function(t,e){return n(e,ce(i,e,0,0,a),de(i,e))?e:null});var e,r,o="string"==typeof n?n.match(ye):"";if(o)switch(o[2]){case"visIdx":case"visible":return o[1]?(e=parseInt(o[1],10))<0?[(r=s.map(function(t,e){return t.bVisible?e:null}))[r.length+e]]:[H(i,e)]:s.map(function(t,e){return t.bVisible?e:null});case"name":return u.map(function(t,e){return t===o[1]?e:null});case"title":return c.map(function(t,e){return t===o[1]?e:null});default:return[]}return n.nodeName&&n._DT_CellIndex?[n._DT_CellIndex.column]:(t=V(d).filter(n).map(function(){return st(this)}).toArray()).length||!n.nodeName?t:(t=V(n).closest("*[data-dt-column]")).length?[t.data("dt-column")]:[]},i,l);var i,e,l,s,u,c,d},1);return t.selector.cols=n,t.selector.opts=a,t}),t("columns().header()","column().header()",function(n){return this.iterator("column",function(t,e){return de(t,e,n)},1)}),t("columns().footer()","column().footer()",function(n){return this.iterator("column",function(t,e){return t.aoFooter.length?t.aoFooter[void 0!==n?n:0][e].cell:null},1)}),t("columns().data()","column().data()",function(){return this.iterator("column-rows",ce,1)}),t("columns().render()","column().render()",function(o){return this.iterator("column-rows",function(t,e,n,a,r){return ce(t,e,0,0,r,o)},1)}),t("columns().dataSrc()","column().dataSrc()",function(){return this.iterator("column",function(t,e){return t.aoColumns[e].mData},1)}),t("columns().cache()","column().cache()",function(o){return this.iterator("column-rows",function(t,e,n,a,r){return v(t.aoData,r,"search"===o?"_aFilterData":"_aSortData",e)},1)}),t("columns().init()","column().init()",function(){return this.iterator("column",function(t,e){return t.aoColumns[e]},1)}),t("columns().nodes()","column().nodes()",function(){return this.iterator("column-rows",function(t,e,n,a,r){return v(t.aoData,r,"anCells",e)},1)}),t("columns().titles()","column().title()",function(n,a){return this.iterator("column",function(t,e){"number"==typeof n&&(a=n,n=void 0);e=V("span.dt-column-title",this.column(e).header(a));return void 0!==n?(e.html(n),this):e.html()},1)}),t("columns().types()","column().type()",function(){return this.iterator("column",function(t,e){e=t.aoColumns[e].sType;return e||B(t),e},1)}),t("columns().visible()","column().visible()",function(n,a){var e=this,r=[],t=this.iterator("column",function(t,e){if(void 0===n)return t.aoColumns[e].bVisible;!function(t,e,n){var a,r,o=t.aoColumns,i=o[e],l=t.aoData;if(void 0===n)return i.bVisible;if(i.bVisible===n)return!1;if(n)for(var s=f(o,"bVisible").indexOf(!0,e+1),u=0,c=l.length;u<c;u++)l[u]&&(r=l[u].nTr,a=l[u].anCells,r)&&r.insertBefore(a[e],a[s]||null);else V(f(t.aoData,"anCells",e)).detach();return i.bVisible=n,Bt(t),!0}(t,e,n)||r.push(e)});return void 0!==n&&this.iterator("table",function(t){St(t,t.aoHeader),St(t,t.aoFooter),t.aiDisplay.length||V(t.nTBody).find("td[colspan]").attr("colspan",W(t)),Gt(t),e.iterator("column",function(t,e){r.includes(e)&&tt(t,null,"column-visibility",[t,e,n,a])}),r.length&&(void 0===a||a)&&e.columns.adjust()}),t}),t("columns().widths()","column().width()",function(){var t=this.columns(":visible").count(),t=V("<tr>").html("<td>"+Array(t).join("</td><td>")+"</td>"),n=(V(this.table().body()).append(t),t.children().map(function(){return V(this).outerWidth()}));return t.remove(),this.iterator("column",function(t,e){t=T(t,e);return null!==t?n[t]:0},1)}),t("columns().indexes()","column().index()",function(n){return this.iterator("column",function(t,e){return"visible"===n?T(t,e):e},1)}),e("columns.adjust()",function(){return this.iterator("table",function(t){M(t)},1)}),e("column.index()",function(t,e){var n;if(0!==this.context.length)return n=this.context[0],"fromVisible"===t||"toData"===t?H(n,e):"fromData"===t||"toVisible"===t?T(n,e):void 0}),e("column()",function(t,e){return le(this.columns(t,e))}),e("cells()",function(g,t,m){var a,r,o,i,l,s,e;return V.isPlainObject(g)&&(void 0===g.row?(m=g,g=null):(m=t,t=null)),V.isPlainObject(t)&&(m=t,t=null),null==t?this.iterator("table",function(t){return a=t,t=g,e=ie(m),d=a.aoData,f=he(a,e),n=b(v(d,f,"anCells")),h=V(E([],n)),p=a.aoColumns.length,oe("cell",t,function(t){var e,n="function"==typeof t;if(null==t||n){for(o=[],i=0,l=f.length;i<l;i++)for(r=f[i],s=0;s<p;s++)u={row:r,column:s},(!n||(c=d[r],t(u,G(a,r,s),c.anCells?c.anCells[s]:null)))&&o.push(u);return o}return V.isPlainObject(t)?void 0!==t.column&&void 0!==t.row&&-1!==f.indexOf(t.row)?[t]:[]:(e=h.filter(t).map(function(t,e){return{row:e._DT_CellIndex.row,column:e._DT_CellIndex.column}}).toArray()).length||!t.nodeName?e:(c=V(t).closest("*[data-dt-row]")).length?[{row:c.data("dt-row"),column:c.data("dt-column")}]:[]},a,e);var a,e,r,o,i,l,s,u,c,d,f,n,h,p}):(e=m?{page:m.page,order:m.order,search:m.search}:{},a=this.columns(t,e),r=this.rows(g,e),e=this.iterator("table",function(t,e){var n=[];for(o=0,i=r[e].length;o<i;o++)for(l=0,s=a[e].length;l<s;l++)n.push({row:r[e][o],column:a[e][l]});return n},1),e=m&&m.selected?this.cells(e,m):e,V.extend(e.selector,{cols:t,rows:g,opts:m}),e)}),t("cells().nodes()","cell().node()",function(){return this.iterator("cell",function(t,e,n){t=t.aoData[e];return t&&t.anCells?t.anCells[n]:void 0},1)}),e("cells().data()",function(){return this.iterator("cell",function(t,e,n){return G(t,e,n)},1)}),t("cells().cache()","cell().cache()",function(a){return a="search"===a?"_aFilterData":"_aSortData",this.iterator("cell",function(t,e,n){return t.aoData[e][a][n]},1)}),t("cells().render()","cell().render()",function(a){return this.iterator("cell",function(t,e,n){return G(t,e,n,a)},1)}),t("cells().indexes()","cell().index()",function(){return this.iterator("cell",function(t,e,n){return{row:e,column:n,columnVisible:T(t,n)}},1)}),t("cells().invalidate()","cell().invalidate()",function(a){return this.iterator("cell",function(t,e,n){gt(t,e,a,n)})}),e("cell()",function(t,e,n){return le(this.cells(t,e,n))}),e("cell().data()",function(t){var e,n,a,r,o,i=this.context,l=this[0];return void 0===t?i.length&&l.length?G(i[0],l[0].row,l[0].column):void 0:(e=i[0],n=l[0].row,a=l[0].column,r=e.aoColumns[a],o=e.aoData[n]._aData,r.fnSetData(o,t,{settings:e,row:n,col:a}),gt(i[0],l[0].row,"data",l[0].column),this)}),e("order()",function(e,t){var n=this.context,a=Array.prototype.slice.call(arguments);return void 0===e?0!==n.length?n[0].aaSorting:void 0:("number"==typeof e?e=[[e,t]]:1<a.length&&(e=a),this.iterator("table",function(t){t.aaSorting=Array.isArray(e)?e.slice():e}))}),e("order.listener()",function(e,n,a){return this.iterator("table",function(t){Vt(t,e,{},n,a)})}),e("order.fixed()",function(e){var t;return e?this.iterator("table",function(t){t.aaSortingFixed=V.extend(!0,{},e)}):(t=(t=this.context).length?t[0].aaSortingFixed:void 0,Array.isArray(t)?{pre:t}:t)}),e(["columns().order()","column().order()"],function(n){var a=this;return n?this.iterator("table",function(t,e){t.aaSorting=a[e].map(function(t){return[t,n]})}):this.iterator("column",function(t,e){for(var n=$t(t),a=0,r=n.length;a<r;a++)if(n[a].col===e)return n[a].dir;return null},1)}),t("columns().orderable()","column().orderable()",function(n){return this.iterator("column",function(t,e){t=t.aoColumns[e];return n?t.asSorting:t.bSortable},1)}),e("processing()",function(e){return this.iterator("table",function(t){w(t,e)})}),e("search()",function(e,n,a,r){var t=this.context;return void 0===e?0!==t.length?t[0].oPreviousSearch.search:void 0:this.iterator("table",function(t){t.oFeatures.bFilter&&Nt(t,"object"==typeof n?V.extend(t.oPreviousSearch,n,{search:e}):V.extend(t.oPreviousSearch,{search:e,regex:null!==n&&n,smart:null===a||a,caseInsensitive:null===r||r}))})}),e("search.fixed()",function(e,n){var t=this.iterator(!0,"table",function(t){t=t.searchFixed;return e?void 0===n?t[e]:(null===n?delete t[e]:t[e]=n,this):Object.keys(t)});return void 0!==e&&void 0===n?t[0]:t}),t("columns().search()","column().search()",function(a,r,o,i){return this.iterator("column",function(t,e){var n=t.aoPreSearchCols;if(void 0===a)return n[e].search;t.oFeatures.bFilter&&("object"==typeof r?V.extend(n[e],r,{search:a}):V.extend(n[e],{search:a,regex:null!==r&&r,smart:null===o||o,caseInsensitive:null===i||i}),Nt(t,t.oPreviousSearch))})}),e(["columns().search.fixed()","column().search.fixed()"],function(n,a){var t=this.iterator(!0,"column",function(t,e){t=t.aoColumns[e].searchFixed;return n?void 0===a?t[n]:(null===a?delete t[n]:t[n]=a,this):Object.keys(t)});return void 0!==n&&void 0===a?t[0]:t}),e("state()",function(t,e){var n;return t?(n=V.extend(!0,{},t),this.iterator("table",function(t){!1!==e&&(n.time=+new Date+100),Jt(t,n,function(){})})):this.context.length?this.context[0].oSavedState:null}),e("state.clear()",function(){return this.iterator("table",function(t){t.fnStateSaveCallback.call(t.oInstance,t,{})})}),e("state.loaded()",function(){return this.context.length?this.context[0].oLoadedState:null}),e("state.save()",function(){return this.iterator("table",function(t){Gt(t)})}),$.use=function(t,e){"lib"===e||t.fn?V=t:"win"==e||t.document?_=(q=t).document:"datetime"!==e&&"DateTime"!==t.type||($.DateTime=t)},$.factory=function(t,e){var n=!1;return t&&t.document&&(_=(q=t).document),e&&e.fn&&e.fn.jquery&&(V=e,n=!0),n},$.versionCheck=function(t,e){for(var n,a,r=(e||$.version).split("."),o=t.split("."),i=0,l=o.length;i<l;i++)if((n=parseInt(r[i],10)||0)!==(a=parseInt(o[i],10)||0))return a<n;return!0},$.isDataTable=function(t){var r=V(t).get(0),o=!1;return t instanceof $.Api||(V.each($.settings,function(t,e){var n=e.nScrollHead?V("table",e.nScrollHead)[0]:null,a=e.nScrollFoot?V("table",e.nScrollFoot)[0]:null;e.nTable!==r&&n!==r&&a!==r||(o=!0)}),o)},$.tables=function(e){var t=!1,n=(V.isPlainObject(e)&&(t=e.api,e=e.visible),$.settings.filter(function(t){return!(e&&!V(t.nTable).is(":visible"))}).map(function(t){return t.nTable}));return t?new U(n):n},$.camelToHungarian=z,e("$()",function(t,e){e=this.rows(e).nodes(),e=V(e);return V([].concat(e.filter(t).toArray(),e.find(t).toArray()))}),V.each(["on","one","off"],function(t,n){e(n+"()",function(){var t=Array.prototype.slice.call(arguments),e=(t[0]=t[0].split(/\s/).map(function(t){return t.match(/\.dt\b/)?t:t+".dt"}).join(" "),V(this.tables().nodes()));return e[n].apply(e,t),this})}),e("clear()",function(){return this.iterator("table",function(t){pt(t)})}),e("error()",function(e){return this.iterator("table",function(t){Z(t,0,e)})}),e("settings()",function(){return new U(this.context,this.context)}),e("init()",function(){var t=this.context;return t.length?t[0].oInit:null}),e("data()",function(){return this.iterator("table",function(t){return f(t.aoData,"_aData")}).flatten()}),e("trigger()",function(e,n,a){return this.iterator("table",function(t){return tt(t,null,e,n,a)}).flatten()}),e("ready()",function(t){var e=this.context;return t?this.tables().every(function(){this.context[0]._bInitComplete?t.call(this):this.on("init",function(){t.call(this)})}):e.length?e[0]._bInitComplete||!1:null}),e("destroy()",function(c){return c=c||!1,this.iterator("table",function(t){var e=t.oClasses,n=t.nTable,a=t.nTBody,r=t.nTHead,o=t.nTFoot,i=V(n),a=V(a),l=V(t.nTableWrapper),s=t.aoData.map(function(t){return t?t.nTr:null}),u=e.order,o=(t.bDestroying=!0,tt(t,"aoDestroyCallback","destroy",[t],!0),c||new U(t).columns().visible(!0),l.off(".DT").find(":not(tbody *)").off(".DT"),V(q).off(".DT-"+t.sInstance),n!=r.parentNode&&(i.children("thead").detach(),i.append(r)),o&&n!=o.parentNode&&(i.children("tfoot").detach(),i.append(o)),t.colgroup.remove(),t.aaSorting=[],t.aaSortingFixed=[],Yt(t),V("th, td",r).removeClass(u.canAsc+" "+u.canDesc+" "+u.isAsc+" "+u.isDesc).css("width",""),a.children().detach(),a.append(s),t.nTableWrapper.parentNode),r=t.nTableWrapper.nextSibling,u=c?"remove":"detach",a=(i[u](),l[u](),!c&&o&&(o.insertBefore(n,r),i.css("width",t.sDestroyWidth).removeClass(e.table)),$.settings.indexOf(t));-1!==a&&$.settings.splice(a,1)})}),V.each(["column","row","cell"],function(t,s){e(s+"s().every()",function(a){var r,o=this.selector.opts,i=this,l=0;return this.iterator("every",function(t,e,n){r=i[s](e,o),"cell"===s?a.call(r,r[0][0].row,r[0][0].column,n,l):a.call(r,e,n,l),l++})})}),e("i18n()",function(t,e,n){var a=this.context[0],t=J(t)(a.oLanguage);return"string"==typeof(t=V.isPlainObject(t=void 0===t?e:t)?void 0!==n&&void 0!==t[n]?t[n]:t._:t)?t.replace("%d",n):t}),$.version="2.0.8",$.settings=[],$.models={},$.models.oSearch={caseInsensitive:!0,search:"",regex:!1,smart:!0,return:!1},$.models.oRow={nTr:null,anCells:null,_aData:[],_aSortData:null,_aFilterData:null,_sFilterRow:null,src:null,idx:-1,displayData:null},$.models.oColumn={idx:null,aDataSort:null,asSorting:null,bSearchable:null,bSortable:null,bVisible:null,_sManualType:null,_bAttrSrc:!1,fnCreatedCell:null,fnGetData:null,fnSetData:null,mData:null,mRender:null,sClass:null,sContentPadding:null,sDefaultContent:null,sName:null,sSortDataType:"std",sSortingClass:null,sTitle:null,sType:null,sWidth:null,sWidthOrig:null,maxLenString:null,searchFixed:null},$.defaults={aaData:null,aaSorting:[[0,"asc"]],aaSortingFixed:[],ajax:null,aLengthMenu:[10,25,50,100],aoColumns:null,aoColumnDefs:null,aoSearchCols:[],bAutoWidth:!0,bDeferRender:!0,bDestroy:!1,bFilter:!0,bInfo:!0,bLengthChange:!0,bPaginate:!0,bProcessing:!1,bRetrieve:!1,bScrollCollapse:!1,bServerSide:!1,bSort:!0,bSortMulti:!0,bSortCellsTop:null,bSortClasses:!0,bStateSave:!1,fnCreatedRow:null,fnDrawCallback:null,fnFooterCallback:null,fnFormatNumber:function(t){return t.toString().replace(/\B(?=(\d{3})+(?!\d))/g,this.oLanguage.sThousands)},fnHeaderCallback:null,fnInfoCallback:null,fnInitComplete:null,fnPreDrawCallback:null,fnRowCallback:null,fnStateLoadCallback:function(t){try{return JSON.parse((-1===t.iStateDuration?sessionStorage:localStorage).getItem("DataTables_"+t.sInstance+"_"+location.pathname))}catch(t){return{}}},fnStateLoadParams:null,fnStateLoaded:null,fnStateSaveCallback:function(t,e){try{(-1===t.iStateDuration?sessionStorage:localStorage).setItem("DataTables_"+t.sInstance+"_"+location.pathname,JSON.stringify(e))}catch(t){}},fnStateSaveParams:null,iStateDuration:7200,iDisplayLength:10,iDisplayStart:0,iTabIndex:0,oClasses:{},oLanguage:{oAria:{orderable:": Activate to sort",orderableReverse:": Activate to invert sorting",orderableRemove:": Activate to remove sorting",paginate:{first:"First",last:"Last",next:"Next",previous:"Previous"}},oPaginate:{sFirst:"«",sLast:"»",sNext:"›",sPrevious:"‹"},entries:{_:"entries",1:"entry"},sEmptyTable:"No data available in table",sInfo:"Showing _START_ to _END_ of _TOTAL_ _ENTRIES-TOTAL_",sInfoEmpty:"Showing 0 to 0 of 0 _ENTRIES-TOTAL_",sInfoFiltered:"(filtered from _MAX_ total _ENTRIES-MAX_)",sInfoPostFix:"",sDecimal:"",sThousands:",",sLengthMenu:"_MENU_ _ENTRIES_ per page",sLoadingRecords:"Loading...",sProcessing:"",sSearch:"Search:",sSearchPlaceholder:"",sUrl:"",sZeroRecords:"No matching records found"},oSearch:V.extend({},$.models.oSearch),layout:{topStart:"pageLength",topEnd:"search",bottomStart:"info",bottomEnd:"paging"},sDom:null,searchDelay:null,sPaginationType:"full_numbers",sScrollX:"",sScrollXInner:"",sScrollY:"",sServerMethod:"GET",renderer:null,rowId:"DT_RowId",caption:null},k($.defaults),$.defaults.column={aDataSort:null,iDataSort:-1,ariaTitle:"",asSorting:["asc","desc",""],bSearchable:!0,bSortable:!0,bVisible:!0,fnCreatedCell:null,mData:null,mRender:null,sCellType:"td",sClass:"",sContentPadding:"",sDefaultContent:null,sName:"",sSortDataType:"std",sTitle:null,sType:null,sWidth:null},k($.defaults.column),$.models.oSettings={oFeatures:{bAutoWidth:null,bDeferRender:null,bFilter:null,bInfo:!0,bLengthChange:!0,bPaginate:null,bProcessing:null,bServerSide:null,bSort:null,bSortMulti:null,bSortClasses:null,bStateSave:null},oScroll:{bCollapse:null,iBarWidth:0,sX:null,sXInner:null,sY:null},oLanguage:{fnInfoCallback:null},oBrowser:{bScrollbarLeft:!1,barWidth:0},ajax:null,aanFeatures:[],aoData:[],aiDisplay:[],aiDisplayMaster:[],aIds:{},aoColumns:[],aoHeader:[],aoFooter:[],oPreviousSearch:{},searchFixed:{},aoPreSearchCols:[],aaSorting:null,aaSortingFixed:[],sDestroyWidth:0,aoRowCallback:[],aoHeaderCallback:[],aoFooterCallback:[],aoDrawCallback:[],aoRowCreatedCallback:[],aoPreDrawCallback:[],aoInitComplete:[],aoStateSaveParams:[],aoStateLoadParams:[],aoStateLoaded:[],sTableId:"",nTable:null,nTHead:null,nTFoot:null,nTBody:null,nTableWrapper:null,bInitialised:!1,aoOpenRows:[],sDom:null,searchDelay:null,sPaginationType:"two_button",pagingControls:0,iStateDuration:0,aoStateSave:[],aoStateLoad:[],oSavedState:null,oLoadedState:null,bAjaxDataGet:!0,jqXHR:null,json:void 0,oAjaxData:void 0,sServerMethod:null,fnFormatNumber:null,aLengthMenu:null,iDraw:0,bDrawing:!1,iDrawError:-1,_iDisplayLength:10,_iDisplayStart:0,_iRecordsTotal:0,_iRecordsDisplay:0,oClasses:{},bFiltered:!1,bSorted:!1,bSortCellsTop:null,oInit:null,aoDestroyCallback:[],fnRecordsTotal:function(){return"ssp"==et(this)?+this._iRecordsTotal:this.aiDisplayMaster.length},fnRecordsDisplay:function(){return"ssp"==et(this)?+this._iRecordsDisplay:this.aiDisplay.length},fnDisplayEnd:function(){var t=this._iDisplayLength,e=this._iDisplayStart,n=e+t,a=this.aiDisplay.length,r=this.oFeatures,o=r.bPaginate;return r.bServerSide?!1===o||-1===t?e+a:Math.min(e+t,this._iRecordsDisplay):!o||a<n||-1===t?a:n},oInstance:null,sInstance:null,iTabIndex:0,nScrollHead:null,nScrollFoot:null,aLastSort:[],oPlugins:{},rowIdFn:null,rowId:null,caption:"",captionNode:null,colgroup:null},$.ext.pager);V.extend(be,{simple:function(){return["previous","next"]},full:function(){return["first","previous","next","last"]},numbers:function(){return["numbers"]},simple_numbers:function(){return["previous","numbers","next"]},full_numbers:function(){return["first","previous","numbers","next","last"]},first_last:function(){return["first","last"]},first_last_numbers:function(){return["first","numbers","last"]},_numbers:Ne,numbers_length:7}),V.extend(!0,$.ext.renderer,{pagingButton:{_:function(t,e,n,a,r){var t=t.oClasses.paging,o=[t.button];return a&&o.push(t.active),r&&o.push(t.disabled),{display:a="ellipsis"===e?V('<span class="ellipsis"></span>').html(n)[0]:V("<button>",{class:o.join(" "),role:"link",type:"button"}).html(n),clicker:a}}},pagingContainer:{_:function(t,e){return e}}});function De(t){return t.replace(/[\W]/g,"_")}function xe(t,e,n,a,r){return q.moment?t[e](r):q.luxon?t[n](r):a?t[a](r):t}var Se=!1;function Te(t,e,n){var a;if(q.moment){if(!(a=q.moment.utc(t,e,n,!0)).isValid())return null}else if(q.luxon){if(!(a=e&&"string"==typeof t?q.luxon.DateTime.fromFormat(t,e):q.luxon.DateTime.fromISO(t)).isValid)return null;a.setLocale(n)}else e?(Se||alert("DataTables warning: Formatted date without Moment.js or Luxon - https://datatables.net/tn/17"),Se=!0):a=new Date(t);return a}function we(s){return function(a,r,o,i){0===arguments.length?(o="en",a=r=null):1===arguments.length?(o="en",r=a,a=null):2===arguments.length&&(o=r,r=a,a=null);var l="datetime"+(r?"-"+De(r):"");return $.ext.type.order[l]||$.type(l,{detect:function(t){return t===l&&l},order:{pre:function(t){return t.valueOf()}},className:"dt-right"}),function(t,e){var n;return null==t&&(t="--now"===i?(n=new Date,new Date(Date.UTC(n.getFullYear(),n.getMonth(),n.getDate(),n.getHours(),n.getMinutes(),n.getSeconds()))):""),"type"===e?l:""===t?"sort"!==e?"":Te("0000-01-01 00:00:00",null,o):!(null===r||a!==r||"sort"===e||"type"===e||t instanceof Date)||null===(n=Te(t,a,o))?t:"sort"===e?n:(t=null===r?xe(n,"toDate","toJSDate","")[s]():xe(n,"format","toFormat","toISOString",r),"display"===e?u(t):t)}}}var _e=",",Ce=".";if(void 0!==q.Intl)try{for(var Ie=(new Intl.NumberFormat).formatToParts(100000.1),a=0;a<Ie.length;a++)"group"===Ie[a].type?_e=Ie[a].value:"decimal"===Ie[a].type&&(Ce=Ie[a].value)}catch(t){}$.datetime=function(n,a){var r="datetime-detect-"+De(n);a=a||"en",$.ext.type.order[r]||$.type(r,{detect:function(t){var e=Te(t,n,a);return!(""!==t&&!e)&&r},order:{pre:function(t){return Te(t,n,a)||0}},className:"dt-right"})},$.render={date:we("toLocaleDateString"),datetime:we("toLocaleString"),time:we("toLocaleTimeString"),number:function(r,o,i,l,s){return null==r&&(r=_e),null==o&&(o=Ce),{display:function(t){if("number"!=typeof t&&"string"!=typeof t)return t;if(""===t||null===t)return t;var e=t<0?"-":"",n=parseFloat(t),a=Math.abs(n);if(1e11<=a||a<1e-4&&0!==a)return(a=n.toExponential(i).split(/e\+?/))[0]+" x 10<sup>"+a[1]+"</sup>";if(isNaN(n))return u(t);n=n.toFixed(i),t=Math.abs(n);a=parseInt(t,10),n=i?o+(t-a).toFixed(i).substring(2):"";return(e=0===a&&0===parseFloat(n)?"":e)+(l||"")+a.toString().replace(/\B(?=(\d{3})+(?!\d))/g,r)+n+(s||"")}}},text:function(){return{display:u,filter:u}}};var i=$.ext.type,Ae=($.type=function(a,t,e){if(!t)return{className:i.className[a],detect:i.detect.find(function(t){return t.name===a}),order:{pre:i.order[a+"-pre"],asc:i.order[a+"-asc"],desc:i.order[a+"-desc"]},render:i.render[a],search:i.search[a]};function n(t,e){i[t][a]=e}function r(n){function t(t,e){return!0===(t=n(t,e))?a:t}Object.defineProperty(t,"name",{value:a});var e=i.detect.findIndex(function(t){return t.name===a});-1===e?i.detect.unshift(t):i.detect.splice(e,1,t)}function o(t){i.order[a+"-pre"]=t.pre,i.order[a+"-asc"]=t.asc,i.order[a+"-desc"]=t.desc}void 0===e&&(e=t,t=null),"className"===t?n("className",e):"detect"===t?r(e):"order"===t?o(e):"render"===t?n("render",e):"search"===t?n("search",e):t||(e.className&&n("className",e.className),void 0!==e.detect&&r(e.detect),e.order&&o(e.order),void 0!==e.render&&n("render",e.render),void 0!==e.search&&n("search",e.search))},$.types=function(){return i.detect.map(function(t){return t.name})},$.type("string",{detect:function(){return"string"},order:{pre:function(t){return y(t)?"":"string"==typeof t?t.toLowerCase():t.toString?t.toString():""}},search:fe(!1,!0)}),$.type("html",{detect:function(t){return y(t)||"string"==typeof t&&-1!==t.indexOf("<")?"html":null},order:{pre:function(t){return y(t)?"":t.replace?I(t).trim().toLowerCase():t+""}},search:fe(!0,!0)}),$.type("date",{className:"dt-type-date",detect:function(t){var e;return(!t||t instanceof Date||N.test(t))&&(null!==(e=Date.parse(t))&&!isNaN(e)||y(t))?"date":null},order:{pre:function(t){t=Date.parse(t);return isNaN(t)?-1/0:t}}}),$.type("html-num-fmt",{className:"dt-type-numeric",detect:function(t,e){e=e.oLanguage.sDecimal;return l(t,e,!0)?"html-num-fmt":null},order:{pre:function(t,e){e=e.oLanguage.sDecimal;return Ae(t,e,L,P)}},search:fe(!0,!0)}),$.type("html-num",{className:"dt-type-numeric",detect:function(t,e){e=e.oLanguage.sDecimal;return l(t,e)?"html-num":null},order:{pre:function(t,e){e=e.oLanguage.sDecimal;return Ae(t,e,L)}},search:fe(!0,!0)}),$.type("num-fmt",{className:"dt-type-numeric",detect:function(t,e){e=e.oLanguage.sDecimal;return o(t,e,!0)?"num-fmt":null},order:{pre:function(t,e){e=e.oLanguage.sDecimal;return Ae(t,e,P)}}}),$.type("num",{className:"dt-type-numeric",detect:function(t,e){e=e.oLanguage.sDecimal;return o(t,e)?"num":null},order:{pre:function(t,e){e=e.oLanguage.sDecimal;return Ae(t,e)}}}),function(t,e,n,a){var r;return 0===t||t&&"-"!==t?"number"==(r=typeof t)||"bigint"==r?t:+(t=(t=e?R(t,e):t).replace&&(n&&(t=t.replace(n,"")),a)?t.replace(a,""):t):-1/0});V.extend(!0,$.ext.renderer,{footer:{_:function(t,e,n){e.addClass(n.tfoot.cell)}},header:{_:function(d,f,h){f.addClass(h.thead.cell),d.oFeatures.bSort||f.addClass(h.order.none);var t=d.bSortCellsTop,e=f.closest("thead").find("tr"),n=f.parent().index();"disable"===f.attr("data-dt-order")||"disable"===f.parent().attr("data-dt-order")||!0===t&&0!==n||!1===t&&n!==e.length-1||V(d.nTable).on("order.dt.DT",function(t,e,n){var a,r,o,i,l,s,u,c;d===e&&(a=h.order,c=e.api.columns(f),r=d.aoColumns[c.flatten()[0]],o=c.orderable().includes(!0),i="",u=c.indexes(),l=c.orderable(!0).flatten(),s=","+n.map(function(t){return t.col}).join(",")+",",f.removeClass(a.isAsc+" "+a.isDesc).toggleClass(a.none,!o).toggleClass(a.canAsc,o&&l.includes("asc")).toggleClass(a.canDesc,o&&l.includes("desc")),-1!==(l=s.indexOf(","+u.toArray().join(",")+","))&&(s=c.order(),f.addClass(s.includes("asc")?a.isAsc:""+s.includes("desc")?a.isDesc:"")),0===l?(u=n[0],c=r.asSorting,f.attr("aria-sort","asc"===u.dir?"ascending":"descending"),i=c[u.index+1]?"Reverse":"Remove"):f.removeAttr("aria-sort"),f.attr("aria-label",o?r.ariaTitle+e.api.i18n("oAria.orderable"+i):r.ariaTitle),o)&&(f.find(".dt-column-title").attr("role","button"),f.attr("tabindex",0))})}},layout:{_:function(t,e,n){var a=V("<div/>").addClass("dt-layout-row").appendTo(e);V.each(n,function(t,e){t=e.table?"":"dt-"+t+" ";e.table&&a.addClass("dt-layout-table"),V("<div/>").attr({id:e.id||null,class:"dt-layout-cell "+t+(e.className||"")}).append(e.contents).appendTo(a)})}}}),$.feature={},$.feature.register=function(t,e,n){$.ext.features[t]=e,n&&C.feature.push({cFeature:n,fnInit:e})},$.feature.register("info",function(t,s){var e,n,u;return t.oFeatures.bInfo?(e=t.oLanguage,n=t.sTableId,u=V("<div/>",{class:t.oClasses.info.container}),s=V.extend({callback:e.fnInfoCallback,empty:e.sInfoEmpty,postfix:e.sInfoPostFix,search:e.sInfoFiltered,text:e.sInfo},s),t.aoDrawCallback.push(function(t){var e=s,n=u,a=t._iDisplayStart+1,r=t.fnDisplayEnd(),o=t.fnRecordsTotal(),i=t.fnRecordsDisplay(),l=i?e.text:e.empty;i!==o&&(l+=" "+e.search),l+=e.postfix,l=ee(t,l),e.callback&&(l=e.callback.call(t.oInstance,t,a,r,o,i,l)),n.html(l),tt(t,null,"info",[t,n[0],l])}),t._infoEl||(u.attr({"aria-live":"polite",id:n+"_info",role:"status"}),V(t.nTable).attr("aria-describedby",n+"_info"),t._infoEl=u),u):null},"i");var Le=0;function Fe(t,e,n,a){var r=t.oLanguage.oPaginate,o={display:"",active:!1,disabled:!1};switch(e){case"ellipsis":o.display="…",o.disabled=!0;break;case"first":o.display=r.sFirst,0===n&&(o.disabled=!0);break;case"previous":o.display=r.sPrevious,0===n&&(o.disabled=!0);break;case"next":o.display=r.sNext,0!==a&&n!==a-1||(o.disabled=!0);break;case"last":o.display=r.sLast,0!==a&&n!==a-1||(o.disabled=!0);break;default:"number"==typeof e&&(o.display=t.fnFormatNumber(e+1),n===e)&&(o.active=!0)}return o}function Ne(t,e,n,a){var r=[],o=Math.floor(n/2),i=a?2:1,l=a?1:0;return e<=n?r=h(0,e):1===n?r=[t]:3===n?t<=1?r=[0,1,"ellipsis"]:e-2<=t?(r=h(e-2,e)).unshift("ellipsis"):r=["ellipsis",t,"ellipsis"]:t<=o?((r=h(0,n-i)).push("ellipsis"),a&&r.push(e-1)):e-1-o<=t?((r=h(e-(n-i),e)).unshift("ellipsis"),a&&r.unshift(0)):((r=h(t-o+i,t+o-l)).push("ellipsis"),r.unshift("ellipsis"),a&&(r.push(e-1),r.unshift(0))),r}$.feature.register("search",function(n,t){var e,a,r,o,i,l,s,u,c,d;return n.oFeatures.bFilter?(e=n.oClasses.search,a=n.sTableId,c=n.oLanguage,r=n.oPreviousSearch,o='<input type="search" class="'+e.input+'"/>',-1===(t=V.extend({placeholder:c.sSearchPlaceholder,text:c.sSearch},t)).text.indexOf("_INPUT_")&&(t.text+="_INPUT_"),t.text=ee(n,t.text),c=t.text.match(/_INPUT_$/),s=t.text.match(/^_INPUT_/),i=t.text.replace(/_INPUT_/,""),l="<label>"+t.text+"</label>",s?l="_INPUT_<label>"+i+"</label>":c&&(l="<label>"+i+"</label>_INPUT_"),(s=V("<div>").addClass(e.container).append(l.replace(/_INPUT_/,o))).find("label").attr("for","dt-search-"+Le),s.find("input").attr("id","dt-search-"+Le),Le++,u=function(t){var e=this.value;r.return&&"Enter"!==t.key||e!=r.search&&(r.search=e,Nt(n,r),n._iDisplayStart=0,S(n))},c=null!==n.searchDelay?n.searchDelay:0,d=V("input",s).val(r.search).attr("placeholder",t.placeholder).on("keyup.DT search.DT input.DT paste.DT cut.DT",c?$.util.debounce(u,c):u).on("mouseup.DT",function(t){setTimeout(function(){u.call(d[0],t)},10)}).on("keypress.DT",function(t){if(13==t.keyCode)return!1}).attr("aria-controls",a),V(n.nTable).on("search.dt.DT",function(t,e){n===e&&d[0]!==_.activeElement&&d.val("function"!=typeof r.search?r.search:"")}),s):null},"f"),$.feature.register("paging",function(t,e){if(!t.oFeatures.bPaginate)return null;(e=V.extend({buttons:$.ext.pager.numbers_length,type:t.sPaginationType,boundaryNumbers:!0},e)).numbers&&(e.buttons=e.numbers);function n(){!function t(e,n,a){if(!e._bInitComplete)return;var r=$.ext.pager[a.type],o=e.oLanguage.oAria.paginate||{},i=e._iDisplayStart,l=e._iDisplayLength,s=e.fnRecordsDisplay(),u=-1===l,c=u?0:Math.ceil(i/l),d=u?1:Math.ceil(s/l),f=r().map(function(t){return"numbers"===t?Ne(c,d,a.buttons,a.boundaryNumbers):t}).flat();var h=[];for(var p=0;p<f.length;p++){var g=f[p],m=Fe(e,g,c,d),v=te(e,"pagingButton")(e,g,m.display,m.active,m.disabled);V(v.clicker).attr({"aria-controls":e.sTableId,"aria-disabled":m.disabled?"true":null,"aria-current":m.active?"page":null,"aria-label":o[g],"data-dt-idx":g,tabIndex:m.disabled?-1:e.iTabIndex}),"number"!=typeof g&&V(v.clicker).addClass(g),Qt(v.clicker,{action:g},function(t){t.preventDefault(),Ht(e,t.data.action,!0)}),h.push(v.display)}i=te(e,"pagingContainer")(e,h);u=n.find(_.activeElement).data("dt-idx");n.empty().append(i);void 0!==u&&n.find("[data-dt-idx="+u+"]").trigger("focus");h.length&&1<a.numbers&&V(n).height()>=2*V(h[0]).outerHeight()-10&&t(e,n,V.extend({},a,{numbers:a.numbers-2}))}(t,a,e)}var a=V("<div/>").addClass(t.oClasses.paging.container+" paging_"+e.type);return t.aoDrawCallback.push(n),V(t.nTable).on("column-sizing.dt.DT",n),a},"p");var je=0;return $.feature.register("pageLength",function(a,t){var e=a.oFeatures;if(!e.bPaginate||!e.bLengthChange)return null;t=V.extend({menu:a.aLengthMenu,text:a.oLanguage.sLengthMenu},t);var e=a.oClasses.length,n=a.sTableId,r=t.menu,o=[],i=[];if(Array.isArray(r[0]))o=r[0],i=r[1];else for(p=0;p<r.length;p++)V.isPlainObject(r[p])?(o.push(r[p].value),i.push(r[p].label)):(o.push(r[p]),i.push(r[p]));for(var l=t.text.match(/_MENU_$/),s=t.text.match(/^_MENU_/),u=t.text.replace(/_MENU_/,""),t="<label>"+t.text+"</label>",c=(s?t="_MENU_<label>"+u+"</label>":l&&(t="<label>"+u+"</label>_MENU_"),V("<div/>").addClass(e.container).append(t.replace("_MENU_","<span></span>"))),d=[],f=(c.find("label")[0].childNodes.forEach(function(t){t.nodeType===Node.TEXT_NODE&&d.push({el:t,text:t.textContent})}),function(e){d.forEach(function(t){t.el.textContent=ee(a,t.text,e)})}),h=V("<select/>",{name:n+"_length","aria-controls":n,class:e.select}),p=0;p<o.length;p++)h[0][p]=new Option("number"==typeof i[p]?a.fnFormatNumber(i[p]):i[p],o[p]);return c.find("label").attr("for","dt-length-"+je),h.attr("id","dt-length-"+je),je++,c.find("span").replaceWith(h),V("select",c).val(a._iDisplayLength).on("change.DT",function(){Mt(a,V(this).val()),S(a)}),V(a.nTable).on("length.dt.DT",function(t,e,n){a===e&&(V("select",c).val(n),f(n))}),f(a._iDisplayLength),c},"l"),((V.fn.dataTable=$).$=V).fn.dataTableSettings=$.settings,V.fn.dataTableExt=$.ext,V.fn.DataTable=function(t){return V(this).dataTable(t).api()},V.each($,function(t,e){V.fn.DataTable[t]=e}),$});node_modules/datatables.net/js/dataTables.min.mjs000064400000305566151676725350016127 0ustar00/*! DataTables 2.0.8 * © SpryMedia Ltd - datatables.net/license */ import jQuery from"jquery";var _ext,_Api,_api_register,_api_registerPlural,$=jQuery,DataTable=function(e,v){var D,y,T;return DataTable.factory(e,v)?DataTable:this instanceof DataTable?$(e).DataTable(v):(y=void 0===(v=e),T=(D=this).length,y&&(v={}),this.api=function(){return new _Api(this)},this.each(function(){var a=1<T?_fnExtend({},v,!0):v,n=0,e=this.getAttribute("id"),r=!1,t=DataTable.defaults,i=$(this);if("table"!=this.nodeName.toLowerCase())_fnLog(null,0,"Non-table node initialisation ("+this.nodeName+")",2);else{$(this).trigger("options.dt",a),_fnCompatOpts(t),_fnCompatCols(t.column),_fnCamelToHungarian(t,t,!0),_fnCamelToHungarian(t.column,t.column,!0),_fnCamelToHungarian(t,$.extend(a,i.data()),!0);for(var o=DataTable.settings,n=0,l=o.length;n<l;n++){var s=o[n];if(s.nTable==this||s.nTHead&&s.nTHead.parentNode==this||s.nTFoot&&s.nTFoot.parentNode==this){var u=(void 0!==a.bRetrieve?a:t).bRetrieve,c=(void 0!==a.bDestroy?a:t).bDestroy;if(y||u)return s.oInstance;if(c){new DataTable.Api(s).destroy();break}return void _fnLog(s,0,"Cannot reinitialise DataTable",3)}if(s.sTableId==this.id){o.splice(n,1);break}}null!==e&&""!==e||(e="DataTables_Table_"+DataTable.ext._unique++,this.id=e);var f=$.extend(!0,{},DataTable.models.oSettings,{sDestroyWidth:i[0].style.width,sInstance:e,sTableId:e,colgroup:$("<colgroup>").prependTo(this),fastData:function(e,t,a){return _fnGetCellData(f,e,t,a)}}),d=(f.nTable=this,f.oInit=a,o.push(f),f.api=new _Api(f),f.oInstance=1===D.length?D:i.dataTable(),_fnCompatOpts(a),a.aLengthMenu&&!a.iDisplayLength&&(a.iDisplayLength=Array.isArray(a.aLengthMenu[0])?a.aLengthMenu[0][0]:$.isPlainObject(a.aLengthMenu[0])?a.aLengthMenu[0].value:a.aLengthMenu[0]),a=_fnExtend($.extend(!0,{},t),a),_fnMap(f.oFeatures,a,["bPaginate","bLengthChange","bFilter","bSort","bSortMulti","bInfo","bProcessing","bAutoWidth","bSortClasses","bServerSide","bDeferRender"]),_fnMap(f,a,["ajax","fnFormatNumber","sServerMethod","aaSorting","aaSortingFixed","aLengthMenu","sPaginationType","iStateDuration","bSortCellsTop","iTabIndex","sDom","fnStateLoadCallback","fnStateSaveCallback","renderer","searchDelay","rowId","caption","layout",["iCookieDuration","iStateDuration"],["oSearch","oPreviousSearch"],["aoSearchCols","aoPreSearchCols"],["iDisplayLength","_iDisplayLength"]]),_fnMap(f.oScroll,a,[["sScrollX","sX"],["sScrollXInner","sXInner"],["sScrollY","sY"],["bScrollCollapse","bCollapse"]]),_fnMap(f.oLanguage,a,"fnInfoCallback"),_fnCallbackReg(f,"aoDrawCallback",a.fnDrawCallback),_fnCallbackReg(f,"aoStateSaveParams",a.fnStateSaveParams),_fnCallbackReg(f,"aoStateLoadParams",a.fnStateLoadParams),_fnCallbackReg(f,"aoStateLoaded",a.fnStateLoaded),_fnCallbackReg(f,"aoRowCallback",a.fnRowCallback),_fnCallbackReg(f,"aoRowCreatedCallback",a.fnCreatedRow),_fnCallbackReg(f,"aoHeaderCallback",a.fnHeaderCallback),_fnCallbackReg(f,"aoFooterCallback",a.fnFooterCallback),_fnCallbackReg(f,"aoInitComplete",a.fnInitComplete),_fnCallbackReg(f,"aoPreDrawCallback",a.fnPreDrawCallback),f.rowIdFn=_fnGetObjectDataFn(a.rowId),_fnBrowserDetect(f),f.oClasses),_=($.extend(d,DataTable.ext.classes,a.oClasses),i.addClass(d.table),f.oFeatures.bPaginate||(a.iDisplayStart=0),void 0===f.iInitDisplayStart&&(f.iInitDisplayStart=a.iDisplayStart,f._iDisplayStart=a.iDisplayStart),f.oLanguage),p=($.extend(!0,_,a.oLanguage),_.sUrl?($.ajax({dataType:"json",url:_.sUrl,success:function(e){_fnCamelToHungarian(t.oLanguage,e),$.extend(!0,_,e,f.oInit.oLanguage),_fnCallbackFire(f,null,"i18n",[f],!0),_fnInitialise(f)},error:function(){_fnLog(f,0,"i18n file loading error",21),_fnInitialise(f)}}),r=!0):_fnCallbackFire(f,null,"i18n",[f]),[]),h=this.getElementsByTagName("thead"),e=_fnDetectHeader(f,h[0]);if(a.aoColumns)p=a.aoColumns;else if(e.length)for(l=e[n=0].length;n<l;n++)p.push(null);for(n=0,l=p.length;n<l;n++)_fnAddColumn(f);_fnApplyColumnDefs(f,a.aoColumnDefs,p,e,function(e,t){_fnColumnOptions(f,e,t)});function g(){if(void 0===a.aaSorting){var e=f.aaSorting;for(n=0,l=e.length;n<l;n++)e[n][1]=f.aoColumns[n].asSorting[0]}_fnSortingClasses(f),_fnCallbackReg(f,"aoDrawCallback",function(){(f.bSorted||"ssp"===_fnDataSource(f)||b.bDeferRender)&&_fnSortingClasses(f)});var t=i.children("caption"),t=(f.caption&&(t=0===t.length?$("<caption/>").appendTo(i):t).html(f.caption),t.length&&(t[0]._captionSide=t.css("caption-side"),f.captionNode=t[0]),0===h.length&&(h=$("<thead/>").appendTo(i)),f.nTHead=h[0],$("tr",h).addClass(d.thead.row),i.children("tbody")),t=(0===t.length&&(t=$("<tbody/>").insertAfter(h)),f.nTBody=t[0],i.children("tfoot"));if(0===t.length&&(t=$("<tfoot/>").appendTo(i)),f.nTFoot=t[0],$("tr",t).addClass(d.tfoot.row),a.aaData)for(n=0;n<a.aaData.length;n++)_fnAddData(f,a.aaData[n]);else"dom"==_fnDataSource(f)&&_fnAddTr(f,$(f.nTBody).children("tr"));f.aiDisplay=f.aiDisplayMaster.slice(),!(f.bInitialised=!0)===r&&_fnInitialise(f)}var m,e=i.children("tbody").find("tr").eq(0),b=(e.length&&(m=function(e,t){return null!==e.getAttribute("data-"+t)?t:null},$(e[0]).children("th, td").each(function(e,t){var a,n=f.aoColumns[e];n||_fnLog(f,0,"Incorrect column count",18),n.mData===e&&(a=m(t,"sort")||m(t,"order"),t=m(t,"filter")||m(t,"search"),null===a&&null===t||(n.mData={_:e+".display",sort:null!==a?e+".@data-"+a:void 0,type:null!==a?e+".@data-"+a:void 0,filter:null!==t?e+".@data-"+t:void 0},n._isArrayHost=!0,_fnColumnOptions(f,e)))})),f.oFeatures);_fnCallbackReg(f,"aoDrawCallback",_fnSaveState),a.bStateSave?(b.bStateSave=!0,_fnLoadState(f,a,g)):g()}}),D=null,this)},_re_dic=(DataTable.ext=_ext={buttons:{},classes:{},builder:"-source-",errMode:"alert",feature:[],features:{},search:[],selector:{cell:[],column:[],row:[]},legacy:{ajax:null},pager:{},renderer:{pageButton:{},header:{}},order:{},type:{className:{},detect:[],render:{},search:{},order:{}},_unique:0,fnVersionCheck:DataTable.fnVersionCheck,iApiIndex:0,sVersion:DataTable.version},$.extend(_ext,{afnFiltering:_ext.search,aTypes:_ext.type.detect,ofnSearch:_ext.type.search,oSort:_ext.type.order,afnSortData:_ext.order,aoFeatures:_ext.feature,oStdClasses:_ext.classes,oPagination:_ext.pager}),$.extend(DataTable.ext.classes,{container:"dt-container",empty:{row:"dt-empty"},info:{container:"dt-info"},length:{container:"dt-length",select:"dt-input"},order:{canAsc:"dt-orderable-asc",canDesc:"dt-orderable-desc",isAsc:"dt-ordering-asc",isDesc:"dt-ordering-desc",none:"dt-orderable-none",position:"sorting_"},processing:{container:"dt-processing"},scrolling:{body:"dt-scroll-body",container:"dt-scroll",footer:{self:"dt-scroll-foot",inner:"dt-scroll-footInner"},header:{self:"dt-scroll-head",inner:"dt-scroll-headInner"}},search:{container:"dt-search",input:"dt-input"},table:"dataTable",tbody:{cell:"",row:""},thead:{cell:"",row:""},tfoot:{cell:"",row:""},paging:{active:"current",button:"dt-paging-button",container:"dt-paging",disabled:"disabled"}}),{}),_re_new_lines=/[\r\n\u2028]/g,_re_html=/<([^>]*>)/g,_max_str_len=Math.pow(2,28),_re_date=/^\d{2,4}[./-]\d{1,2}[./-]\d{1,2}([T ]{1}\d{1,2}[:.]\d{2}([.:]\d{2})?)?$/,_re_escape_regex=new RegExp("(\\"+["/",".","*","+","?","|","(",")","[","]","{","}","\\","$","^","-"].join("|\\")+")","g"),_re_formatted_numeric=/['\u00A0,$£€¥%\u2009\u202F\u20BD\u20a9\u20BArfkɃΞ]/gi,_empty=function(e){return!e||!0===e||"-"===e},_intVal=function(e){var t=parseInt(e,10);return!isNaN(t)&&isFinite(e)?t:null},_numToDecimal=function(e,t){return _re_dic[t]||(_re_dic[t]=new RegExp(_fnEscapeRegex(t),"g")),"string"==typeof e&&"."!==t?e.replace(/\./g,"").replace(_re_dic[t],"."):e},_isNumber=function(e,t,a){var n=typeof e,r="string"==n;return"number"==n||"bigint"==n||!!_empty(e)||(t&&r&&(e=_numToDecimal(e,t)),a&&r&&(e=e.replace(_re_formatted_numeric,"")),!isNaN(parseFloat(e))&&isFinite(e))},_isHtml=function(e){return _empty(e)||"string"==typeof e},_htmlNumeric=function(e,t,a){return!!_empty(e)||("string"!=typeof e||!e.match(/<(input|select)/i))&&_isHtml(e)&&!!_isNumber(_stripHtml(e),t,a)||null},_pluck=function(e,t,a){var n=[],r=0,i=e.length;if(void 0!==a)for(;r<i;r++)e[r]&&e[r][t]&&n.push(e[r][t][a]);else for(;r<i;r++)e[r]&&n.push(e[r][t]);return n},_pluck_order=function(e,t,a,n){var r=[],i=0,o=t.length;if(void 0!==n)for(;i<o;i++)e[t[i]][a]&&r.push(e[t[i]][a][n]);else for(;i<o;i++)e[t[i]]&&r.push(e[t[i]][a]);return r},_range=function(e,t){var a,n=[];void 0===t?(t=0,a=e):(a=t,t=e);for(var r=t;r<a;r++)n.push(r);return n},_removeEmpty=function(e){for(var t=[],a=0,n=e.length;a<n;a++)e[a]&&t.push(e[a]);return t},_stripHtml=function(e){if(e.length>_max_str_len)throw new Error("Exceeded max str len");var t;for(e=e.replace(_re_html,"");(e=(t=e).replace(/<script/i,""))!==t;);return t},_escapeHtml=function(e){return"string"==typeof(e=Array.isArray(e)?e.join(","):e)?e.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,"""):e},_normalize=function(e,t){var a;return"string"!=typeof e?e:(a=e.normalize("NFD")).length!==e.length?(!0===t?e+" ":"")+a.replace(/[\u0300-\u036f]/g,""):a},_areAllUnique=function(e){if(!(e.length<2))for(var t=e.slice().sort(),a=t[0],n=1,r=t.length;n<r;n++){if(t[n]===a)return!1;a=t[n]}return!0},_unique=function(e){if(Array.from&&Set)return Array.from(new Set(e));if(_areAllUnique(e))return e.slice();var t,a,n,r=[],i=e.length,o=0;e:for(a=0;a<i;a++){for(t=e[a],n=0;n<o;n++)if(r[n]===t)continue e;r.push(t),o++}return r},_flatten=function(e,t){if(Array.isArray(t))for(var a=0;a<t.length;a++)_flatten(e,t[a]);else e.push(t);return e};function _addClass(t,e){e&&e.split(" ").forEach(function(e){e&&t.classList.add(e)})}function _fnHungarianMap(t){var a,n,r={};$.each(t,function(e){(a=e.match(/^([^A-Z]+?)([A-Z])/))&&-1!=="a aa ai ao as b fn i m o s ".indexOf(a[1]+" ")&&(n=e.replace(a[0],a[2].toLowerCase()),r[n]=e,"o"===a[1])&&_fnHungarianMap(t[e])}),t._hungarianMap=r}function _fnCamelToHungarian(t,a,n){var r;t._hungarianMap||_fnHungarianMap(t),$.each(a,function(e){void 0===(r=t._hungarianMap[e])||!n&&void 0!==a[r]||("o"===r.charAt(0)?(a[r]||(a[r]={}),$.extend(!0,a[r],a[e]),_fnCamelToHungarian(t[r],a[r],n)):a[r]=a[e])})}DataTable.util={diacritics:function(e,t){if("function"!=typeof e)return _normalize(e,t);_normalize=e},debounce:function(a,n){var r;return function(){var e=this,t=arguments;clearTimeout(r),r=setTimeout(function(){a.apply(e,t)},n||250)}},throttle:function(n,e){var r,i,o=void 0!==e?e:200;return function(){var e=this,t=+new Date,a=arguments;r&&t<r+o?(clearTimeout(i),i=setTimeout(function(){r=void 0,n.apply(e,a)},o)):(r=t,n.apply(e,a))}},escapeRegex:function(e){return e.replace(_re_escape_regex,"\\$1")},set:function(n){var d;return $.isPlainObject(n)?DataTable.util.set(n._):null===n?function(){}:"function"==typeof n?function(e,t,a){n(e,"set",t,a)}:"string"!=typeof n||-1===n.indexOf(".")&&-1===n.indexOf("[")&&-1===n.indexOf("(")?function(e,t){e[n]=t}:(d=function(e,t,a){for(var n,r,i,o,l=_fnSplitObjNotation(a),a=l[l.length-1],s=0,u=l.length-1;s<u;s++){if("__proto__"===l[s]||"constructor"===l[s])throw new Error("Cannot set prototype values");if(n=l[s].match(__reArray),r=l[s].match(__reFn),n){if(l[s]=l[s].replace(__reArray,""),e[l[s]]=[],(n=l.slice()).splice(0,s+1),o=n.join("."),Array.isArray(t))for(var c=0,f=t.length;c<f;c++)d(i={},t[c],o),e[l[s]].push(i);else e[l[s]]=t;return}r&&(l[s]=l[s].replace(__reFn,""),e=e[l[s]](t)),null!==e[l[s]]&&void 0!==e[l[s]]||(e[l[s]]={}),e=e[l[s]]}a.match(__reFn)?e[a.replace(__reFn,"")](t):e[a.replace(__reArray,"")]=t},function(e,t){return d(e,t,n)})},get:function(r){var i,d;return $.isPlainObject(r)?(i={},$.each(r,function(e,t){t&&(i[e]=DataTable.util.get(t))}),function(e,t,a,n){var r=i[t]||i._;return void 0!==r?r(e,t,a,n):e}):null===r?function(e){return e}:"function"==typeof r?function(e,t,a,n){return r(e,t,a,n)}:"string"!=typeof r||-1===r.indexOf(".")&&-1===r.indexOf("[")&&-1===r.indexOf("(")?function(e){return e[r]}:(d=function(e,t,a){var n,r,i;if(""!==a)for(var o=_fnSplitObjNotation(a),l=0,s=o.length;l<s;l++){if(f=o[l].match(__reArray),n=o[l].match(__reFn),f){if(o[l]=o[l].replace(__reArray,""),""!==o[l]&&(e=e[o[l]]),r=[],o.splice(0,l+1),i=o.join("."),Array.isArray(e))for(var u=0,c=e.length;u<c;u++)r.push(d(e[u],t,i));var f=f[0].substring(1,f[0].length-1);e=""===f?r:r.join(f);break}if(n)o[l]=o[l].replace(__reFn,""),e=e[o[l]]();else{if(null===e||null===e[o[l]])return null;if(void 0===e||void 0===e[o[l]])return;e=e[o[l]]}}return e},function(e,t){return d(e,t,r)})},stripHtml:function(e){var t=typeof e;if("function"!=t)return"string"==t?_stripHtml(e):e;_stripHtml=e},escapeHtml:function(e){var t=typeof e;if("function"!=t)return"string"==t||Array.isArray(e)?_escapeHtml(e):e;_escapeHtml=e},unique:_unique};var _fnCompatMap=function(e,t,a){void 0!==e[t]&&(e[a]=e[t])};function _fnCompatOpts(e){_fnCompatMap(e,"ordering","bSort"),_fnCompatMap(e,"orderMulti","bSortMulti"),_fnCompatMap(e,"orderClasses","bSortClasses"),_fnCompatMap(e,"orderCellsTop","bSortCellsTop"),_fnCompatMap(e,"order","aaSorting"),_fnCompatMap(e,"orderFixed","aaSortingFixed"),_fnCompatMap(e,"paging","bPaginate"),_fnCompatMap(e,"pagingType","sPaginationType"),_fnCompatMap(e,"pageLength","iDisplayLength"),_fnCompatMap(e,"searching","bFilter"),"boolean"==typeof e.sScrollX&&(e.sScrollX=e.sScrollX?"100%":""),"boolean"==typeof e.scrollX&&(e.scrollX=e.scrollX?"100%":"");var t=e.aoSearchCols;if(t)for(var a=0,n=t.length;a<n;a++)t[a]&&_fnCamelToHungarian(DataTable.models.oSearch,t[a]);e.serverSide&&!e.searchDelay&&(e.searchDelay=400)}function _fnCompatCols(e){_fnCompatMap(e,"orderable","bSortable"),_fnCompatMap(e,"orderData","aDataSort"),_fnCompatMap(e,"orderSequence","asSorting"),_fnCompatMap(e,"orderDataType","sortDataType");var t=e.aDataSort;"number"!=typeof t||Array.isArray(t)||(e.aDataSort=[t])}function _fnBrowserDetect(e){var t,a,n,r;DataTable.__browser||(DataTable.__browser=t={},r=(n=(a=$("<div/>").css({position:"fixed",top:0,left:-1*window.pageXOffset,height:1,width:1,overflow:"hidden"}).append($("<div/>").css({position:"absolute",top:1,left:1,width:100,overflow:"scroll"}).append($("<div/>").css({width:"100%",height:10}))).appendTo("body")).children()).children(),t.barWidth=n[0].offsetWidth-n[0].clientWidth,t.bScrollbarLeft=1!==Math.round(r.offset().left),a.remove()),$.extend(e.oBrowser,DataTable.__browser),e.oScroll.iBarWidth=DataTable.__browser.barWidth}function _fnAddColumn(e){var t=DataTable.defaults.column,a=e.aoColumns.length,t=$.extend({},DataTable.models.oColumn,t,{aDataSort:t.aDataSort||[a],mData:t.mData||a,idx:a,searchFixed:{},colEl:$("<col>").attr("data-dt-column",a)}),t=(e.aoColumns.push(t),e.aoPreSearchCols);t[a]=$.extend({},DataTable.models.oSearch,t[a])}function _fnColumnOptions(e,t,a){function n(e){return"string"==typeof e&&-1!==e.indexOf("@")}var r=e.aoColumns[t],i=(null!=a&&(_fnCompatCols(a),_fnCamelToHungarian(DataTable.defaults.column,a,!0),void 0===a.mDataProp||a.mData||(a.mData=a.mDataProp),a.sType&&(r._sManualType=a.sType),a.className&&!a.sClass&&(a.sClass=a.className),t=r.sClass,$.extend(r,a),_fnMap(r,a,"sWidth","sWidthOrig"),t!==r.sClass&&(r.sClass=t+" "+r.sClass),void 0!==a.iDataSort&&(r.aDataSort=[a.iDataSort]),_fnMap(r,a,"aDataSort")),r.mData),o=_fnGetObjectDataFn(i);r.mRender&&Array.isArray(r.mRender)&&(a=(t=r.mRender.slice()).shift(),r.mRender=DataTable.render[a].apply(window,t)),r._render=r.mRender?_fnGetObjectDataFn(r.mRender):null;r._bAttrSrc=$.isPlainObject(i)&&(n(i.sort)||n(i.type)||n(i.filter)),r._setter=null,r.fnGetData=function(e,t,a){var n=o(e,t,void 0,a);return r._render&&t?r._render(n,t,e,a):n},r.fnSetData=function(e,t,a){return _fnSetObjectDataFn(i)(e,t,a)},"number"==typeof i||r._isArrayHost||(e._rowReadObject=!0),e.oFeatures.bSort||(r.bSortable=!1)}function _fnAdjustColumnSizing(e){_fnCalculateColumnWidths(e),_fnColumnSizes(e);var t=e.oScroll;""===t.sY&&""===t.sX||_fnScrollDraw(e),_fnCallbackFire(e,null,"column-sizing",[e])}function _fnColumnSizes(e){for(var t=e.aoColumns,a=0;a<t.length;a++){var n=_fnColumnsSumWidth(e,[a],!1,!1);t[a].colEl.css("width",n)}}function _fnVisibleToColumnIndex(e,t){e=_fnGetColumns(e,"bVisible");return"number"==typeof e[t]?e[t]:null}function _fnColumnIndexToVisible(e,t){e=_fnGetColumns(e,"bVisible").indexOf(t);return-1!==e?e:null}function _fnVisbleColumns(e){var t=e.aoHeader,a=e.aoColumns,n=0;if(t.length)for(var r=0,i=t[0].length;r<i;r++)a[r].bVisible&&"none"!==$(t[0][r].cell).css("display")&&n++;return n}function _fnGetColumns(e,a){var n=[];return e.aoColumns.map(function(e,t){e[a]&&n.push(t)}),n}function _fnColumnTypes(e){for(var t,a,n,r,i,o,l,s=e.aoColumns,u=e.aoData,c=DataTable.ext.type.detect,f=0,d=s.length;f<d;f++){if(l=[],!(i=s[f]).sType&&i._sManualType)i.sType=i._sManualType;else if(!i.sType){for(t=0,a=c.length;t<a;t++){for(n=0,r=u.length;n<r;n++)if(u[n]){if(void 0===l[n]&&(l[n]=_fnGetCellData(e,n,f,"type")),!(o=c[t](l[n],e))&&t!==c.length-2)break;if("html"===o&&!_empty(l[n]))break}if(o){i.sType=o;break}}i.sType||(i.sType="string")}var _=_ext.type.className[i.sType],_=(_&&(_columnAutoClass(e.aoHeader,f,_),_columnAutoClass(e.aoFooter,f,_)),_ext.type.render[i.sType]);_&&!i._render&&(i._render=DataTable.util.get(_),_columnAutoRender(e,f))}}function _columnAutoRender(e,t){for(var a,n=e.aoData,r=0;r<n.length;r++)n[r].nTr&&(a=_fnGetCellData(e,r,t,"display"),n[r].displayData[t]=a,_fnWriteCell(n[r].anCells[t],a))}function _columnAutoClass(e,t,a){e.forEach(function(e){e[t]&&e[t].unique&&_addClass(e[t].cell,a)})}function _fnApplyColumnDefs(e,t,a,n,r){var i,o,l,s,u=e.aoColumns;if(a)for(i=0,o=a.length;i<o;i++)a[i]&&a[i].name&&(u[i].sName=a[i].name);if(t)for(i=t.length-1;0<=i;i--)for(var c,f=void 0!==(c=t[i]).target?c.target:void 0!==c.targets?c.targets:c.aTargets,d=0,_=(f=Array.isArray(f)?f:[f]).length;d<_;d++){var p=f[d];if("number"==typeof p&&0<=p){for(;u.length<=p;)_fnAddColumn(e);r(p,c)}else if("number"==typeof p&&p<0)r(u.length+p,c);else if("string"==typeof p)for(l=0,s=u.length;l<s;l++)"_all"===p?r(l,c):-1!==p.indexOf(":name")?u[l].sName===p.replace(":name","")&&r(l,c):n.forEach(function(e){e[l]&&(e=$(e[l].cell),p.match(/^[a-z][\w-]*$/i)&&(p="."+p),e.is(p))&&r(l,c)})}if(a)for(i=0,o=a.length;i<o;i++)r(i,a[i])}function _fnColumnsSumWidth(e,t,a,n){Array.isArray(t)||(t=_fnColumnsFromHeader(t));for(var r,i=0,o=e.aoColumns,l=0,s=t.length;l<s;l++){var u=o[t[l]],c=a?u.sWidthOrig:u.sWidth;if(n||!1!==u.bVisible){if(null==c)return null;"number"==typeof c?(r="px",i+=c):(u=c.match(/([\d\.]+)([^\d]*)/))&&(i+=+u[1],r=3===u.length?u[2]:"px")}}return i+r}function _fnColumnsFromHeader(e){e=$(e).closest("[data-dt-column]").attr("data-dt-column");return e?e.split(",").map(function(e){return+e}):[]}function _fnAddData(e,t,a,n){for(var r=e.aoData.length,i=$.extend(!0,{},DataTable.models.oRow,{src:a?"dom":"data",idx:r}),o=(i._aData=t,e.aoData.push(i),e.aoColumns),l=0,s=o.length;l<s;l++)o[l].sType=null;e.aiDisplayMaster.push(r);t=e.rowIdFn(t);return void 0!==t&&(e.aIds[t]=i),!a&&e.oFeatures.bDeferRender||_fnCreateTr(e,r,a,n),r}function _fnAddTr(a,e){var n;return(e=e instanceof $?e:$(e)).map(function(e,t){return n=_fnGetRowElements(a,t),_fnAddData(a,n.data,t,n.cells)})}function _fnGetCellData(e,t,a,n){"search"===n?n="filter":"order"===n&&(n="sort");var r=e.aoData[t];if(r){var i=e.iDraw,o=e.aoColumns[a],r=r._aData,l=o.sDefaultContent,s=o.fnGetData(r,n,{settings:e,row:t,col:a});if(void 0===(s="display"!==n&&s&&"object"==typeof s&&s.nodeName?s.innerHTML:s))return e.iDrawError!=i&&null===l&&(_fnLog(e,0,"Requested unknown parameter "+("function"==typeof o.mData?"{function}":"'"+o.mData+"'")+" for row "+t+", column "+a,4),e.iDrawError=i),l;if(s!==r&&null!==s||null===l||void 0===n){if("function"==typeof s)return s.call(r)}else s=l;return null===s&&"display"===n?"":s="filter"===n&&(t=DataTable.ext.type.search)[o.sType]?t[o.sType](s):s}}function _fnSetCellData(e,t,a,n){var r=e.aoColumns[a],i=e.aoData[t]._aData;r.fnSetData(i,n,{settings:e,row:t,col:a})}function _fnWriteCell(e,t){t&&"object"==typeof t&&t.nodeName?$(e).empty().append(t):e.innerHTML=t}var __reArray=/\[.*?\]$/,__reFn=/\(\)$/;function _fnSplitObjNotation(e){return(e.match(/(\\.|[^.])+/g)||[""]).map(function(e){return e.replace(/\\\./g,".")})}var _fnGetObjectDataFn=DataTable.util.get,_fnSetObjectDataFn=DataTable.util.set;function _fnGetDataMaster(e){return _pluck(e.aoData,"_aData")}function _fnClearTable(e){e.aoData.length=0,e.aiDisplayMaster.length=0,e.aiDisplay.length=0,e.aIds={}}function _fnInvalidate(e,t,a,n){var r,i,o=e.aoData[t];if(o._aSortData=null,o._aFilterData=null,o.displayData=null,"dom"!==a&&(a&&"auto"!==a||"dom"!==o.src)){var l=o.anCells,s=_fnGetRowDisplay(e,t);if(l)if(void 0!==n)_fnWriteCell(l[n],s[n]);else for(r=0,i=l.length;r<i;r++)_fnWriteCell(l[r],s[r])}else o._aData=_fnGetRowElements(e,o,n,void 0===n?void 0:o._aData).data;var u=e.aoColumns;if(void 0!==n)u[n].sType=null,u[n].maxLenString=null;else{for(r=0,i=u.length;r<i;r++)u[r].sType=null,u[r].maxLenString=null;_fnRowAttributes(e,o)}}function _fnGetRowElements(e,t,a,n){function r(e,t){var a;"string"==typeof e&&-1!==(a=e.indexOf("@"))&&(a=e.substring(a+1),_fnSetObjectDataFn(e)(n,t.getAttribute(a)))}function i(e){void 0!==a&&a!==f||(l=d[f],s=e.innerHTML.trim(),l&&l._bAttrSrc?(_fnSetObjectDataFn(l.mData._)(n,s),r(l.mData.sort,e),r(l.mData.type,e),r(l.mData.filter,e)):_?(l._setter||(l._setter=_fnSetObjectDataFn(l.mData)),l._setter(n,s)):n[f]=s),f++}var o,l,s,u=[],c=t.firstChild,f=0,d=e.aoColumns,_=e._rowReadObject;n=void 0!==n?n:_?{}:[];if(c)for(;c;)"TD"!=(o=c.nodeName.toUpperCase())&&"TH"!=o||(i(c),u.push(c)),c=c.nextSibling;else for(var p=0,h=(u=t.anCells).length;p<h;p++)i(u[p]);var t=t.firstChild?t:t.nTr;return t&&(t=t.getAttribute("id"))&&_fnSetObjectDataFn(e.rowId)(n,t),{data:n,cells:u}}function _fnGetRowDisplay(e,t){var a=e.aoData[t],n=e.aoColumns;if(!a.displayData){a.displayData=[];for(var r=0,i=n.length;r<i;r++)a.displayData.push(_fnGetCellData(e,t,r,"display"))}return a.displayData}function _fnCreateTr(e,t,a,n){var r,i,o,l,s,u,c=e.aoData[t],f=c._aData,d=[],_=e.oClasses.tbody.row;if(null===c.nTr){for(r=a||document.createElement("tr"),c.nTr=r,c.anCells=d,_addClass(r,_),r._DT_RowIndex=t,_fnRowAttributes(e,c),l=0,s=e.aoColumns.length;l<s;l++){o=e.aoColumns[l],(i=(u=!a||!n[l])?document.createElement(o.sCellType):n[l])||_fnLog(e,0,"Incorrect column count",18),i._DT_CellIndex={row:t,column:l},d.push(i);var p=_fnGetRowDisplay(e,t);!u&&(!o.mRender&&o.mData===l||$.isPlainObject(o.mData)&&o.mData._===l+".display")||_fnWriteCell(i,p[l]),o.bVisible&&u?r.appendChild(i):o.bVisible||u||i.parentNode.removeChild(i),o.fnCreatedCell&&o.fnCreatedCell.call(e.oInstance,i,_fnGetCellData(e,t,l),f,t,l)}_fnCallbackFire(e,"aoRowCreatedCallback","row-created",[r,f,t,d])}else _addClass(c.nTr,_)}function _fnRowAttributes(e,t){var a=t.nTr,n=t._aData;a&&((e=e.rowIdFn(n))&&(a.id=e),n.DT_RowClass&&(e=n.DT_RowClass.split(" "),t.__rowc=t.__rowc?_unique(t.__rowc.concat(e)):e,$(a).removeClass(t.__rowc.join(" ")).addClass(n.DT_RowClass)),n.DT_RowAttr&&$(a).attr(n.DT_RowAttr),n.DT_RowData)&&$(a).data(n.DT_RowData)}function _fnBuildHead(e,t){var a,n=e.oClasses,r=e.aoColumns,i="header"===t?e.nTHead:e.nTFoot,o="header"===t?"sTitle":t;if(i){if(("header"===t||_pluck(e.aoColumns,o).join(""))&&1===(a=(a=$("tr",i)).length?a:$("<tr/>").appendTo(i)).length)for(var l=$("td, th",a).length,s=r.length;l<s;l++)$("<th/>").html(r[l][o]||"").appendTo(a);var u=_fnDetectHeader(e,i,!0);"header"===t?e.aoHeader=u:e.aoFooter=u,$(i).children("tr").attr("role","row"),$(i).children("tr").children("th, td").each(function(){_fnRenderer(e,t)(e,$(this),n)})}}function _fnHeaderLayout(e,t,a){var n,r,i,o,l,s=[],u=[],c=e.aoColumns,e=c.length;if(t){for(a=a||_range(e).filter(function(e){return c[e].bVisible}),n=0;n<t.length;n++)s[n]=t[n].slice().filter(function(e,t){return a.includes(t)}),u.push([]);for(n=0;n<s.length;n++)for(r=0;r<s[n].length;r++)if(l=o=1,void 0===u[n][r]){for(i=s[n][r].cell;void 0!==s[n+o]&&s[n][r].cell==s[n+o][r].cell;)u[n+o][r]=null,o++;for(;void 0!==s[n][r+l]&&s[n][r].cell==s[n][r+l].cell;){for(var f=0;f<o;f++)u[n+f][r+l]=null;l++}var d=$("span.dt-column-title",i);u[n][r]={cell:i,colspan:l,rowspan:o,title:(d.length?d:$(i)).html()}}return u}}function _fnDrawHead(e,t){for(var a,n,r=_fnHeaderLayout(e,t),i=0;i<t.length;i++){if(a=t[i].row)for(;n=a.firstChild;)a.removeChild(n);for(var o=0;o<r[i].length;o++){var l=r[i][o];l&&$(l.cell).appendTo(a).attr("rowspan",l.rowspan).attr("colspan",l.colspan)}}}function _fnDraw(e,t){if(_fnStart(e),-1!==_fnCallbackFire(e,"aoPreDrawCallback","preDraw",[e]).indexOf(!1))_fnProcessingDisplay(e,!1);else{var a=[],n=0,r="ssp"==_fnDataSource(e),i=e.aiDisplay,o=e._iDisplayStart,l=e.fnDisplayEnd(),s=e.aoColumns,u=$(e.nTBody);if(e.bDrawing=!0,r){if(!e.bDestroying&&!t)return 0===e.iDraw&&u.empty().append(_emptyRow(e)),void _fnAjaxUpdate(e)}else e.iDraw++;if(0!==i.length)for(var c=r?e.aoData.length:l,f=r?0:o;f<c;f++){for(var d=i[f],_=e.aoData[d],p=(null===_.nTr&&_fnCreateTr(e,d),_.nTr),h=0;h<s.length;h++){var g=s[h],m=_.anCells[h];_addClass(m,_ext.type.className[g.sType]),_addClass(m,g.sClass),_addClass(m,e.oClasses.tbody.cell)}_fnCallbackFire(e,"aoRowCallback",null,[p,_._aData,n,f,d]),a.push(p),n++}else a[0]=_emptyRow(e);_fnCallbackFire(e,"aoHeaderCallback","header",[$(e.nTHead).children("tr")[0],_fnGetDataMaster(e),o,l,i]),_fnCallbackFire(e,"aoFooterCallback","footer",[$(e.nTFoot).children("tr")[0],_fnGetDataMaster(e),o,l,i]),u[0].replaceChildren?u[0].replaceChildren.apply(u[0],a):(u.children().detach(),u.append($(a))),$(e.nTableWrapper).toggleClass("dt-empty-footer",0===$("tr",e.nTFoot).length),_fnCallbackFire(e,"aoDrawCallback","draw",[e],!0),e.bSorted=!1,e.bFiltered=!1,e.bDrawing=!1}}function _fnReDraw(e,t,a){var n=e.oFeatures,r=n.bSort,n=n.bFilter;void 0!==a&&!0!==a||(r&&_fnSort(e),n?_fnFilterComplete(e,e.oPreviousSearch):e.aiDisplay=e.aiDisplayMaster.slice()),!0!==t&&(e._iDisplayStart=0),e._drawHold=t,_fnDraw(e),e._drawHold=!1}function _emptyRow(e){var t=e.oLanguage,a=t.sZeroRecords,n=_fnDataSource(e);return e.iDraw<1&&"ssp"===n||e.iDraw<=1&&"ajax"===n?a=t.sLoadingRecords:t.sEmptyTable&&0===e.fnRecordsTotal()&&(a=t.sEmptyTable),$("<tr/>").append($("<td />",{colSpan:_fnVisbleColumns(e),class:e.oClasses.empty.row}).html(a))[0]}function _layoutArray(e,t,a){for(var o={},n=($.each(t,function(e,t){if(null!==t){function a(t,a){$.isPlainObject(a)?Object.keys(a).map(function(e){t.push({feature:e,opts:a[e]})}):t.push(a)}var e=e.replace(/([A-Z])/g," $1").split(" "),n=(o[e[0]]||(o[e[0]]={}),1===e.length?"full":e[1].toLowerCase()),r=o[e[0]];if(r[n]&&r[n].contents||(r[n]={contents:[]}),Array.isArray(t))for(var i=0;i<t.length;i++)a(r[n].contents,t[i]);else a(r[n].contents,t);Array.isArray(r[n].contents)||(r[n].contents=[r[n].contents])}}),Object.keys(o).map(function(e){return 0!==e.indexOf(a)?null:{name:e,val:o[e]}}).filter(function(e){return null!==e})),r=(n.sort(function(e,t){e=+e.name.replace(/[^0-9]/g,"");return+t.name.replace(/[^0-9]/g,"")-e}),"bottom"===a&&n.reverse(),[]),i=0,l=n.length;i<l;i++)n[i].val.full&&(r.push({full:n[i].val.full}),_layoutResolve(e,r[r.length-1]),delete n[i].val.full),Object.keys(n[i].val).length&&(r.push(n[i].val),_layoutResolve(e,r[r.length-1]));return r}function _layoutResolve(i,o){function l(e,t){return _ext.features[e]||_fnLog(i,0,"Unknown feature: "+e),_ext.features[e].apply(this,[i,t])}$.each(o,function(e){for(var t,a=o[e].contents,n=0,r=a.length;n<r;n++)a[n]&&("string"==typeof a[n]?a[n]=l(a[n],null):$.isPlainObject(a[n])?a[n]=l(a[n].feature,a[n].opts):"function"==typeof a[n].node?a[n]=a[n].node(i):"function"==typeof a[n]&&(t=a[n](i),a[n]="function"==typeof t.node?t.node():t))})}function _fnAddOptionsHtml(t){var a,e=t.oClasses,n=$(t.nTable),r=$("<div/>").attr({id:t.sTableId+"_wrapper",class:e.container}).insertBefore(n);t.nTableWrapper=r[0],t.sDom?_fnLayoutDom(t,t.sDom,r):(e=_layoutArray(t,t.layout,"top"),n=_layoutArray(t,t.layout,"bottom"),a=_fnRenderer(t,"layout"),e.forEach(function(e){a(t,r,e)}),a(t,r,{full:{table:!0,contents:[_fnFeatureHtmlTable(t)]}}),n.forEach(function(e){a(t,r,e)})),_processingHtml(t)}function _fnLayoutDom(t,e,a){for(var n=e.match(/(".*?")|('.*?')|./g),r=0;r<n.length;r++){var i,o,l,s,u,c=null;"<"==(u=n[r])?(l=$("<div/>"),"'"!=(s=n[r+1])[0]&&'"'!=s[0]||(-1!=(i=s.replace(/['"]/g,s="")).indexOf(".")?(s=(o=i.split("."))[0],o=o[1]):"#"==i[0]?s=i:o=i,l.attr("id",s.substring(1)).addClass(o),r++),a.append(l),a=l):">"==u?a=a.parent():"t"==u?c=_fnFeatureHtmlTable(t):DataTable.ext.feature.forEach(function(e){u==e.cFeature&&(c=e.fnInit(t))}),c&&a.append(c)}}function _fnDetectHeader(e,t,a){for(var n,r,i,o,l,s,u=e.aoColumns,c=$(t).children("tr"),f=t&&"thead"===t.nodeName.toLowerCase(),d=[],_=0,p=c.length;_<p;_++)d.push([]);for(_=0,p=c.length;_<p;_++)for(r=(n=c[_]).firstChild;r;){if("TD"==r.nodeName.toUpperCase()||"TH"==r.nodeName.toUpperCase()){var h,g,m,b,v,D=[];for(b=(b=+r.getAttribute("colspan"))&&0!=b&&1!=b?b:1,v=(v=+r.getAttribute("rowspan"))&&0!=v&&1!=v?v:1,l=function(e,t,a){for(var n=e[t];n[a];)a++;return a}(d,_,0),s=1==b,a&&(s&&(_fnColumnOptions(e,l,$(r).data()),h=u[l],g=r.getAttribute("width")||null,(m=r.style.width.match(/width:\s*(\d+[pxem%]+)/))&&(g=m[1]),h.sWidthOrig=h.sWidth||g,f?(null===h.sTitle||h.autoTitle||(r.innerHTML=h.sTitle),!h.sTitle&&s&&(h.sTitle=_stripHtml(r.innerHTML),h.autoTitle=!0)):h.footer&&(r.innerHTML=h.footer),h.ariaTitle||(h.ariaTitle=$(r).attr("aria-label")||h.sTitle),h.className)&&$(r).addClass(h.className),0===$("span.dt-column-title",r).length&&$("<span>").addClass("dt-column-title").append(r.childNodes).appendTo(r),f)&&0===$("span.dt-column-order",r).length&&$("<span>").addClass("dt-column-order").appendTo(r),o=0;o<b;o++){for(i=0;i<v;i++)d[_+i][l+o]={cell:r,unique:s},d[_+i].row=n;D.push(l+o)}r.setAttribute("data-dt-column",_unique(D).join(","))}r=r.nextSibling}return d}function _fnStart(e){var t="ssp"==_fnDataSource(e),a=e.iInitDisplayStart;void 0!==a&&-1!==a&&(e._iDisplayStart=!t&&a>=e.fnRecordsDisplay()?0:a,e.iInitDisplayStart=-1)}function _fnBuildAjax(a,e,n){function t(e){var t=a.jqXHR?a.jqXHR.status:null;(null===e||"number"==typeof t&&204==t)&&_fnAjaxDataSrc(a,e={},[]),(t=e.error||e.sError)&&_fnLog(a,0,t),a.json=e,_fnCallbackFire(a,null,"xhr",[a,e,a.jqXHR],!0),n(e)}var r,i=a.ajax,o=a.oInstance,l=($.isPlainObject(i)&&i.data&&(l="function"==typeof(r=i.data)?r(e,a):r,e="function"==typeof r&&l?l:$.extend(!0,e,l),delete i.data),{url:"string"==typeof i?i:"",data:e,success:t,dataType:"json",cache:!1,type:a.sServerMethod,error:function(e,t){-1===_fnCallbackFire(a,null,"xhr",[a,null,a.jqXHR],!0).indexOf(!0)&&("parsererror"==t?_fnLog(a,0,"Invalid JSON response",1):4===e.readyState&&_fnLog(a,0,"Ajax error",7)),_fnProcessingDisplay(a,!1)}});$.isPlainObject(i)&&$.extend(l,i),a.oAjaxData=e,_fnCallbackFire(a,null,"preXhr",[a,e,l],!0),"function"==typeof i?a.jqXHR=i.call(o,e,t,a):""===i.url?(o={},DataTable.util.set(i.dataSrc)(o,[]),t(o)):(a.jqXHR=$.ajax(l),r&&(i.data=r))}function _fnAjaxUpdate(t){t.iDraw++,_fnProcessingDisplay(t,!0),_fnBuildAjax(t,_fnAjaxParameters(t),function(e){_fnAjaxUpdateDraw(t,e)})}function _fnAjaxParameters(t){function a(e,t){return"function"==typeof n[e][t]?"function":n[e][t]}var n=t.aoColumns,e=t.oFeatures,r=t.oPreviousSearch,i=t.aoPreSearchCols;return{draw:t.iDraw,columns:n.map(function(t,e){return{data:a(e,"mData"),name:t.sName,searchable:t.bSearchable,orderable:t.bSortable,search:{value:i[e].search,regex:i[e].regex,fixed:Object.keys(t.searchFixed).map(function(e){return{name:e,term:t.searchFixed[e].toString()}})}}}),order:_fnSortFlatten(t).map(function(e){return{column:e.col,dir:e.dir,name:a(e.col,"sName")}}),start:t._iDisplayStart,length:e.bPaginate?t._iDisplayLength:-1,search:{value:r.search,regex:r.regex,fixed:Object.keys(t.searchFixed).map(function(e){return{name:e,term:t.searchFixed[e].toString()}})}}}function _fnAjaxUpdateDraw(e,t){var a=_fnAjaxDataSrc(e,t),n=_fnAjaxDataSrcParam(e,"draw",t),r=_fnAjaxDataSrcParam(e,"recordsTotal",t),t=_fnAjaxDataSrcParam(e,"recordsFiltered",t);if(void 0!==n){if(+n<e.iDraw)return;e.iDraw=+n}a=a||[],_fnClearTable(e),e._iRecordsTotal=parseInt(r,10),e._iRecordsDisplay=parseInt(t,10);for(var i=0,o=a.length;i<o;i++)_fnAddData(e,a[i]);e.aiDisplay=e.aiDisplayMaster.slice(),_fnDraw(e,!0),_fnInitComplete(e),_fnProcessingDisplay(e,!1)}function _fnAjaxDataSrc(e,t,a){var n="data";if($.isPlainObject(e.ajax)&&void 0!==e.ajax.dataSrc&&("string"==typeof(e=e.ajax.dataSrc)||"function"==typeof e?n=e:void 0!==e.data&&(n=e.data)),!a)return"data"===n?t.aaData||t[n]:""!==n?_fnGetObjectDataFn(n)(t):t;_fnSetObjectDataFn(n)(t,a)}function _fnAjaxDataSrcParam(e,t,a){var e=$.isPlainObject(e.ajax)?e.ajax.dataSrc:null;return e&&e[t]?_fnGetObjectDataFn(e[t])(a):(e="","draw"===t?e="sEcho":"recordsTotal"===t?e="iTotalRecords":"recordsFiltered"===t&&(e="iTotalDisplayRecords"),void 0!==a[e]?a[e]:a[t])}function _fnFilterComplete(a,e){var t=a.aoPreSearchCols;if(_fnColumnTypes(a),"ssp"!=_fnDataSource(a)){_fnFilterData(a),a.aiDisplay=a.aiDisplayMaster.slice(),_fnFilter(a.aiDisplay,a,e.search,e),$.each(a.searchFixed,function(e,t){_fnFilter(a.aiDisplay,a,t,{})});for(var n=0;n<t.length;n++){var r=t[n];_fnFilter(a.aiDisplay,a,r.search,r,n),$.each(a.aoColumns[n].searchFixed,function(e,t){_fnFilter(a.aiDisplay,a,t,{},n)})}_fnFilterCustom(a)}a.bFiltered=!0,_fnCallbackFire(a,null,"search",[a])}function _fnFilterCustom(e){for(var t,a,n=DataTable.ext.search,r=e.aiDisplay,i=0,o=n.length;i<o;i++){for(var l=[],s=0,u=r.length;s<u;s++)a=r[s],t=e.aoData[a],n[i](e,t._aFilterData,a,t._aData,s)&&l.push(a);r.length=0,r.push.apply(r,l)}}function _fnFilter(e,t,a,n,r){if(""!==a){for(var i=0,o=[],l="function"==typeof a?a:null,s=a instanceof RegExp?a:l?null:_fnFilterCreateSearch(a,n),i=0;i<e.length;i++){var u=t.aoData[e[i]],c=void 0===r?u._sFilterRow:u._aFilterData[r];(l&&l(c,u._aData,e[i],r)||s&&s.test(c))&&o.push(e[i])}for(e.length=o.length,i=0;i<o.length;i++)e[i]=o[i]}}function _fnFilterCreateSearch(e,t){var a,n,r,i=[],t=$.extend({},{boundary:!1,caseInsensitive:!0,exact:!1,regex:!1,smart:!0},t);return"string"!=typeof e&&(e=e.toString()),e=_normalize(e),t.exact?new RegExp("^"+_fnEscapeRegex(e)+"$",t.caseInsensitive?"i":""):(e=t.regex?e:_fnEscapeRegex(e),t.smart&&(a=(e.match(/!?["\u201C][^"\u201D]+["\u201D]|[^ ]+/g)||[""]).map(function(e){var t,a=!1;return"!"===e.charAt(0)&&(a=!0,e=e.substring(1)),'"'===e.charAt(0)?e=(t=e.match(/^"(.*)"$/))?t[1]:e:"“"===e.charAt(0)&&(e=(t=e.match(/^\u201C(.*)\u201D$/))?t[1]:e),a&&(1<e.length&&i.push("(?!"+e+")"),e=""),e.replace(/"/g,"")}),n=i.length?i.join(""):"",e="^(?=.*?"+(r=t.boundary?"\\b":"")+a.join(")(?=.*?"+r)+")("+n+".)*$"),new RegExp(e,t.caseInsensitive?"i":""))}var _fnEscapeRegex=DataTable.util.escapeRegex,__filter_div=$("<div>")[0],__filter_div_textContent=void 0!==__filter_div.textContent;function _fnFilterData(e){for(var t,a,n,r,i,o=e.aoColumns,l=e.aoData,s=!1,u=0;u<l.length;u++)if(l[u]&&!(i=l[u])._aFilterData){for(n=[],t=0,a=o.length;t<a;t++)o[t].bSearchable?"string"!=typeof(r=null===(r=_fnGetCellData(e,u,t,"filter"))?"":r)&&r.toString&&(r=r.toString()):r="",r.indexOf&&-1!==r.indexOf("&")&&(__filter_div.innerHTML=r,r=__filter_div_textContent?__filter_div.textContent:__filter_div.innerText),r.replace&&(r=r.replace(/[\r\n\u2028]/g,"")),n.push(r);i._aFilterData=n,i._sFilterRow=n.join(" "),s=!0}return s}function _fnInitialise(a){var n,e,r=a.iInitDisplayStart;a.bInitialised?(_fnBuildHead(a,"header"),_fnBuildHead(a,"footer"),_fnDrawHead(a,a.aoHeader),_fnDrawHead(a,a.aoFooter),_fnAddOptionsHtml(a),_fnSortInit(a),_colGroup(a),_fnProcessingDisplay(a,!0),_fnCallbackFire(a,null,"preInit",[a],!0),_fnReDraw(a),"ssp"!=(e=_fnDataSource(a))&&("ajax"==e?_fnBuildAjax(a,{},function(e){var t=_fnAjaxDataSrc(a,e);for(n=0;n<t.length;n++)_fnAddData(a,t[n]);a.iInitDisplayStart=r,_fnReDraw(a),_fnProcessingDisplay(a,!1),_fnInitComplete(a)},a):(_fnInitComplete(a),_fnProcessingDisplay(a,!1)))):setTimeout(function(){_fnInitialise(a)},200)}function _fnInitComplete(e){var t;e._bInitComplete||(t=[e,e.json],e._bInitComplete=!0,_fnAdjustColumnSizing(e),_fnCallbackFire(e,null,"plugin-init",t,!0),_fnCallbackFire(e,"aoInitComplete","init",t,!0))}function _fnLengthChange(e,t){t=parseInt(t,10);e._iDisplayLength=t,_fnLengthOverflow(e),_fnCallbackFire(e,null,"length",[e,t])}function _fnPageChange(e,t,a){var n=e._iDisplayStart,r=e._iDisplayLength,i=e.fnRecordsDisplay();if(0===i||-1===r)n=0;else if("number"==typeof t)i<(n=t*r)&&(n=0);else if("first"==t)n=0;else if("previous"==t)(n=0<=r?n-r:0)<0&&(n=0);else if("next"==t)n+r<i&&(n+=r);else if("last"==t)n=Math.floor((i-1)/r)*r;else{if("ellipsis"===t)return;_fnLog(e,0,"Unknown paging action: "+t,5)}i=e._iDisplayStart!==n;return e._iDisplayStart=n,_fnCallbackFire(e,null,i?"page":"page-nc",[e]),i&&a&&_fnDraw(e),i}function _processingHtml(e){var n,t=e.nTable,a=""!==e.oScroll.sX||""!==e.oScroll.sY;e.oFeatures.bProcessing&&(n=$("<div/>",{id:e.sTableId+"_processing",class:e.oClasses.processing.container,role:"status"}).html(e.oLanguage.sProcessing).append("<div><div></div><div></div><div></div><div></div></div>"),a?n.prependTo($("div.dt-scroll",e.nTableWrapper)):n.insertBefore(t),$(t).on("processing.dt.DT",function(e,t,a){n.css("display",a?"block":"none")}))}function _fnProcessingDisplay(e,t){_fnCallbackFire(e,null,"processing",[e,t])}function _fnFeatureHtmlTable(e){var t,a,n,r,i,o,l,s,u,c,f,d,_,p=$(e.nTable),h=e.oScroll;return""===h.sX&&""===h.sY?e.nTable:(t=h.sX,a=h.sY,n=e.oClasses.scrolling,i=(r=e.captionNode)?r._captionSide:null,u=$(p[0].cloneNode(!1)),o=$(p[0].cloneNode(!1)),c=function(e){return e?_fnStringToCss(e):null},(l=p.children("tfoot")).length||(l=null),u=$(s="<div/>",{class:n.container}).append($(s,{class:n.header.self}).css({overflow:"hidden",position:"relative",border:0,width:t?c(t):"100%"}).append($(s,{class:n.header.inner}).css({"box-sizing":"content-box",width:h.sXInner||"100%"}).append(u.removeAttr("id").css("margin-left",0).append("top"===i?r:null).append(p.children("thead"))))).append($(s,{class:n.body}).css({position:"relative",overflow:"auto",width:c(t)}).append(p)),l&&u.append($(s,{class:n.footer.self}).css({overflow:"hidden",border:0,width:t?c(t):"100%"}).append($(s,{class:n.footer.inner}).append(o.removeAttr("id").css("margin-left",0).append("bottom"===i?r:null).append(p.children("tfoot"))))),c=u.children(),f=c[0],d=c[1],_=l?c[2]:null,$(d).on("scroll.DT",function(){var e=this.scrollLeft;f.scrollLeft=e,l&&(_.scrollLeft=e)}),$("th, td",f).on("focus",function(){var e=f.scrollLeft;d.scrollLeft=e,l&&(d.scrollLeft=e)}),$(d).css("max-height",a),h.bCollapse||$(d).css("height",a),e.nScrollHead=f,e.nScrollBody=d,e.nScrollFoot=_,e.aoDrawCallback.push(_fnScrollDraw),u[0])}function _fnScrollDraw(t){var e=t.oScroll.iBarWidth,a=$(t.nScrollHead).children("div"),n=a.children("table"),r=t.nScrollBody,i=$(r),o=$(t.nScrollFoot).children("div"),l=o.children("table"),s=$(t.nTHead),u=$(t.nTable),c=t.nTFoot&&$("th, td",t.nTFoot).length?$(t.nTFoot):null,f=t.oBrowser,d=r.scrollHeight>r.clientHeight;if(t.scrollBarVis!==d&&void 0!==t.scrollBarVis)t.scrollBarVis=d,_fnAdjustColumnSizing(t);else{if(t.scrollBarVis=d,u.children("thead, tfoot").remove(),(d=s.clone().prependTo(u)).find("th, td").removeAttr("tabindex"),d.find("[id]").removeAttr("id"),c&&(g=c.clone().prependTo(u)).find("[id]").removeAttr("id"),t.aiDisplay.length)for(var _=u.children("tbody").eq(0).children("tr").eq(0).children("th, td").map(function(e){return{idx:_fnVisibleToColumnIndex(t,e),width:$(this).outerWidth()}}),p=0;p<_.length;p++){var h=t.aoColumns[_[p].idx].colEl[0];h.style.width.replace("px","")!==_[p].width&&(h.style.width=_[p].width+"px")}n.find("colgroup").remove(),n.append(t.colgroup.clone()),c&&(l.find("colgroup").remove(),l.append(t.colgroup.clone())),$("th, td",d).each(function(){$(this.childNodes).wrapAll('<div class="dt-scroll-sizing">')}),c&&$("th, td",g).each(function(){$(this.childNodes).wrapAll('<div class="dt-scroll-sizing">')});var s=Math.floor(u.height())>r.clientHeight||"scroll"==i.css("overflow-y"),d="padding"+(f.bScrollbarLeft?"Left":"Right"),g=u.outerWidth();n.css("width",_fnStringToCss(g)),a.css("width",_fnStringToCss(g)).css(d,s?e+"px":"0px"),c&&(l.css("width",_fnStringToCss(g)),o.css("width",_fnStringToCss(g)).css(d,s?e+"px":"0px")),u.children("colgroup").prependTo(u),i.trigger("scroll"),!t.bSorted&&!t.bFiltered||t._drawHold||(r.scrollTop=0)}}function _fnCalculateColumnWidths(t){if(t.oFeatures.bAutoWidth){var e=t.nTable,a=t.aoColumns,n=t.oScroll,r=n.sY,i=n.sX,n=n.sXInner,o=_fnGetColumns(t,"bVisible"),l=e.getAttribute("width"),s=e.parentNode,u=e.style.width,u=(u&&-1!==u.indexOf("%")&&(l=u),_fnCallbackFire(t,null,"column-calc",{visible:o},!1),$(e.cloneNode()).css("visibility","hidden").removeAttr("id")),c=(u.append("<tbody>"),$("<tr/>").appendTo(u.find("tbody")));for(u.append($(t.nTHead).clone()).append($(t.nTFoot).clone()),u.find("tfoot th, tfoot td").css("width",""),u.find("thead th, thead td").each(function(){var e=_fnColumnsSumWidth(t,this,!0,!1);e?(this.style.width=e,i&&$(this).append($("<div/>").css({width:e,margin:0,padding:0,border:0,height:1}))):this.style.width=""}),b=0;b<o.length;b++){var f=a[d=o[b]],d=_fnGetMaxLenString(t,d),_=_ext.type.className[f.sType],p=d+f.sContentPadding,d=-1===d.indexOf("<")?document.createTextNode(p):p;$("<td/>").addClass(_).addClass(f.sClass).append(d).appendTo(c)}$("[name]",u).removeAttr("name");for(var h=$("<div/>").css(i||r?{position:"absolute",top:0,left:0,height:1,right:0,overflow:"hidden"}:{}).append(u).appendTo(s),g=(i&&n?u.width(n):i?(u.css("width","auto"),u.removeAttr("width"),u.width()<s.clientWidth&&l&&u.width(s.clientWidth)):r?u.width(s.clientWidth):l&&u.width(l),0),m=u.find("tbody tr").eq(0).children(),b=0;b<o.length;b++){var v=m[b].getBoundingClientRect().width;g+=v,a[o[b]].sWidth=_fnStringToCss(v)}e.style.width=_fnStringToCss(g),h.remove(),l&&(e.style.width=_fnStringToCss(l)),!l&&!i||t._reszEvt||($(window).on("resize.DT-"+t.sInstance,DataTable.util.throttle(function(){t.bDestroying||_fnAdjustColumnSizing(t)})),t._reszEvt=!0)}}function _fnGetMaxLenString(e,t){var a=e.aoColumns[t];if(!a.maxLenString){for(var n="",r=-1,i=0,o=e.aiDisplayMaster.length;i<o;i++){var l,s=_fnGetRowDisplay(e,e.aiDisplayMaster[i])[t],s=(s=s&&"object"==typeof s&&s.nodeType?s.innerHTML:s+"").replace(/id=".*?"/g,"").replace(/name=".*?"/g,"");(l=_stripHtml(s).replace(/ /g," ")).length>r&&(n=s,r=l.length)}a.maxLenString=n}return a.maxLenString}function _fnStringToCss(e){return null===e?"0px":"number"==typeof e?e<0?"0px":e+"px":e.match(/\d$/)?e+"px":e}function _colGroup(e){var t=e.aoColumns;for(e.colgroup.empty(),i=0;i<t.length;i++)t[i].bVisible&&e.colgroup.append(t[i].colEl)}function _fnSortInit(e){var t=e.nTHead,a=t.querySelectorAll("tr"),n=e.bSortCellsTop,r=':not([data-dt-order="disable"]):not([data-dt-order="icon-only"])',n=(!0===n?t=a[0]:!1===n&&(t=a[a.length-1]),_fnSortAttachListener(e,t,t===e.nTHead?"tr"+r+" th"+r+", tr"+r+" td"+r:"th"+r+", td"+r),[]);_fnSortResolve(e,n,e.aaSorting),e.aaSorting=n}function _fnSortAttachListener(i,e,t,o,l){_fnBindAction(e,t,function(e){var t=!1,a=void 0===o?_fnColumnsFromHeader(e.target):[o];if(a.length){for(var n=0,r=a.length;n<r;n++)if(!1!==_fnSortAdd(i,a[n],n,e.shiftKey)&&(t=!0),1===i.aaSorting.length&&""===i.aaSorting[0][1])break;t&&(_fnProcessingDisplay(i,!0),setTimeout(function(){_fnSort(i),_fnSortDisplay(i,i.aiDisplay),_fnProcessingDisplay(i,!1),_fnReDraw(i,!1,!1),l&&l()},0))}})}function _fnSortDisplay(e,t){if(!(t.length<2)){for(var a=e.aiDisplayMaster,n={},r={},i=0;i<a.length;i++)n[a[i]]=i;for(i=0;i<t.length;i++)r[t[i]]=n[t[i]];t.sort(function(e,t){return r[e]-r[t]})}}function _fnSortResolve(a,n,e){function t(e){var t;$.isPlainObject(e)?void 0!==e.idx?n.push([e.idx,e.dir]):e.name&&-1!==(t=_pluck(a.aoColumns,"sName").indexOf(e.name))&&n.push([t,e.dir]):n.push(e)}if($.isPlainObject(e))t(e);else if(e.length&&"number"==typeof e[0])t(e);else if(e.length)for(var r=0;r<e.length;r++)t(e[r])}function _fnSortFlatten(e){var t,a,n,r,i,o,l,s=[],u=DataTable.ext.type.order,c=e.aoColumns,f=e.aaSortingFixed,d=$.isPlainObject(f),_=[];if(e.oFeatures.bSort)for(Array.isArray(f)&&_fnSortResolve(e,_,f),d&&f.pre&&_fnSortResolve(e,_,f.pre),_fnSortResolve(e,_,e.aaSorting),d&&f.post&&_fnSortResolve(e,_,f.post),t=0;t<_.length;t++)if(c[l=_[t][0]])for(a=0,n=(r=c[l].aDataSort).length;a<n;a++)o=c[i=r[a]].sType||"string",void 0===_[t]._idx&&(_[t]._idx=c[i].asSorting.indexOf(_[t][1])),_[t][1]&&s.push({src:l,col:i,dir:_[t][1],index:_[t]._idx,type:o,formatter:u[o+"-pre"],sorter:u[o+"-"+_[t][1]]});return s}function _fnSort(e,t,a){var n,r,i,o,c,f=[],l=DataTable.ext.type.order,d=e.aoData,s=e.aiDisplayMaster;for(_fnColumnTypes(e),void 0!==t?(o=e.aoColumns[t],c=[{src:t,col:t,dir:a,index:0,type:o.sType,formatter:l[o.sType+"-pre"],sorter:l[o.sType+"-"+a]}],s=s.slice()):c=_fnSortFlatten(e),n=0,r=c.length;n<r;n++)_fnSortData(e,c[n].col);if("ssp"!=_fnDataSource(e)&&0!==c.length){for(n=0,i=s.length;n<i;n++)f[n]=n;c.length&&"desc"===c[0].dir&&f.reverse(),s.sort(function(e,t){for(var a,n,r,i,o=c.length,l=d[e]._aSortData,s=d[t]._aSortData,u=0;u<o;u++)if(a=l[(i=c[u]).col],n=s[i.col],i.sorter){if(0!==(r=i.sorter(a,n)))return r}else if(0!==(r=a<n?-1:n<a?1:0))return"asc"===i.dir?r:-r;return(a=f[e])<(n=f[t])?-1:n<a?1:0})}else 0===c.length&&s.sort(function(e,t){return e<t?-1:t<e?1:0});return void 0===t&&(e.bSorted=!0,_fnCallbackFire(e,null,"order",[e,c])),s}function _fnSortAdd(e,t,a,n){function r(e,t){var a=e._idx;return(a=void 0===a?s.indexOf(e[1]):a)+1<s.length?a+1:t?null:0}var i,o=e.aoColumns[t],l=e.aaSorting,s=o.asSorting;if(!o.bSortable)return!1;"number"==typeof l[0]&&(l=e.aaSorting=[l]),(n||a)&&e.oFeatures.bSortMulti?-1!==(o=_pluck(l,"0").indexOf(t))?null===(i=null===(i=r(l[o],!0))&&1===l.length?0:i)?l.splice(o,1):(l[o][1]=s[i],l[o]._idx=i):(n?l.push([t,s[0],0]):l.push([t,l[0][1],0]),l[l.length-1]._idx=0):l.length&&l[0][0]==t?(i=r(l[0]),l.length=1,l[0][1]=s[i],l[0]._idx=i):(l.length=0,l.push([t,s[0]]),l[0]._idx=0)}function _fnSortingClasses(e){var t,a,n,r=e.aLastSort,i=e.oClasses.order.position,o=_fnSortFlatten(e),l=e.oFeatures;if(l.bSort&&l.bSortClasses){for(t=0,a=r.length;t<a;t++)n=r[t].src,$(_pluck(e.aoData,"anCells",n)).removeClass(i+(t<2?t+1:3));for(t=0,a=o.length;t<a;t++)n=o[t].src,$(_pluck(e.aoData,"anCells",n)).addClass(i+(t<2?t+1:3))}e.aLastSort=o}function _fnSortData(e,t){for(var a,n,r,i=e.aoColumns[t],o=DataTable.ext.order[i.sSortDataType],l=(o&&(a=o.call(e.oInstance,e,t,_fnColumnIndexToVisible(e,t))),DataTable.ext.type.order[i.sType+"-pre"]),s=e.aoData,u=0;u<s.length;u++)s[u]&&((n=s[u])._aSortData||(n._aSortData=[]),n._aSortData[t]&&!o||(r=o?a[u]:_fnGetCellData(e,u,t,"sort"),n._aSortData[t]=l?l(r,e):r))}function _fnSaveState(a){var e;a._bLoadingState||(e={time:+new Date,start:a._iDisplayStart,length:a._iDisplayLength,order:$.extend(!0,[],a.aaSorting),search:$.extend({},a.oPreviousSearch),columns:a.aoColumns.map(function(e,t){return{visible:e.bVisible,search:$.extend({},a.aoPreSearchCols[t])}})},a.oSavedState=e,_fnCallbackFire(a,"aoStateSaveParams","stateSaveParams",[a,e]),a.oFeatures.bStateSave&&!a.bDestroying&&a.fnStateSaveCallback.call(a.oInstance,a,e))}function _fnLoadState(t,e,a){var n;if(t.oFeatures.bStateSave)return void 0!==(n=t.fnStateLoadCallback.call(t.oInstance,t,function(e){_fnImplementState(t,e,a)}))&&_fnImplementState(t,n,a),!0;a()}function _fnImplementState(a,e,t){var n,r,i=a.aoColumns,o=(a._bLoadingState=!0,a._bInitComplete?new DataTable.Api(a):null);if(e&&e.time){var l=a.iStateDuration;if(0<l&&e.time<+new Date-1e3*l)a._bLoadingState=!1;else if(-1!==_fnCallbackFire(a,"aoStateLoadParams","stateLoadParams",[a,e]).indexOf(!1))a._bLoadingState=!1;else if(e.columns&&i.length!==e.columns.length)a._bLoadingState=!1;else{if(a.oLoadedState=$.extend(!0,{},e),_fnCallbackFire(a,null,"stateLoadInit",[a,e],!0),void 0!==e.length&&(o?o.page.len(e.length):a._iDisplayLength=e.length),void 0!==e.start&&(null===o?(a._iDisplayStart=e.start,a.iInitDisplayStart=e.start):_fnPageChange(a,e.start/a._iDisplayLength)),void 0!==e.order&&(a.aaSorting=[],$.each(e.order,function(e,t){a.aaSorting.push(t[0]>=i.length?[0,t[1]]:t)})),void 0!==e.search&&$.extend(a.oPreviousSearch,e.search),e.columns){for(n=0,r=e.columns.length;n<r;n++){var s=e.columns[n];void 0!==s.visible&&(o?o.column(n).visible(s.visible,!1):i[n].bVisible=s.visible),void 0!==s.search&&$.extend(a.aoPreSearchCols[n],s.search)}o&&o.columns.adjust()}a._bLoadingState=!1,_fnCallbackFire(a,"aoStateLoaded","stateLoaded",[a,e])}}else a._bLoadingState=!1;t()}function _fnLog(e,t,a,n){if(a="DataTables warning: "+(e?"table id="+e.sTableId+" - ":"")+a,n&&(a+=". For more information about this error, please see https://datatables.net/tn/"+n),t)window.console&&console.log&&console.log(a);else{t=DataTable.ext,t=t.sErrMode||t.errMode;if(e&&_fnCallbackFire(e,null,"dt-error",[e,n,a],!0),"alert"==t)alert(a);else{if("throw"==t)throw new Error(a);"function"==typeof t&&t(e,n,a)}}}function _fnMap(a,n,e,t){Array.isArray(e)?$.each(e,function(e,t){Array.isArray(t)?_fnMap(a,n,t[0],t[1]):_fnMap(a,n,t)}):(void 0===t&&(t=e),void 0!==n[e]&&(a[t]=n[e]))}function _fnExtend(e,t,a){var n,r;for(r in t)Object.prototype.hasOwnProperty.call(t,r)&&(n=t[r],$.isPlainObject(n)?($.isPlainObject(e[r])||(e[r]={}),$.extend(!0,e[r],n)):a&&"data"!==r&&"aaData"!==r&&Array.isArray(n)?e[r]=n.slice():e[r]=n);return e}function _fnBindAction(e,t,a){$(e).on("click.DT",t,function(e){a(e)}).on("keypress.DT",t,function(e){13===e.which&&(e.preventDefault(),a(e))}).on("selectstart.DT",t,function(){return!1})}function _fnCallbackReg(e,t,a){a&&e[t].push(a)}function _fnCallbackFire(t,e,a,n,r){var i=[];return e&&(i=t[e].slice().reverse().map(function(e){return e.apply(t.oInstance,n)})),null!==a&&(e=$.Event(a+".dt"),a=$(t.nTable),e.dt=t.api,a[r?"trigger":"triggerHandler"](e,n),r&&0===a.parents("body").length&&$("body").trigger(e,n),i.push(e.result)),i}function _fnLengthOverflow(e){var t=e._iDisplayStart,a=e.fnDisplayEnd(),n=e._iDisplayLength;a<=t&&(t=a-n),t-=t%n,e._iDisplayStart=t=-1===n||t<0?0:t}function _fnRenderer(e,t){var e=e.renderer,a=DataTable.ext.renderer[t];return $.isPlainObject(e)&&e[t]?a[e[t]]||a._:"string"==typeof e&&a[e]||a._}function _fnDataSource(e){return e.oFeatures.bServerSide?"ssp":e.ajax?"ajax":"dom"}function _fnMacros(e,t,a){var n=e.fnFormatNumber,r=e._iDisplayStart+1,i=e._iDisplayLength,o=e.fnRecordsDisplay(),l=e.fnRecordsTotal(),s=-1===i;return t.replace(/_START_/g,n.call(e,r)).replace(/_END_/g,n.call(e,e.fnDisplayEnd())).replace(/_MAX_/g,n.call(e,l)).replace(/_TOTAL_/g,n.call(e,o)).replace(/_PAGE_/g,n.call(e,s?1:Math.ceil(r/i))).replace(/_PAGES_/g,n.call(e,s?1:Math.ceil(o/i))).replace(/_ENTRIES_/g,e.api.i18n("entries","",a)).replace(/_ENTRIES-MAX_/g,e.api.i18n("entries","",l)).replace(/_ENTRIES-TOTAL_/g,e.api.i18n("entries","",o))}var __apiStruct=[],__arrayProto=Array.prototype,_toSettings=function(e){var t,a,n=DataTable.settings,r=_pluck(n,"nTable");return e?e.nTable&&e.oFeatures?[e]:e.nodeName&&"table"===e.nodeName.toLowerCase()?-1!==(t=r.indexOf(e))?[n[t]]:null:e&&"function"==typeof e.settings?e.settings().toArray():("string"==typeof e?a=$(e).get():e instanceof $&&(a=e.get()),a?n.filter(function(e,t){return a.includes(r[t])}):void 0):[]};function _api_scope(t,a,n){return function(){var e=a.apply(t||this,arguments);return _Api.extend(e,e,n.methodExt),e}}function _api_find(e,t){for(var a=0,n=e.length;a<n;a++)if(e[a].name===t)return e[a];return null}_Api=function(e,t){if(!(this instanceof _Api))return new _Api(e,t);function a(e){(e=_toSettings(e))&&n.push.apply(n,e)}var n=[];if(Array.isArray(e))for(var r=0,i=e.length;r<i;r++)a(e[r]);else a(e);this.context=1<n.length?_unique(n):n,t&&this.push.apply(this,t),this.selector={rows:null,cols:null,opts:null},_Api.extend(this,this,__apiStruct)},DataTable.Api=_Api,$.extend(_Api.prototype,{any:function(){return 0!==this.count()},context:[],count:function(){return this.flatten().length},each:function(e){for(var t=0,a=this.length;t<a;t++)e.call(this,this[t],t,this);return this},eq:function(e){var t=this.context;return t.length>e?new _Api(t[e],this[e]):null},filter:function(e){e=__arrayProto.filter.call(this,e,this);return new _Api(this.context,e)},flatten:function(){var e=[];return new _Api(this.context,e.concat.apply(e,this.toArray()))},get:function(e){return this[e]},join:__arrayProto.join,includes:function(e){return-1!==this.indexOf(e)},indexOf:__arrayProto.indexOf,iterator:function(e,t,a,n){var r,i,o,l,s,u,c,f,d=[],_=this.context,p=this.selector;for("string"==typeof e&&(n=a,a=t,t=e,e=!1),i=0,o=_.length;i<o;i++){var h=new _Api(_[i]);if("table"===t)void 0!==(r=a.call(h,_[i],i))&&d.push(r);else if("columns"===t||"rows"===t)void 0!==(r=a.call(h,_[i],this[i],i))&&d.push(r);else if("every"===t||"column"===t||"column-rows"===t||"row"===t||"cell"===t)for(c=this[i],"column-rows"===t&&(u=_selector_row_indexes(_[i],p.opts)),l=0,s=c.length;l<s;l++)f=c[l],void 0!==(r="cell"===t?a.call(h,_[i],f.row,f.column,i,l):a.call(h,_[i],f,i,l,u))&&d.push(r)}return d.length||n?((e=(n=new _Api(_,e?d.concat.apply([],d):d)).selector).rows=p.rows,e.cols=p.cols,e.opts=p.opts,n):this},lastIndexOf:__arrayProto.lastIndexOf,length:0,map:function(e){e=__arrayProto.map.call(this,e,this);return new _Api(this.context,e)},pluck:function(e){var t=DataTable.util.get(e);return this.map(function(e){return t(e)})},pop:__arrayProto.pop,push:__arrayProto.push,reduce:__arrayProto.reduce,reduceRight:__arrayProto.reduceRight,reverse:__arrayProto.reverse,selector:null,shift:__arrayProto.shift,slice:function(){return new _Api(this.context,this)},sort:__arrayProto.sort,splice:__arrayProto.splice,toArray:function(){return __arrayProto.slice.call(this)},to$:function(){return $(this)},toJQuery:function(){return $(this)},unique:function(){return new _Api(this.context,_unique(this.toArray()))},unshift:__arrayProto.unshift}),window.__apiStruct=__apiStruct,_Api.extend=function(e,t,a){if(a.length&&t&&(t instanceof _Api||t.__dt_wrapper))for(var n,r=0,i=a.length;r<i;r++)"__proto__"!==(n=a[r]).name&&(t[n.name]="function"===n.type?_api_scope(e,n.val,n):"object"===n.type?{}:n.val,t[n.name].__dt_wrapper=!0,_Api.extend(e,t[n.name],n.propExt))},_Api.register=_api_register=function(e,t){if(Array.isArray(e))for(var a=0,n=e.length;a<n;a++)_Api.register(e[a],t);else for(var r=e.split("."),i=__apiStruct,o=0,l=r.length;o<l;o++){var s,u,c=_api_find(i,u=(s=-1!==r[o].indexOf("()"))?r[o].replace("()",""):r[o]);c||i.push(c={name:u,val:{},methodExt:[],propExt:[],type:"object"}),o===l-1?(c.val=t,c.type="function"==typeof t?"function":$.isPlainObject(t)?"object":"other"):i=s?c.methodExt:c.propExt}},_Api.registerPlural=_api_registerPlural=function(e,t,a){_Api.register(e,a),_Api.register(t,function(){var e=a.apply(this,arguments);return e===this?this:e instanceof _Api?e.length?Array.isArray(e[0])?new _Api(e.context,e[0]):e[0]:void 0:e})};var __table_selector=function(e,t){var a,n;return Array.isArray(e)?(a=[],e.forEach(function(e){e=__table_selector(e,t);a.push.apply(a,e)}),a.filter(function(e){return e})):"number"==typeof e?[t[e]]:(n=t.map(function(e){return e.nTable}),$(n).filter(e).map(function(){var e=n.indexOf(this);return t[e]}).toArray())},__reload=(_api_register("tables()",function(e){return null!=e?new _Api(__table_selector(e,this.context)):this}),_api_register("table()",function(e){var e=this.tables(e),t=e.context;return t.length?new _Api(t[0]):e}),[["nodes","node","nTable"],["body","body","nTBody"],["header","header","nTHead"],["footer","footer","nTFoot"]].forEach(function(t){_api_registerPlural("tables()."+t[0]+"()","table()."+t[1]+"()",function(){return this.iterator("table",function(e){return e[t[2]]},1)})}),[["header","aoHeader"],["footer","aoFooter"]].forEach(function(a){_api_register("table()."+a[0]+".structure()",function(e){var e=this.columns(e).indexes().flatten(),t=this.context[0];return _fnHeaderLayout(t,t[a[1]],e)})}),_api_registerPlural("tables().containers()","table().container()",function(){return this.iterator("table",function(e){return e.nTableWrapper},1)}),_api_register("tables().every()",function(a){var n=this;return this.iterator("table",function(e,t){a.call(n.table(t),t)})}),_api_register("caption()",function(r,i){var e,t=this.context;return void 0===r?(e=t[0].captionNode)&&t.length?e.innerHTML:null:this.iterator("table",function(e){var t=$(e.nTable),a=$(e.captionNode),n=$(e.nTableWrapper);a.length||(a=$("<caption/>").html(r),e.captionNode=a[0],i)||(t.prepend(a),i=a.css("caption-side")),a.html(r),i&&(a.css("caption-side",i),a[0]._captionSide=i),(n.find("div.dataTables_scroll").length?(e="top"===i?"Head":"Foot",n.find("div.dataTables_scroll"+e+" table")):t).prepend(a)},1)}),_api_register("caption.node()",function(){var e=this.context;return e.length?e[0].captionNode:null}),_api_register("draw()",function(t){return this.iterator("table",function(e){"page"===t?_fnDraw(e):_fnReDraw(e,!1===(t="string"==typeof t?"full-hold"!==t:t))})}),_api_register("page()",function(t){return void 0===t?this.page.info().page:this.iterator("table",function(e){_fnPageChange(e,t)})}),_api_register("page.info()",function(){var e,t,a,n,r;if(0!==this.context.length)return t=(e=this.context[0])._iDisplayStart,a=e.oFeatures.bPaginate?e._iDisplayLength:-1,n=e.fnRecordsDisplay(),{page:(r=-1===a)?0:Math.floor(t/a),pages:r?1:Math.ceil(n/a),start:t,end:e.fnDisplayEnd(),length:a,recordsTotal:e.fnRecordsTotal(),recordsDisplay:n,serverSide:"ssp"===_fnDataSource(e)}}),_api_register("page.len()",function(t){return void 0===t?0!==this.context.length?this.context[0]._iDisplayLength:void 0:this.iterator("table",function(e){_fnLengthChange(e,t)})}),function(r,i,e){var t,a;e&&(t=new _Api(r)).one("draw",function(){e(t.ajax.json())}),"ssp"==_fnDataSource(r)?_fnReDraw(r,i):(_fnProcessingDisplay(r,!0),(a=r.jqXHR)&&4!==a.readyState&&a.abort(),_fnBuildAjax(r,{},function(e){_fnClearTable(r);for(var t=_fnAjaxDataSrc(r,e),a=0,n=t.length;a<n;a++)_fnAddData(r,t[a]);_fnReDraw(r,i),_fnInitComplete(r),_fnProcessingDisplay(r,!1)}))}),_selector_run=(_api_register("ajax.json()",function(){var e=this.context;if(0<e.length)return e[0].json}),_api_register("ajax.params()",function(){var e=this.context;if(0<e.length)return e[0].oAjaxData}),_api_register("ajax.reload()",function(t,a){return this.iterator("table",function(e){__reload(e,!1===a,t)})}),_api_register("ajax.url()",function(t){var e=this.context;return void 0===t?0===e.length?void 0:(e=e[0],$.isPlainObject(e.ajax)?e.ajax.url:e.ajax):this.iterator("table",function(e){$.isPlainObject(e.ajax)?e.ajax.url=t:e.ajax=t})}),_api_register("ajax.url().load()",function(t,a){return this.iterator("table",function(e){__reload(e,!1===a,t)})}),function(e,t,a,n,r){for(var i,o,l,s,u=[],c=typeof t,f=0,d=(t=t&&"string"!=c&&"function"!=c&&void 0!==t.length?t:[t]).length;f<d;f++)for(l=0,s=(o=t[f]&&t[f].split&&!t[f].match(/[[(:]/)?t[f].split(","):[t[f]]).length;l<s;l++)(i=(i=a("string"==typeof o[l]?o[l].trim():o[l])).filter(function(e){return null!=e}))&&i.length&&(u=u.concat(i));var _=_ext.selector[e];if(_.length)for(f=0,d=_.length;f<d;f++)u=_[f](n,r,u);return _unique(u)}),_selector_opts=function(e){return(e=e||{}).filter&&void 0===e.search&&(e.search=e.filter),$.extend({search:"none",order:"current",page:"all"},e)},_selector_first=function(e){var t=new _Api(e.context[0]);return e.length&&t.push(e[0]),t.selector=e.selector,t.length&&1<t[0].length&&t[0].splice(1),t},_selector_row_indexes=function(e,t){var a,n=[],r=e.aiDisplay,i=e.aiDisplayMaster,o=t.search,l=t.order,t=t.page;if("ssp"==_fnDataSource(e))return"removed"===o?[]:_range(0,i.length);if("current"==t)for(u=e._iDisplayStart,c=e.fnDisplayEnd();u<c;u++)n.push(r[u]);else if("current"==l||"applied"==l){if("none"==o)n=i.slice();else if("applied"==o)n=r.slice();else if("removed"==o){for(var s={},u=0,c=r.length;u<c;u++)s[r[u]]=null;i.forEach(function(e){Object.prototype.hasOwnProperty.call(s,e)||n.push(e)})}}else if("index"==l||"original"==l)for(u=0,c=e.aoData.length;u<c;u++)e.aoData[u]&&("none"==o||-1===(a=r.indexOf(u))&&"removed"==o||0<=a&&"applied"==o)&&n.push(u);else if("number"==typeof l){var f=_fnSort(e,l,"asc");if("none"===o)n=f;else for(u=0;u<f.length;u++)(-1===(a=r.indexOf(f[u]))&&"removed"==o||0<=a&&"applied"==o)&&n.push(f[u])}return n},__row_selector=function(r,e,i){var o,e=_selector_run("row",e,function(a){var e=_intVal(a),n=r.aoData;if(null!==e&&!i)return[e];if(o=o||_selector_row_indexes(r,i),null!==e&&-1!==o.indexOf(e))return[e];if(null==a||""===a)return o;if("function"==typeof a)return o.map(function(e){var t=n[e];return a(e,t._aData,t.nTr)?e:null});if(a.nodeName)return e=a._DT_RowIndex,t=a._DT_CellIndex,void 0!==e?n[e]&&n[e].nTr===a?[e]:[]:t?n[t.row]&&n[t.row].nTr===a.parentNode?[t.row]:[]:(e=$(a).closest("*[data-dt-row]")).length?[e.data("dt-row")]:[];if("string"==typeof a&&"#"===a.charAt(0)){var t=r.aIds[a.replace(/^#/,"")];if(void 0!==t)return[t.idx]}e=_removeEmpty(_pluck_order(r.aoData,o,"nTr"));return $(e).filter(a).map(function(){return this._DT_RowIndex}).toArray()},r,i);return"current"!==i.order&&"applied"!==i.order||_fnSortDisplay(r,e),e},__details_state_load=(_api_register("rows()",function(t,a){void 0===t?t="":$.isPlainObject(t)&&(a=t,t=""),a=_selector_opts(a);var e=this.iterator("table",function(e){return __row_selector(e,t,a)},1);return e.selector.rows=t,e.selector.opts=a,e}),_api_register("rows().nodes()",function(){return this.iterator("row",function(e,t){return e.aoData[t].nTr||void 0},1)}),_api_register("rows().data()",function(){return this.iterator(!0,"rows",function(e,t){return _pluck_order(e.aoData,t,"_aData")},1)}),_api_registerPlural("rows().cache()","row().cache()",function(a){return this.iterator("row",function(e,t){e=e.aoData[t];return"search"===a?e._aFilterData:e._aSortData},1)}),_api_registerPlural("rows().invalidate()","row().invalidate()",function(a){return this.iterator("row",function(e,t){_fnInvalidate(e,t,a)})}),_api_registerPlural("rows().indexes()","row().index()",function(){return this.iterator("row",function(e,t){return t},1)}),_api_registerPlural("rows().ids()","row().id()",function(e){for(var t=[],a=this.context,n=0,r=a.length;n<r;n++)for(var i=0,o=this[n].length;i<o;i++){var l=a[n].rowIdFn(a[n].aoData[this[n][i]]._aData);t.push((!0===e?"#":"")+l)}return new _Api(a,t)}),_api_registerPlural("rows().remove()","row().remove()",function(){return this.iterator("row",function(e,t){var a=e.aoData,n=a[t],r=e.aiDisplayMaster.indexOf(t),r=(-1!==r&&e.aiDisplayMaster.splice(r,1),0<e._iRecordsDisplay&&e._iRecordsDisplay--,_fnLengthOverflow(e),e.rowIdFn(n._aData));void 0!==r&&delete e.aIds[r],a[t]=null}),this}),_api_register("rows.add()",function(i){var e=this.iterator("table",function(e){for(var t,a=[],n=0,r=i.length;n<r;n++)(t=i[n]).nodeName&&"TR"===t.nodeName.toUpperCase()?a.push(_fnAddTr(e,t)[0]):a.push(_fnAddData(e,t));return a},1),t=this.rows(-1);return t.pop(),t.push.apply(t,e),t}),_api_register("row()",function(e,t){return _selector_first(this.rows(e,t))}),_api_register("row().data()",function(e){var t,a=this.context;return void 0===e?a.length&&this.length&&this[0].length?a[0].aoData[this[0]]._aData:void 0:((t=a[0].aoData[this[0]])._aData=e,Array.isArray(e)&&t.nTr&&t.nTr.id&&_fnSetObjectDataFn(a[0].rowId)(e,t.nTr.id),_fnInvalidate(a[0],this[0],"data"),this)}),_api_register("row().node()",function(){var e=this.context;if(e.length&&this.length&&this[0].length){e=e[0].aoData[this[0]];if(e&&e.nTr)return e.nTr}return null}),_api_register("row.add()",function(t){t instanceof $&&t.length&&(t=t[0]);var e=this.iterator("table",function(e){return t.nodeName&&"TR"===t.nodeName.toUpperCase()?_fnAddTr(e,t)[0]:_fnAddData(e,t)});return this.row(e[0])}),$(document).on("plugin-init.dt",function(e,t){var n=new _Api(t);n.on("stateSaveParams.DT",function(e,t,a){for(var n=t.rowIdFn,r=t.aiDisplayMaster,i=[],o=0;o<r.length;o++){var l=r[o],l=t.aoData[l];l._detailsShow&&i.push("#"+n(l._aData))}a.childRows=i}),n.on("stateLoaded.DT",function(e,t,a){__details_state_load(n,a)}),__details_state_load(n,n.state.loaded())}),function(e,t){t&&t.childRows&&e.rows(t.childRows.map(function(e){return e.replace(/([^:\\]*(?:\\.[^:\\]*)*):/g,"$1\\:")})).every(function(){_fnCallbackFire(e.settings()[0],null,"requestChild",[this])})}),__details_add=function(i,o,e,t){function l(e,t){var a;if(Array.isArray(e)||e instanceof $)for(var n=0,r=e.length;n<r;n++)l(e[n],t);else e.nodeName&&"tr"===e.nodeName.toLowerCase()?(e.setAttribute("data-dt-row",o.idx),s.push(e)):(a=$("<tr><td></td></tr>").attr("data-dt-row",o.idx).addClass(t),$("td",a).addClass(t).html(e)[0].colSpan=_fnVisbleColumns(i),s.push(a[0]))}var s=[];l(e,t),o._details&&o._details.detach(),o._details=$(s),o._detailsShow&&o._details.insertAfter(o.nTr)},__details_state=DataTable.util.throttle(function(e){_fnSaveState(e[0])},500),__details_remove=function(e,t){var a=e.context;a.length&&(t=a[0].aoData[void 0!==t?t:e[0]])&&t._details&&(t._details.remove(),t._detailsShow=void 0,t._details=void 0,$(t.nTr).removeClass("dt-hasChild"),__details_state(a))},__details_display=function(e,t){var a,n=e.context;n.length&&e.length&&(a=n[0].aoData[e[0]])._details&&((a._detailsShow=t)?(a._details.insertAfter(a.nTr),$(a.nTr).addClass("dt-hasChild")):(a._details.detach(),$(a.nTr).removeClass("dt-hasChild")),_fnCallbackFire(n[0],null,"childRow",[t,e.row(e[0])]),__details_events(n[0]),__details_state(n))},__details_events=function(o){var r=new _Api(o),e=".dt.DT_details",t="draw"+e,a="column-sizing"+e,e="destroy"+e,l=o.aoData;r.off(t+" "+a+" "+e),0<_pluck(l,"_details").length&&(r.on(t,function(e,t){o===t&&r.rows({page:"current"}).eq(0).each(function(e){e=l[e];e._detailsShow&&e._details.insertAfter(e.nTr)})}),r.on(a,function(e,t){if(o===t)for(var a,n=_fnVisbleColumns(t),r=0,i=l.length;r<i;r++)(a=l[r])&&a._details&&a._details.each(function(){var e=$(this).children("td");1==e.length&&e.attr("colspan",n)})}),r.on(e,function(e,t){if(o===t)for(var a=0,n=l.length;a<n;a++)l[a]&&l[a]._details&&__details_remove(r,a)}))},_emp="",_child_obj=_emp+"row().child",_child_mth=_child_obj+"()",__re_column_selector=(_api_register(_child_mth,function(e,t){var a=this.context;return void 0===e?a.length&&this.length&&a[0].aoData[this[0]]?a[0].aoData[this[0]]._details:void 0:(!0===e?this.child.show():!1===e?__details_remove(this):a.length&&this.length&&__details_add(a[0],a[0].aoData[this[0]],e,t),this)}),_api_register([_child_obj+".show()",_child_mth+".show()"],function(){return __details_display(this,!0),this}),_api_register([_child_obj+".hide()",_child_mth+".hide()"],function(){return __details_display(this,!1),this}),_api_register([_child_obj+".remove()",_child_mth+".remove()"],function(){return __details_remove(this),this}),_api_register(_child_obj+".isShown()",function(){var e=this.context;return e.length&&this.length&&e[0].aoData[this[0]]&&e[0].aoData[this[0]]._detailsShow||!1}),/^([^:]+)?:(name|title|visIdx|visible)$/),__columnData=function(e,t,a,n,r,i){for(var o=[],l=0,s=r.length;l<s;l++)o.push(_fnGetCellData(e,r[l],t,i));return o},__column_header=function(e,t,a){var n=e.aoHeader;return n[void 0!==a?a:e.bSortCellsTop?0:n.length-1][t].cell},__column_selector=function(o,e,l){var s=o.aoColumns,u=_pluck(s,"sName"),c=_pluck(s,"sTitle"),t=DataTable.util.get("[].[].cell")(o.aoHeader),f=_unique(_flatten([],t));return _selector_run("column",e,function(a){var n,e=_intVal(a);if(""===a)return _range(s.length);if(null!==e)return[0<=e?e:s.length+e];if("function"==typeof a)return n=_selector_row_indexes(o,l),s.map(function(e,t){return a(t,__columnData(o,t,0,0,n),__column_header(o,t))?t:null});var t,r,i="string"==typeof a?a.match(__re_column_selector):"";if(i)switch(i[2]){case"visIdx":case"visible":return i[1]?(t=parseInt(i[1],10))<0?[(r=s.map(function(e,t){return e.bVisible?t:null}))[r.length+t]]:[_fnVisibleToColumnIndex(o,t)]:s.map(function(e,t){return e.bVisible?t:null});case"name":return u.map(function(e,t){return e===i[1]?t:null});case"title":return c.map(function(e,t){return e===i[1]?t:null});default:return[]}return a.nodeName&&a._DT_CellIndex?[a._DT_CellIndex.column]:(e=$(f).filter(a).map(function(){return _fnColumnsFromHeader(this)}).toArray()).length||!a.nodeName?e:(e=$(a).closest("*[data-dt-column]")).length?[e.data("dt-column")]:[]},o,l)},__setColumnVis=function(e,t,a){var n,r,i=e.aoColumns,o=i[t],l=e.aoData;if(void 0===a)return o.bVisible;if(o.bVisible===a)return!1;if(a)for(var s=_pluck(i,"bVisible").indexOf(!0,t+1),u=0,c=l.length;u<c;u++)l[u]&&(r=l[u].nTr,n=l[u].anCells,r)&&r.insertBefore(n[t],n[s]||null);else $(_pluck(e.aoData,"anCells",t)).detach();return o.bVisible=a,_colGroup(e),!0},__cell_selector=(_api_register("columns()",function(t,a){void 0===t?t="":$.isPlainObject(t)&&(a=t,t=""),a=_selector_opts(a);var e=this.iterator("table",function(e){return __column_selector(e,t,a)},1);return e.selector.cols=t,e.selector.opts=a,e}),_api_registerPlural("columns().header()","column().header()",function(a){return this.iterator("column",function(e,t){return __column_header(e,t,a)},1)}),_api_registerPlural("columns().footer()","column().footer()",function(a){return this.iterator("column",function(e,t){return e.aoFooter.length?e.aoFooter[void 0!==a?a:0][t].cell:null},1)}),_api_registerPlural("columns().data()","column().data()",function(){return this.iterator("column-rows",__columnData,1)}),_api_registerPlural("columns().render()","column().render()",function(i){return this.iterator("column-rows",function(e,t,a,n,r){return __columnData(e,t,a,n,r,i)},1)}),_api_registerPlural("columns().dataSrc()","column().dataSrc()",function(){return this.iterator("column",function(e,t){return e.aoColumns[t].mData},1)}),_api_registerPlural("columns().cache()","column().cache()",function(i){return this.iterator("column-rows",function(e,t,a,n,r){return _pluck_order(e.aoData,r,"search"===i?"_aFilterData":"_aSortData",t)},1)}),_api_registerPlural("columns().init()","column().init()",function(){return this.iterator("column",function(e,t){return e.aoColumns[t]},1)}),_api_registerPlural("columns().nodes()","column().nodes()",function(){return this.iterator("column-rows",function(e,t,a,n,r){return _pluck_order(e.aoData,r,"anCells",t)},1)}),_api_registerPlural("columns().titles()","column().title()",function(a,n){return this.iterator("column",function(e,t){"number"==typeof a&&(n=a,a=void 0);t=$("span.dt-column-title",this.column(t).header(n));return void 0!==a?(t.html(a),this):t.html()},1)}),_api_registerPlural("columns().types()","column().type()",function(){return this.iterator("column",function(e,t){t=e.aoColumns[t].sType;return t||_fnColumnTypes(e),t},1)}),_api_registerPlural("columns().visible()","column().visible()",function(a,n){var t=this,r=[],e=this.iterator("column",function(e,t){if(void 0===a)return e.aoColumns[t].bVisible;__setColumnVis(e,t,a)&&r.push(t)});return void 0!==a&&this.iterator("table",function(e){_fnDrawHead(e,e.aoHeader),_fnDrawHead(e,e.aoFooter),e.aiDisplay.length||$(e.nTBody).find("td[colspan]").attr("colspan",_fnVisbleColumns(e)),_fnSaveState(e),t.iterator("column",function(e,t){r.includes(t)&&_fnCallbackFire(e,null,"column-visibility",[e,t,a,n])}),r.length&&(void 0===n||n)&&t.columns.adjust()}),e}),_api_registerPlural("columns().widths()","column().width()",function(){var e=this.columns(":visible").count(),e=$("<tr>").html("<td>"+Array(e).join("</td><td>")+"</td>"),a=($(this.table().body()).append(e),e.children().map(function(){return $(this).outerWidth()}));return e.remove(),this.iterator("column",function(e,t){e=_fnColumnIndexToVisible(e,t);return null!==e?a[e]:0},1)}),_api_registerPlural("columns().indexes()","column().index()",function(a){return this.iterator("column",function(e,t){return"visible"===a?_fnColumnIndexToVisible(e,t):t},1)}),_api_register("columns.adjust()",function(){return this.iterator("table",function(e){_fnAdjustColumnSizing(e)},1)}),_api_register("column.index()",function(e,t){var a;if(0!==this.context.length)return a=this.context[0],"fromVisible"===e||"toData"===e?_fnVisibleToColumnIndex(a,t):"fromData"===e||"toVisible"===e?_fnColumnIndexToVisible(a,t):void 0}),_api_register("column()",function(e,t){return _selector_first(this.columns(e,t))}),function(n,e,t){var r,i,o,l,s,u,c,f=n.aoData,d=_selector_row_indexes(n,t),a=_removeEmpty(_pluck_order(f,d,"anCells")),_=$(_flatten([],a)),p=n.aoColumns.length;return _selector_run("cell",e,function(e){var t,a="function"==typeof e;if(null==e||a){for(i=[],o=0,l=d.length;o<l;o++)for(r=d[o],s=0;s<p;s++)u={row:r,column:s},(!a||(c=f[r],e(u,_fnGetCellData(n,r,s),c.anCells?c.anCells[s]:null)))&&i.push(u);return i}return $.isPlainObject(e)?void 0!==e.column&&void 0!==e.row&&-1!==d.indexOf(e.row)?[e]:[]:(t=_.filter(e).map(function(e,t){return{row:t._DT_CellIndex.row,column:t._DT_CellIndex.column}}).toArray()).length||!e.nodeName?t:(c=$(e).closest("*[data-dt-row]")).length?[{row:c.data("dt-row"),column:c.data("dt-column")}]:[]},n,t)}),extPagination=(_api_register("cells()",function(t,e,a){var n,r,i,o,l,s,u;return $.isPlainObject(t)&&(void 0===t.row?(a=t,t=null):(a=e,e=null)),$.isPlainObject(e)&&(a=e,e=null),null==e?this.iterator("table",function(e){return __cell_selector(e,t,_selector_opts(a))}):(u=a?{page:a.page,order:a.order,search:a.search}:{},n=this.columns(e,u),r=this.rows(t,u),u=this.iterator("table",function(e,t){var a=[];for(i=0,o=r[t].length;i<o;i++)for(l=0,s=n[t].length;l<s;l++)a.push({row:r[t][i],column:n[t][l]});return a},1),u=a&&a.selected?this.cells(u,a):u,$.extend(u.selector,{cols:e,rows:t,opts:a}),u)}),_api_registerPlural("cells().nodes()","cell().node()",function(){return this.iterator("cell",function(e,t,a){e=e.aoData[t];return e&&e.anCells?e.anCells[a]:void 0},1)}),_api_register("cells().data()",function(){return this.iterator("cell",function(e,t,a){return _fnGetCellData(e,t,a)},1)}),_api_registerPlural("cells().cache()","cell().cache()",function(n){return n="search"===n?"_aFilterData":"_aSortData",this.iterator("cell",function(e,t,a){return e.aoData[t][n][a]},1)}),_api_registerPlural("cells().render()","cell().render()",function(n){return this.iterator("cell",function(e,t,a){return _fnGetCellData(e,t,a,n)},1)}),_api_registerPlural("cells().indexes()","cell().index()",function(){return this.iterator("cell",function(e,t,a){return{row:t,column:a,columnVisible:_fnColumnIndexToVisible(e,a)}},1)}),_api_registerPlural("cells().invalidate()","cell().invalidate()",function(n){return this.iterator("cell",function(e,t,a){_fnInvalidate(e,t,n,a)})}),_api_register("cell()",function(e,t,a){return _selector_first(this.cells(e,t,a))}),_api_register("cell().data()",function(e){var t=this.context,a=this[0];return void 0===e?t.length&&a.length?_fnGetCellData(t[0],a[0].row,a[0].column):void 0:(_fnSetCellData(t[0],a[0].row,a[0].column,e),_fnInvalidate(t[0],a[0].row,"data",a[0].column),this)}),_api_register("order()",function(t,e){var a=this.context,n=Array.prototype.slice.call(arguments);return void 0===t?0!==a.length?a[0].aaSorting:void 0:("number"==typeof t?t=[[t,e]]:1<n.length&&(t=n),this.iterator("table",function(e){e.aaSorting=Array.isArray(t)?t.slice():t}))}),_api_register("order.listener()",function(t,a,n){return this.iterator("table",function(e){_fnSortAttachListener(e,t,{},a,n)})}),_api_register("order.fixed()",function(t){var e;return t?this.iterator("table",function(e){e.aaSortingFixed=$.extend(!0,{},t)}):(e=(e=this.context).length?e[0].aaSortingFixed:void 0,Array.isArray(e)?{pre:e}:e)}),_api_register(["columns().order()","column().order()"],function(a){var n=this;return a?this.iterator("table",function(e,t){e.aaSorting=n[t].map(function(e){return[e,a]})}):this.iterator("column",function(e,t){for(var a=_fnSortFlatten(e),n=0,r=a.length;n<r;n++)if(a[n].col===t)return a[n].dir;return null},1)}),_api_registerPlural("columns().orderable()","column().orderable()",function(a){return this.iterator("column",function(e,t){e=e.aoColumns[t];return a?e.asSorting:e.bSortable},1)}),_api_register("processing()",function(t){return this.iterator("table",function(e){_fnProcessingDisplay(e,t)})}),_api_register("search()",function(t,a,n,r){var e=this.context;return void 0===t?0!==e.length?e[0].oPreviousSearch.search:void 0:this.iterator("table",function(e){e.oFeatures.bFilter&&_fnFilterComplete(e,"object"==typeof a?$.extend(e.oPreviousSearch,a,{search:t}):$.extend(e.oPreviousSearch,{search:t,regex:null!==a&&a,smart:null===n||n,caseInsensitive:null===r||r}))})}),_api_register("search.fixed()",function(t,a){var e=this.iterator(!0,"table",function(e){e=e.searchFixed;return t?void 0===a?e[t]:(null===a?delete e[t]:e[t]=a,this):Object.keys(e)});return void 0!==t&&void 0===a?e[0]:e}),_api_registerPlural("columns().search()","column().search()",function(n,r,i,o){return this.iterator("column",function(e,t){var a=e.aoPreSearchCols;if(void 0===n)return a[t].search;e.oFeatures.bFilter&&("object"==typeof r?$.extend(a[t],r,{search:n}):$.extend(a[t],{search:n,regex:null!==r&&r,smart:null===i||i,caseInsensitive:null===o||o}),_fnFilterComplete(e,e.oPreviousSearch))})}),_api_register(["columns().search.fixed()","column().search.fixed()"],function(a,n){var e=this.iterator(!0,"column",function(e,t){e=e.aoColumns[t].searchFixed;return a?void 0===n?e[a]:(null===n?delete e[a]:e[a]=n,this):Object.keys(e)});return void 0!==a&&void 0===n?e[0]:e}),_api_register("state()",function(e,t){var a;return e?(a=$.extend(!0,{},e),this.iterator("table",function(e){!1!==t&&(a.time=+new Date+100),_fnImplementState(e,a,function(){})})):this.context.length?this.context[0].oSavedState:null}),_api_register("state.clear()",function(){return this.iterator("table",function(e){e.fnStateSaveCallback.call(e.oInstance,e,{})})}),_api_register("state.loaded()",function(){return this.context.length?this.context[0].oLoadedState:null}),_api_register("state.save()",function(){return this.iterator("table",function(e){_fnSaveState(e)})}),DataTable.use=function(e,t){"lib"===t||e.fn?$=e:"win"==t||e.document?(window=e,document=e.document):"datetime"!==t&&"DateTime"!==e.type||(DataTable.DateTime=e)},DataTable.factory=function(e,t){var a=!1;return e&&e.document&&(window=e,document=e.document),t&&t.fn&&t.fn.jquery&&($=t,a=!0),a},DataTable.versionCheck=function(e,t){for(var a,n,r=(t||DataTable.version).split("."),i=e.split("."),o=0,l=i.length;o<l;o++)if((a=parseInt(r[o],10)||0)!==(n=parseInt(i[o],10)||0))return n<a;return!0},DataTable.isDataTable=function(e){var r=$(e).get(0),i=!1;return e instanceof DataTable.Api||($.each(DataTable.settings,function(e,t){var a=t.nScrollHead?$("table",t.nScrollHead)[0]:null,n=t.nScrollFoot?$("table",t.nScrollFoot)[0]:null;t.nTable!==r&&a!==r&&n!==r||(i=!0)}),i)},DataTable.tables=function(t){var e=!1,a=($.isPlainObject(t)&&(e=t.api,t=t.visible),DataTable.settings.filter(function(e){return!(t&&!$(e.nTable).is(":visible"))}).map(function(e){return e.nTable}));return e?new _Api(a):a},DataTable.camelToHungarian=_fnCamelToHungarian,_api_register("$()",function(e,t){t=this.rows(t).nodes(),t=$(t);return $([].concat(t.filter(e).toArray(),t.find(e).toArray()))}),$.each(["on","one","off"],function(e,a){_api_register(a+"()",function(){var e=Array.prototype.slice.call(arguments),t=(e[0]=e[0].split(/\s/).map(function(e){return e.match(/\.dt\b/)?e:e+".dt"}).join(" "),$(this.tables().nodes()));return t[a].apply(t,e),this})}),_api_register("clear()",function(){return this.iterator("table",function(e){_fnClearTable(e)})}),_api_register("error()",function(t){return this.iterator("table",function(e){_fnLog(e,0,t)})}),_api_register("settings()",function(){return new _Api(this.context,this.context)}),_api_register("init()",function(){var e=this.context;return e.length?e[0].oInit:null}),_api_register("data()",function(){return this.iterator("table",function(e){return _pluck(e.aoData,"_aData")}).flatten()}),_api_register("trigger()",function(t,a,n){return this.iterator("table",function(e){return _fnCallbackFire(e,null,t,a,n)}).flatten()}),_api_register("ready()",function(e){var t=this.context;return e?this.tables().every(function(){this.context[0]._bInitComplete?e.call(this):this.on("init",function(){e.call(this)})}):t.length?t[0]._bInitComplete||!1:null}),_api_register("destroy()",function(c){return c=c||!1,this.iterator("table",function(e){var t=e.oClasses,a=e.nTable,n=e.nTBody,r=e.nTHead,i=e.nTFoot,o=$(a),n=$(n),l=$(e.nTableWrapper),s=e.aoData.map(function(e){return e?e.nTr:null}),u=t.order,i=(e.bDestroying=!0,_fnCallbackFire(e,"aoDestroyCallback","destroy",[e],!0),c||new _Api(e).columns().visible(!0),l.off(".DT").find(":not(tbody *)").off(".DT"),$(window).off(".DT-"+e.sInstance),a!=r.parentNode&&(o.children("thead").detach(),o.append(r)),i&&a!=i.parentNode&&(o.children("tfoot").detach(),o.append(i)),e.colgroup.remove(),e.aaSorting=[],e.aaSortingFixed=[],_fnSortingClasses(e),$("th, td",r).removeClass(u.canAsc+" "+u.canDesc+" "+u.isAsc+" "+u.isDesc).css("width",""),n.children().detach(),n.append(s),e.nTableWrapper.parentNode),r=e.nTableWrapper.nextSibling,u=c?"remove":"detach",n=(o[u](),l[u](),!c&&i&&(i.insertBefore(a,r),o.css("width",e.sDestroyWidth).removeClass(t.table)),DataTable.settings.indexOf(e));-1!==n&&DataTable.settings.splice(n,1)})}),$.each(["column","row","cell"],function(e,s){_api_register(s+"s().every()",function(n){var r,i=this.selector.opts,o=this,l=0;return this.iterator("every",function(e,t,a){r=o[s](t,i),"cell"===s?n.call(r,r[0][0].row,r[0][0].column,a,l):n.call(r,t,a,l),l++})})}),_api_register("i18n()",function(e,t,a){var n=this.context[0],e=_fnGetObjectDataFn(e)(n.oLanguage);return"string"==typeof(e=$.isPlainObject(e=void 0===e?t:e)?void 0!==a&&void 0!==e[a]?e[a]:e._:e)?e.replace("%d",a):e}),DataTable.version="2.0.8",DataTable.settings=[],DataTable.models={},DataTable.models.oSearch={caseInsensitive:!0,search:"",regex:!1,smart:!0,return:!1},DataTable.models.oRow={nTr:null,anCells:null,_aData:[],_aSortData:null,_aFilterData:null,_sFilterRow:null,src:null,idx:-1,displayData:null},DataTable.models.oColumn={idx:null,aDataSort:null,asSorting:null,bSearchable:null,bSortable:null,bVisible:null,_sManualType:null,_bAttrSrc:!1,fnCreatedCell:null,fnGetData:null,fnSetData:null,mData:null,mRender:null,sClass:null,sContentPadding:null,sDefaultContent:null,sName:null,sSortDataType:"std",sSortingClass:null,sTitle:null,sType:null,sWidth:null,sWidthOrig:null,maxLenString:null,searchFixed:null},DataTable.defaults={aaData:null,aaSorting:[[0,"asc"]],aaSortingFixed:[],ajax:null,aLengthMenu:[10,25,50,100],aoColumns:null,aoColumnDefs:null,aoSearchCols:[],bAutoWidth:!0,bDeferRender:!0,bDestroy:!1,bFilter:!0,bInfo:!0,bLengthChange:!0,bPaginate:!0,bProcessing:!1,bRetrieve:!1,bScrollCollapse:!1,bServerSide:!1,bSort:!0,bSortMulti:!0,bSortCellsTop:null,bSortClasses:!0,bStateSave:!1,fnCreatedRow:null,fnDrawCallback:null,fnFooterCallback:null,fnFormatNumber:function(e){return e.toString().replace(/\B(?=(\d{3})+(?!\d))/g,this.oLanguage.sThousands)},fnHeaderCallback:null,fnInfoCallback:null,fnInitComplete:null,fnPreDrawCallback:null,fnRowCallback:null,fnStateLoadCallback:function(e){try{return JSON.parse((-1===e.iStateDuration?sessionStorage:localStorage).getItem("DataTables_"+e.sInstance+"_"+location.pathname))}catch(e){return{}}},fnStateLoadParams:null,fnStateLoaded:null,fnStateSaveCallback:function(e,t){try{(-1===e.iStateDuration?sessionStorage:localStorage).setItem("DataTables_"+e.sInstance+"_"+location.pathname,JSON.stringify(t))}catch(e){}},fnStateSaveParams:null,iStateDuration:7200,iDisplayLength:10,iDisplayStart:0,iTabIndex:0,oClasses:{},oLanguage:{oAria:{orderable:": Activate to sort",orderableReverse:": Activate to invert sorting",orderableRemove:": Activate to remove sorting",paginate:{first:"First",last:"Last",next:"Next",previous:"Previous"}},oPaginate:{sFirst:"«",sLast:"»",sNext:"›",sPrevious:"‹"},entries:{_:"entries",1:"entry"},sEmptyTable:"No data available in table",sInfo:"Showing _START_ to _END_ of _TOTAL_ _ENTRIES-TOTAL_",sInfoEmpty:"Showing 0 to 0 of 0 _ENTRIES-TOTAL_",sInfoFiltered:"(filtered from _MAX_ total _ENTRIES-MAX_)",sInfoPostFix:"",sDecimal:"",sThousands:",",sLengthMenu:"_MENU_ _ENTRIES_ per page",sLoadingRecords:"Loading...",sProcessing:"",sSearch:"Search:",sSearchPlaceholder:"",sUrl:"",sZeroRecords:"No matching records found"},oSearch:$.extend({},DataTable.models.oSearch),layout:{topStart:"pageLength",topEnd:"search",bottomStart:"info",bottomEnd:"paging"},sDom:null,searchDelay:null,sPaginationType:"full_numbers",sScrollX:"",sScrollXInner:"",sScrollY:"",sServerMethod:"GET",renderer:null,rowId:"DT_RowId",caption:null},_fnHungarianMap(DataTable.defaults),DataTable.defaults.column={aDataSort:null,iDataSort:-1,ariaTitle:"",asSorting:["asc","desc",""],bSearchable:!0,bSortable:!0,bVisible:!0,fnCreatedCell:null,mData:null,mRender:null,sCellType:"td",sClass:"",sContentPadding:"",sDefaultContent:null,sName:"",sSortDataType:"std",sTitle:null,sType:null,sWidth:null},_fnHungarianMap(DataTable.defaults.column),DataTable.models.oSettings={oFeatures:{bAutoWidth:null,bDeferRender:null,bFilter:null,bInfo:!0,bLengthChange:!0,bPaginate:null,bProcessing:null,bServerSide:null,bSort:null,bSortMulti:null,bSortClasses:null,bStateSave:null},oScroll:{bCollapse:null,iBarWidth:0,sX:null,sXInner:null,sY:null},oLanguage:{fnInfoCallback:null},oBrowser:{bScrollbarLeft:!1,barWidth:0},ajax:null,aanFeatures:[],aoData:[],aiDisplay:[],aiDisplayMaster:[],aIds:{},aoColumns:[],aoHeader:[],aoFooter:[],oPreviousSearch:{},searchFixed:{},aoPreSearchCols:[],aaSorting:null,aaSortingFixed:[],sDestroyWidth:0,aoRowCallback:[],aoHeaderCallback:[],aoFooterCallback:[],aoDrawCallback:[],aoRowCreatedCallback:[],aoPreDrawCallback:[],aoInitComplete:[],aoStateSaveParams:[],aoStateLoadParams:[],aoStateLoaded:[],sTableId:"",nTable:null,nTHead:null,nTFoot:null,nTBody:null,nTableWrapper:null,bInitialised:!1,aoOpenRows:[],sDom:null,searchDelay:null,sPaginationType:"two_button",pagingControls:0,iStateDuration:0,aoStateSave:[],aoStateLoad:[],oSavedState:null,oLoadedState:null,bAjaxDataGet:!0,jqXHR:null,json:void 0,oAjaxData:void 0,sServerMethod:null,fnFormatNumber:null,aLengthMenu:null,iDraw:0,bDrawing:!1,iDrawError:-1,_iDisplayLength:10,_iDisplayStart:0,_iRecordsTotal:0,_iRecordsDisplay:0,oClasses:{},bFiltered:!1,bSorted:!1,bSortCellsTop:null,oInit:null,aoDestroyCallback:[],fnRecordsTotal:function(){return"ssp"==_fnDataSource(this)?+this._iRecordsTotal:this.aiDisplayMaster.length},fnRecordsDisplay:function(){return"ssp"==_fnDataSource(this)?+this._iRecordsDisplay:this.aiDisplay.length},fnDisplayEnd:function(){var e=this._iDisplayLength,t=this._iDisplayStart,a=t+e,n=this.aiDisplay.length,r=this.oFeatures,i=r.bPaginate;return r.bServerSide?!1===i||-1===e?t+n:Math.min(t+e,this._iRecordsDisplay):!i||n<a||-1===e?n:a},oInstance:null,sInstance:null,iTabIndex:0,nScrollHead:null,nScrollFoot:null,aLastSort:[],oPlugins:{},rowIdFn:null,rowId:null,caption:"",captionNode:null,colgroup:null},DataTable.ext.pager),_filterString=($.extend(extPagination,{simple:function(){return["previous","next"]},full:function(){return["first","previous","next","last"]},numbers:function(){return["numbers"]},simple_numbers:function(){return["previous","numbers","next"]},full_numbers:function(){return["first","previous","numbers","next","last"]},first_last:function(){return["first","last"]},first_last_numbers:function(){return["first","numbers","last"]},_numbers:_pagingNumbers,numbers_length:7}),$.extend(!0,DataTable.ext.renderer,{pagingButton:{_:function(e,t,a,n,r){var e=e.oClasses.paging,i=[e.button];return n&&i.push(e.active),r&&i.push(e.disabled),{display:n="ellipsis"===t?$('<span class="ellipsis"></span>').html(a)[0]:$("<button>",{class:i.join(" "),role:"link",type:"button"}).html(a),clicker:n}}},pagingContainer:{_:function(e,t){return t}}}),function(t,a){return function(e){return _empty(e)||"string"!=typeof e||(e=e.replace(_re_new_lines," "),t&&(e=_stripHtml(e)),a&&(e=_normalize(e,!1))),e}});function __mldFnName(e){return e.replace(/[\W]/g,"_")}function __mld(e,t,a,n,r){return window.moment?e[t](r):window.luxon?e[a](r):n?e[n](r):e}var __mlWarning=!1;function __mldObj(e,t,a){var n;if(window.moment){if(!(n=window.moment.utc(e,t,a,!0)).isValid())return null}else if(window.luxon){if(!(n=t&&"string"==typeof e?window.luxon.DateTime.fromFormat(e,t):window.luxon.DateTime.fromISO(e)).isValid)return null;n.setLocale(a)}else t?(__mlWarning||alert("DataTables warning: Formatted date without Moment.js or Luxon - https://datatables.net/tn/17"),__mlWarning=!0):n=new Date(e);return n}function __mlHelper(s){return function(n,r,i,o){0===arguments.length?(i="en",n=r=null):1===arguments.length?(i="en",r=n,n=null):2===arguments.length&&(i=r,r=n,n=null);var l="datetime"+(r?"-"+__mldFnName(r):"");return DataTable.ext.type.order[l]||DataTable.type(l,{detect:function(e){return e===l&&l},order:{pre:function(e){return e.valueOf()}},className:"dt-right"}),function(e,t){var a;return null==e&&(e="--now"===o?(a=new Date,new Date(Date.UTC(a.getFullYear(),a.getMonth(),a.getDate(),a.getHours(),a.getMinutes(),a.getSeconds()))):""),"type"===t?l:""===e?"sort"!==t?"":__mldObj("0000-01-01 00:00:00",null,i):!(null===r||n!==r||"sort"===t||"type"===t||e instanceof Date)||null===(a=__mldObj(e,n,i))?e:"sort"===t?a:(e=null===r?__mld(a,"toDate","toJSDate","")[s]():__mld(a,"format","toFormat","toISOString",r),"display"===t?_escapeHtml(e):e)}}}var __thousands=",",__decimal=".";if(void 0!==window.Intl)try{for(var num=(new Intl.NumberFormat).formatToParts(100000.1),i=0;i<num.length;i++)"group"===num[i].type?__thousands=num[i].value:"decimal"===num[i].type&&(__decimal=num[i].value)}catch(e){}DataTable.datetime=function(a,n){var r="datetime-detect-"+__mldFnName(a);n=n||"en",DataTable.ext.type.order[r]||DataTable.type(r,{detect:function(e){var t=__mldObj(e,a,n);return!(""!==e&&!t)&&r},order:{pre:function(e){return __mldObj(e,a,n)||0}},className:"dt-right"})},DataTable.render={date:__mlHelper("toLocaleDateString"),datetime:__mlHelper("toLocaleString"),time:__mlHelper("toLocaleTimeString"),number:function(r,i,o,l,s){return null==r&&(r=__thousands),null==i&&(i=__decimal),{display:function(e){if("number"!=typeof e&&"string"!=typeof e)return e;if(""===e||null===e)return e;var t=e<0?"-":"",a=parseFloat(e),n=Math.abs(a);if(1e11<=n||n<1e-4&&0!==n)return(n=a.toExponential(o).split(/e\+?/))[0]+" x 10<sup>"+n[1]+"</sup>";if(isNaN(a))return _escapeHtml(e);a=a.toFixed(o),e=Math.abs(a);n=parseInt(e,10),a=o?i+(e-n).toFixed(o).substring(2):"";return(t=0===n&&0===parseFloat(a)?"":t)+(l||"")+n.toString().replace(/\B(?=(\d{3})+(?!\d))/g,r)+a+(s||"")}}},text:function(){return{display:_escapeHtml,filter:_escapeHtml}}};var _extTypes=DataTable.ext.type,__numericReplace=(DataTable.type=function(n,e,t){if(!e)return{className:_extTypes.className[n],detect:_extTypes.detect.find(function(e){return e.name===n}),order:{pre:_extTypes.order[n+"-pre"],asc:_extTypes.order[n+"-asc"],desc:_extTypes.order[n+"-desc"]},render:_extTypes.render[n],search:_extTypes.search[n]};function a(e,t){_extTypes[e][n]=t}function r(a){function e(e,t){return!0===(e=a(e,t))?n:e}Object.defineProperty(e,"name",{value:n});var t=_extTypes.detect.findIndex(function(e){return e.name===n});-1===t?_extTypes.detect.unshift(e):_extTypes.detect.splice(t,1,e)}function i(e){_extTypes.order[n+"-pre"]=e.pre,_extTypes.order[n+"-asc"]=e.asc,_extTypes.order[n+"-desc"]=e.desc}void 0===t&&(t=e,e=null),"className"===e?a("className",t):"detect"===e?r(t):"order"===e?i(t):"render"===e?a("render",t):"search"===e?a("search",t):e||(t.className&&a("className",t.className),void 0!==t.detect&&r(t.detect),t.order&&i(t.order),void 0!==t.render&&a("render",t.render),void 0!==t.search&&a("search",t.search))},DataTable.types=function(){return _extTypes.detect.map(function(e){return e.name})},DataTable.type("string",{detect:function(){return"string"},order:{pre:function(e){return _empty(e)?"":"string"==typeof e?e.toLowerCase():e.toString?e.toString():""}},search:_filterString(!1,!0)}),DataTable.type("html",{detect:function(e){return _empty(e)||"string"==typeof e&&-1!==e.indexOf("<")?"html":null},order:{pre:function(e){return _empty(e)?"":e.replace?_stripHtml(e).trim().toLowerCase():e+""}},search:_filterString(!0,!0)}),DataTable.type("date",{className:"dt-type-date",detect:function(e){var t;return(!e||e instanceof Date||_re_date.test(e))&&(null!==(t=Date.parse(e))&&!isNaN(t)||_empty(e))?"date":null},order:{pre:function(e){e=Date.parse(e);return isNaN(e)?-1/0:e}}}),DataTable.type("html-num-fmt",{className:"dt-type-numeric",detect:function(e,t){t=t.oLanguage.sDecimal;return _htmlNumeric(e,t,!0)?"html-num-fmt":null},order:{pre:function(e,t){t=t.oLanguage.sDecimal;return __numericReplace(e,t,_re_html,_re_formatted_numeric)}},search:_filterString(!0,!0)}),DataTable.type("html-num",{className:"dt-type-numeric",detect:function(e,t){t=t.oLanguage.sDecimal;return _htmlNumeric(e,t)?"html-num":null},order:{pre:function(e,t){t=t.oLanguage.sDecimal;return __numericReplace(e,t,_re_html)}},search:_filterString(!0,!0)}),DataTable.type("num-fmt",{className:"dt-type-numeric",detect:function(e,t){t=t.oLanguage.sDecimal;return _isNumber(e,t,!0)?"num-fmt":null},order:{pre:function(e,t){t=t.oLanguage.sDecimal;return __numericReplace(e,t,_re_formatted_numeric)}}}),DataTable.type("num",{className:"dt-type-numeric",detect:function(e,t){t=t.oLanguage.sDecimal;return _isNumber(e,t)?"num":null},order:{pre:function(e,t){t=t.oLanguage.sDecimal;return __numericReplace(e,t)}}}),function(e,t,a,n){var r;return 0===e||e&&"-"!==e?"number"==(r=typeof e)||"bigint"==r?e:+(e=(e=t?_numToDecimal(e,t):e).replace&&(a&&(e=e.replace(a,"")),n)?e.replace(n,""):e):-1/0});function _fnUpdateInfo(e,t,a){var n=e._iDisplayStart+1,r=e.fnDisplayEnd(),i=e.fnRecordsTotal(),o=e.fnRecordsDisplay(),l=o?t.text:t.empty;o!==i&&(l+=" "+t.search),l=_fnMacros(e,l+=t.postfix),t.callback&&(l=t.callback.call(e.oInstance,e,n,r,i,o,l)),a.html(l),_fnCallbackFire(e,null,"info",[e,a[0],l])}$.extend(!0,DataTable.ext.renderer,{footer:{_:function(e,t,a){t.addClass(a.tfoot.cell)}},header:{_:function(f,d,_){d.addClass(_.thead.cell),f.oFeatures.bSort||d.addClass(_.order.none);var e=f.bSortCellsTop,t=d.closest("thead").find("tr"),a=d.parent().index();"disable"===d.attr("data-dt-order")||"disable"===d.parent().attr("data-dt-order")||!0===e&&0!==a||!1===e&&a!==t.length-1||$(f.nTable).on("order.dt.DT",function(e,t,a){var n,r,i,o,l,s,u,c;f===t&&(n=_.order,c=t.api.columns(d),r=f.aoColumns[c.flatten()[0]],i=c.orderable().includes(!0),o="",u=c.indexes(),l=c.orderable(!0).flatten(),s=","+a.map(function(e){return e.col}).join(",")+",",d.removeClass(n.isAsc+" "+n.isDesc).toggleClass(n.none,!i).toggleClass(n.canAsc,i&&l.includes("asc")).toggleClass(n.canDesc,i&&l.includes("desc")),-1!==(l=s.indexOf(","+u.toArray().join(",")+","))&&(s=c.order(),d.addClass(s.includes("asc")?n.isAsc:""+s.includes("desc")?n.isDesc:"")),0===l?(u=a[0],c=r.asSorting,d.attr("aria-sort","asc"===u.dir?"ascending":"descending"),o=c[u.index+1]?"Reverse":"Remove"):d.removeAttr("aria-sort"),d.attr("aria-label",i?r.ariaTitle+t.api.i18n("oAria.orderable"+o):r.ariaTitle),i)&&(d.find(".dt-column-title").attr("role","button"),d.attr("tabindex",0))})}},layout:{_:function(e,t,a){var n=$("<div/>").addClass("dt-layout-row").appendTo(t);$.each(a,function(e,t){e=t.table?"":"dt-"+e+" ";t.table&&n.addClass("dt-layout-table"),$("<div/>").attr({id:t.id||null,class:"dt-layout-cell "+e+(t.className||"")}).append(t.contents).appendTo(n)})}}}),DataTable.feature={},DataTable.feature.register=function(e,t,a){DataTable.ext.features[e]=t,a&&_ext.feature.push({cFeature:a,fnInit:t})},DataTable.feature.register("info",function(e,t){var a,n,r;return e.oFeatures.bInfo?(a=e.oLanguage,n=e.sTableId,r=$("<div/>",{class:e.oClasses.info.container}),t=$.extend({callback:a.fnInfoCallback,empty:a.sInfoEmpty,postfix:a.sInfoPostFix,search:a.sInfoFiltered,text:a.sInfo},t),e.aoDrawCallback.push(function(e){_fnUpdateInfo(e,t,r)}),e._infoEl||(r.attr({"aria-live":"polite",id:n+"_info",role:"status"}),$(e.nTable).attr("aria-describedby",n+"_info"),e._infoEl=r),r):null},"i");var __searchCounter=0;function _pagingDraw(t,e,a){if(t._bInitComplete){for(var n=DataTable.ext.pager[a.type],r=t.oLanguage.oAria.paginate||{},i=t._iDisplayStart,o=t._iDisplayLength,l=t.fnRecordsDisplay(),s=-1===o,u=s?0:Math.ceil(i/o),c=s?1:Math.ceil(l/o),f=n().map(function(e){return"numbers"===e?_pagingNumbers(u,c,a.buttons,a.boundaryNumbers):e}).flat(),d=[],_=0;_<f.length;_++){var p=f[_],h=_pagingButtonInfo(t,p,u,c),g=_fnRenderer(t,"pagingButton")(t,p,h.display,h.active,h.disabled);$(g.clicker).attr({"aria-controls":t.sTableId,"aria-disabled":h.disabled?"true":null,"aria-current":h.active?"page":null,"aria-label":r[p],"data-dt-idx":p,tabIndex:h.disabled?-1:t.iTabIndex}),"number"!=typeof p&&$(g.clicker).addClass(p),_fnBindAction(g.clicker,{action:p},function(e){e.preventDefault(),_fnPageChange(t,e.data.action,!0)}),d.push(g.display)}i=_fnRenderer(t,"pagingContainer")(t,d),s=e.find(document.activeElement).data("dt-idx");e.empty().append(i),void 0!==s&&e.find("[data-dt-idx="+s+"]").trigger("focus"),d.length&&1<a.numbers&&$(e).height()>=2*$(d[0]).outerHeight()-10&&_pagingDraw(t,e,$.extend({},a,{numbers:a.numbers-2}))}}function _pagingButtonInfo(e,t,a,n){var r=e.oLanguage.oPaginate,i={display:"",active:!1,disabled:!1};switch(t){case"ellipsis":i.display="…",i.disabled=!0;break;case"first":i.display=r.sFirst,0===a&&(i.disabled=!0);break;case"previous":i.display=r.sPrevious,0===a&&(i.disabled=!0);break;case"next":i.display=r.sNext,0!==n&&a!==n-1||(i.disabled=!0);break;case"last":i.display=r.sLast,0!==n&&a!==n-1||(i.disabled=!0);break;default:"number"==typeof t&&(i.display=e.fnFormatNumber(t+1),a===t)&&(i.active=!0)}return i}function _pagingNumbers(e,t,a,n){var r=[],i=Math.floor(a/2),o=n?2:1,l=n?1:0;return t<=a?r=_range(0,t):1===a?r=[e]:3===a?e<=1?r=[0,1,"ellipsis"]:t-2<=e?(r=_range(t-2,t)).unshift("ellipsis"):r=["ellipsis",e,"ellipsis"]:e<=i?((r=_range(0,a-o)).push("ellipsis"),n&&r.push(t-1)):t-1-i<=e?((r=_range(t-(a-o),t)).unshift("ellipsis"),n&&r.unshift(0)):((r=_range(e-i+o,e+i-l)).push("ellipsis"),r.unshift("ellipsis"),n&&(r.push(t-1),r.unshift(0))),r}DataTable.feature.register("search",function(a,e){var t,n,r,i,o,l,s,u,c,f;return a.oFeatures.bFilter?(t=a.oClasses.search,n=a.sTableId,c=a.oLanguage,r=a.oPreviousSearch,i='<input type="search" class="'+t.input+'"/>',-1===(e=$.extend({placeholder:c.sSearchPlaceholder,text:c.sSearch},e)).text.indexOf("_INPUT_")&&(e.text+="_INPUT_"),e.text=_fnMacros(a,e.text),c=e.text.match(/_INPUT_$/),s=e.text.match(/^_INPUT_/),o=e.text.replace(/_INPUT_/,""),l="<label>"+e.text+"</label>",s?l="_INPUT_<label>"+o+"</label>":c&&(l="<label>"+o+"</label>_INPUT_"),(s=$("<div>").addClass(t.container).append(l.replace(/_INPUT_/,i))).find("label").attr("for","dt-search-"+__searchCounter),s.find("input").attr("id","dt-search-"+__searchCounter),__searchCounter++,u=function(e){var t=this.value;r.return&&"Enter"!==e.key||t!=r.search&&(r.search=t,_fnFilterComplete(a,r),a._iDisplayStart=0,_fnDraw(a))},c=null!==a.searchDelay?a.searchDelay:0,f=$("input",s).val(r.search).attr("placeholder",e.placeholder).on("keyup.DT search.DT input.DT paste.DT cut.DT",c?DataTable.util.debounce(u,c):u).on("mouseup.DT",function(e){setTimeout(function(){u.call(f[0],e)},10)}).on("keypress.DT",function(e){if(13==e.keyCode)return!1}).attr("aria-controls",n),$(a.nTable).on("search.dt.DT",function(e,t){a===t&&f[0]!==document.activeElement&&f.val("function"!=typeof r.search?r.search:"")}),s):null},"f"),DataTable.feature.register("paging",function(e,t){if(!e.oFeatures.bPaginate)return null;(t=$.extend({buttons:DataTable.ext.pager.numbers_length,type:e.sPaginationType,boundaryNumbers:!0},t)).numbers&&(t.buttons=t.numbers);function a(){_pagingDraw(e,n,t)}var n=$("<div/>").addClass(e.oClasses.paging.container+" paging_"+t.type);return e.aoDrawCallback.push(a),$(e.nTable).on("column-sizing.dt.DT",a),n},"p");var __lengthCounter=0;DataTable.feature.register("pageLength",function(n,e){var t=n.oFeatures;if(!t.bPaginate||!t.bLengthChange)return null;e=$.extend({menu:n.aLengthMenu,text:n.oLanguage.sLengthMenu},e);var t=n.oClasses.length,a=n.sTableId,r=e.menu,i=[],o=[];if(Array.isArray(r[0]))i=r[0],o=r[1];else for(p=0;p<r.length;p++)$.isPlainObject(r[p])?(i.push(r[p].value),o.push(r[p].label)):(i.push(r[p]),o.push(r[p]));function l(t){d.forEach(function(e){e.el.textContent=_fnMacros(n,e.text,t)})}for(var s=e.text.match(/_MENU_$/),u=e.text.match(/^_MENU_/),c=e.text.replace(/_MENU_/,""),e="<label>"+e.text+"</label>",f=(u?e="_MENU_<label>"+c+"</label>":s&&(e="<label>"+c+"</label>_MENU_"),$("<div/>").addClass(t.container).append(e.replace("_MENU_","<span></span>"))),d=[],_=(f.find("label")[0].childNodes.forEach(function(e){e.nodeType===Node.TEXT_NODE&&d.push({el:e,text:e.textContent})}),$("<select/>",{name:a+"_length","aria-controls":a,class:t.select})),p=0;p<i.length;p++)_[0][p]=new Option("number"==typeof o[p]?n.fnFormatNumber(o[p]):o[p],i[p]);return f.find("label").attr("for","dt-length-"+__lengthCounter),_.attr("id","dt-length-"+__lengthCounter),__lengthCounter++,f.find("span").replaceWith(_),$("select",f).val(n._iDisplayLength).on("change.DT",function(){_fnLengthChange(n,$(this).val()),_fnDraw(n)}),$(n.nTable).on("length.dt.DT",function(e,t,a){n===t&&($("select",f).val(a),l(a))}),l(n._iDisplayLength),f},"l"),(($.fn.dataTable=DataTable).$=$).fn.dataTableSettings=DataTable.settings,$.fn.dataTableExt=DataTable.ext,$.fn.DataTable=function(e){return $(this).dataTable(e).api()},$.each(DataTable,function(e,t){$.fn.DataTable[e]=t});export default DataTable;node_modules/datatables.net/js/dataTables.mjs000064400001261170151676725350015336 0ustar00/*! DataTables 2.0.8 * © SpryMedia Ltd - datatables.net/license */ import jQuery from 'jquery'; // DataTables code uses $ internally, but we want to be able to // reassign $ with the `use` method, so it is a regular var. var $ = jQuery; var DataTable = function ( selector, options ) { // Check if called with a window or jQuery object for DOM less applications // This is for backwards compatibility if (DataTable.factory(selector, options)) { return DataTable; } // When creating with `new`, create a new DataTable, returning the API instance if (this instanceof DataTable) { return $(selector).DataTable(options); } else { // Argument switching options = selector; } var _that = this; var emptyInit = options === undefined; var len = this.length; if ( emptyInit ) { options = {}; } // Method to get DT API instance from jQuery object this.api = function () { return new _Api( this ); }; this.each(function() { // For each initialisation we want to give it a clean initialisation // object that can be bashed around var o = {}; var oInit = len > 1 ? // optimisation for single table case _fnExtend( o, options, true ) : options; var i=0, iLen; var sId = this.getAttribute( 'id' ); var bInitHandedOff = false; var defaults = DataTable.defaults; var $this = $(this); /* Sanity check */ if ( this.nodeName.toLowerCase() != 'table' ) { _fnLog( null, 0, 'Non-table node initialisation ('+this.nodeName+')', 2 ); return; } $(this).trigger( 'options.dt', oInit ); /* Backwards compatibility for the defaults */ _fnCompatOpts( defaults ); _fnCompatCols( defaults.column ); /* Convert the camel-case defaults to Hungarian */ _fnCamelToHungarian( defaults, defaults, true ); _fnCamelToHungarian( defaults.column, defaults.column, true ); /* Setting up the initialisation object */ _fnCamelToHungarian( defaults, $.extend( oInit, $this.data() ), true ); /* Check to see if we are re-initialising a table */ var allSettings = DataTable.settings; for ( i=0, iLen=allSettings.length ; i<iLen ; i++ ) { var s = allSettings[i]; /* Base check on table node */ if ( s.nTable == this || (s.nTHead && s.nTHead.parentNode == this) || (s.nTFoot && s.nTFoot.parentNode == this) ) { var bRetrieve = oInit.bRetrieve !== undefined ? oInit.bRetrieve : defaults.bRetrieve; var bDestroy = oInit.bDestroy !== undefined ? oInit.bDestroy : defaults.bDestroy; if ( emptyInit || bRetrieve ) { return s.oInstance; } else if ( bDestroy ) { new DataTable.Api(s).destroy(); break; } else { _fnLog( s, 0, 'Cannot reinitialise DataTable', 3 ); return; } } /* If the element we are initialising has the same ID as a table which was previously * initialised, but the table nodes don't match (from before) then we destroy the old * instance by simply deleting it. This is under the assumption that the table has been * destroyed by other methods. Anyone using non-id selectors will need to do this manually */ if ( s.sTableId == this.id ) { allSettings.splice( i, 1 ); break; } } /* Ensure the table has an ID - required for accessibility */ if ( sId === null || sId === "" ) { sId = "DataTables_Table_"+(DataTable.ext._unique++); this.id = sId; } /* Create the settings object for this table and set some of the default parameters */ var oSettings = $.extend( true, {}, DataTable.models.oSettings, { "sDestroyWidth": $this[0].style.width, "sInstance": sId, "sTableId": sId, colgroup: $('<colgroup>').prependTo(this), fastData: function (row, column, type) { return _fnGetCellData(oSettings, row, column, type); } } ); oSettings.nTable = this; oSettings.oInit = oInit; allSettings.push( oSettings ); // Make a single API instance available for internal handling oSettings.api = new _Api( oSettings ); // Need to add the instance after the instance after the settings object has been added // to the settings array, so we can self reference the table instance if more than one oSettings.oInstance = (_that.length===1) ? _that : $this.dataTable(); // Backwards compatibility, before we apply all the defaults _fnCompatOpts( oInit ); // If the length menu is given, but the init display length is not, use the length menu if ( oInit.aLengthMenu && ! oInit.iDisplayLength ) { oInit.iDisplayLength = Array.isArray(oInit.aLengthMenu[0]) ? oInit.aLengthMenu[0][0] : $.isPlainObject( oInit.aLengthMenu[0] ) ? oInit.aLengthMenu[0].value : oInit.aLengthMenu[0]; } // Apply the defaults and init options to make a single init object will all // options defined from defaults and instance options. oInit = _fnExtend( $.extend( true, {}, defaults ), oInit ); // Map the initialisation options onto the settings object _fnMap( oSettings.oFeatures, oInit, [ "bPaginate", "bLengthChange", "bFilter", "bSort", "bSortMulti", "bInfo", "bProcessing", "bAutoWidth", "bSortClasses", "bServerSide", "bDeferRender" ] ); _fnMap( oSettings, oInit, [ "ajax", "fnFormatNumber", "sServerMethod", "aaSorting", "aaSortingFixed", "aLengthMenu", "sPaginationType", "iStateDuration", "bSortCellsTop", "iTabIndex", "sDom", "fnStateLoadCallback", "fnStateSaveCallback", "renderer", "searchDelay", "rowId", "caption", "layout", [ "iCookieDuration", "iStateDuration" ], // backwards compat [ "oSearch", "oPreviousSearch" ], [ "aoSearchCols", "aoPreSearchCols" ], [ "iDisplayLength", "_iDisplayLength" ] ] ); _fnMap( oSettings.oScroll, oInit, [ [ "sScrollX", "sX" ], [ "sScrollXInner", "sXInner" ], [ "sScrollY", "sY" ], [ "bScrollCollapse", "bCollapse" ] ] ); _fnMap( oSettings.oLanguage, oInit, "fnInfoCallback" ); /* Callback functions which are array driven */ _fnCallbackReg( oSettings, 'aoDrawCallback', oInit.fnDrawCallback ); _fnCallbackReg( oSettings, 'aoStateSaveParams', oInit.fnStateSaveParams ); _fnCallbackReg( oSettings, 'aoStateLoadParams', oInit.fnStateLoadParams ); _fnCallbackReg( oSettings, 'aoStateLoaded', oInit.fnStateLoaded ); _fnCallbackReg( oSettings, 'aoRowCallback', oInit.fnRowCallback ); _fnCallbackReg( oSettings, 'aoRowCreatedCallback', oInit.fnCreatedRow ); _fnCallbackReg( oSettings, 'aoHeaderCallback', oInit.fnHeaderCallback ); _fnCallbackReg( oSettings, 'aoFooterCallback', oInit.fnFooterCallback ); _fnCallbackReg( oSettings, 'aoInitComplete', oInit.fnInitComplete ); _fnCallbackReg( oSettings, 'aoPreDrawCallback', oInit.fnPreDrawCallback ); oSettings.rowIdFn = _fnGetObjectDataFn( oInit.rowId ); /* Browser support detection */ _fnBrowserDetect( oSettings ); var oClasses = oSettings.oClasses; $.extend( oClasses, DataTable.ext.classes, oInit.oClasses ); $this.addClass( oClasses.table ); if (! oSettings.oFeatures.bPaginate) { oInit.iDisplayStart = 0; } if ( oSettings.iInitDisplayStart === undefined ) { /* Display start point, taking into account the save saving */ oSettings.iInitDisplayStart = oInit.iDisplayStart; oSettings._iDisplayStart = oInit.iDisplayStart; } /* Language definitions */ var oLanguage = oSettings.oLanguage; $.extend( true, oLanguage, oInit.oLanguage ); if ( oLanguage.sUrl ) { /* Get the language definitions from a file - because this Ajax call makes the language * get async to the remainder of this function we use bInitHandedOff to indicate that * _fnInitialise will be fired by the returned Ajax handler, rather than the constructor */ $.ajax( { dataType: 'json', url: oLanguage.sUrl, success: function ( json ) { _fnCamelToHungarian( defaults.oLanguage, json ); $.extend( true, oLanguage, json, oSettings.oInit.oLanguage ); _fnCallbackFire( oSettings, null, 'i18n', [oSettings], true); _fnInitialise( oSettings ); }, error: function () { // Error occurred loading language file _fnLog( oSettings, 0, 'i18n file loading error', 21 ); // continue on as best we can _fnInitialise( oSettings ); } } ); bInitHandedOff = true; } else { _fnCallbackFire( oSettings, null, 'i18n', [oSettings]); } /* * Columns * See if we should load columns automatically or use defined ones */ var columnsInit = []; var thead = this.getElementsByTagName('thead'); var initHeaderLayout = _fnDetectHeader( oSettings, thead[0] ); // If we don't have a columns array, then generate one with nulls if ( oInit.aoColumns ) { columnsInit = oInit.aoColumns; } else if ( initHeaderLayout.length ) { for ( i=0, iLen=initHeaderLayout[0].length ; i<iLen ; i++ ) { columnsInit.push( null ); } } // Add the columns for ( i=0, iLen=columnsInit.length ; i<iLen ; i++ ) { _fnAddColumn( oSettings ); } // Apply the column definitions _fnApplyColumnDefs( oSettings, oInit.aoColumnDefs, columnsInit, initHeaderLayout, function (iCol, oDef) { _fnColumnOptions( oSettings, iCol, oDef ); } ); /* HTML5 attribute detection - build an mData object automatically if the * attributes are found */ var rowOne = $this.children('tbody').find('tr').eq(0); if ( rowOne.length ) { var a = function ( cell, name ) { return cell.getAttribute( 'data-'+name ) !== null ? name : null; }; $( rowOne[0] ).children('th, td').each( function (i, cell) { var col = oSettings.aoColumns[i]; if (! col) { _fnLog( oSettings, 0, 'Incorrect column count', 18 ); } if ( col.mData === i ) { var sort = a( cell, 'sort' ) || a( cell, 'order' ); var filter = a( cell, 'filter' ) || a( cell, 'search' ); if ( sort !== null || filter !== null ) { col.mData = { _: i+'.display', sort: sort !== null ? i+'.@data-'+sort : undefined, type: sort !== null ? i+'.@data-'+sort : undefined, filter: filter !== null ? i+'.@data-'+filter : undefined }; col._isArrayHost = true; _fnColumnOptions( oSettings, i ); } } } ); } var features = oSettings.oFeatures; var loadedInit = function () { /* * Sorting * @todo For modularisation (1.11) this needs to do into a sort start up handler */ // If aaSorting is not defined, then we use the first indicator in asSorting // in case that has been altered, so the default sort reflects that option if ( oInit.aaSorting === undefined ) { var sorting = oSettings.aaSorting; for ( i=0, iLen=sorting.length ; i<iLen ; i++ ) { sorting[i][1] = oSettings.aoColumns[ i ].asSorting[0]; } } /* Do a first pass on the sorting classes (allows any size changes to be taken into * account, and also will apply sorting disabled classes if disabled */ _fnSortingClasses( oSettings ); _fnCallbackReg( oSettings, 'aoDrawCallback', function () { if ( oSettings.bSorted || _fnDataSource( oSettings ) === 'ssp' || features.bDeferRender ) { _fnSortingClasses( oSettings ); } } ); /* * Final init * Cache the header, body and footer as required, creating them if needed */ var caption = $this.children('caption'); if ( oSettings.caption ) { if ( caption.length === 0 ) { caption = $('<caption/>').appendTo( $this ); } caption.html( oSettings.caption ); } // Store the caption side, so we can remove the element from the document // when creating the element if (caption.length) { caption[0]._captionSide = caption.css('caption-side'); oSettings.captionNode = caption[0]; } if ( thead.length === 0 ) { thead = $('<thead/>').appendTo($this); } oSettings.nTHead = thead[0]; $('tr', thead).addClass(oClasses.thead.row); var tbody = $this.children('tbody'); if ( tbody.length === 0 ) { tbody = $('<tbody/>').insertAfter(thead); } oSettings.nTBody = tbody[0]; var tfoot = $this.children('tfoot'); if ( tfoot.length === 0 ) { // If we are a scrolling table, and no footer has been given, then we need to create // a tfoot element for the caption element to be appended to tfoot = $('<tfoot/>').appendTo($this); } oSettings.nTFoot = tfoot[0]; $('tr', tfoot).addClass(oClasses.tfoot.row); // Check if there is data passing into the constructor if ( oInit.aaData ) { for ( i=0 ; i<oInit.aaData.length ; i++ ) { _fnAddData( oSettings, oInit.aaData[ i ] ); } } else if ( _fnDataSource( oSettings ) == 'dom' ) { // Grab the data from the page _fnAddTr( oSettings, $(oSettings.nTBody).children('tr') ); } /* Copy the data index array */ oSettings.aiDisplay = oSettings.aiDisplayMaster.slice(); /* Initialisation complete - table can be drawn */ oSettings.bInitialised = true; /* Check if we need to initialise the table (it might not have been handed off to the * language processor) */ if ( bInitHandedOff === false ) { _fnInitialise( oSettings ); } }; /* Must be done after everything which can be overridden by the state saving! */ _fnCallbackReg( oSettings, 'aoDrawCallback', _fnSaveState ); if ( oInit.bStateSave ) { features.bStateSave = true; _fnLoadState( oSettings, oInit, loadedInit ); } else { loadedInit(); } } ); _that = null; return this; }; /** * DataTables extensions * * This namespace acts as a collection area for plug-ins that can be used to * extend DataTables capabilities. Indeed many of the build in methods * use this method to provide their own capabilities (sorting methods for * example). * * Note that this namespace is aliased to `jQuery.fn.dataTableExt` for legacy * reasons * * @namespace */ DataTable.ext = _ext = { /** * Buttons. For use with the Buttons extension for DataTables. This is * defined here so other extensions can define buttons regardless of load * order. It is _not_ used by DataTables core. * * @type object * @default {} */ buttons: {}, /** * Element class names * * @type object * @default {} */ classes: {}, /** * DataTables build type (expanded by the download builder) * * @type string */ builder: "-source-", /** * Error reporting. * * How should DataTables report an error. Can take the value 'alert', * 'throw', 'none' or a function. * * @type string|function * @default alert */ errMode: "alert", /** * Legacy so v1 plug-ins don't throw js errors on load */ feature: [], /** * Feature plug-ins. * * This is an object of callbacks which provide the features for DataTables * to be initialised via the `layout` option. */ features: {}, /** * Row searching. * * This method of searching is complimentary to the default type based * searching, and a lot more comprehensive as it allows you complete control * over the searching logic. Each element in this array is a function * (parameters described below) that is called for every row in the table, * and your logic decides if it should be included in the searching data set * or not. * * Searching functions have the following input parameters: * * 1. `{object}` DataTables settings object: see * {@link DataTable.models.oSettings} * 2. `{array|object}` Data for the row to be processed (same as the * original format that was passed in as the data source, or an array * from a DOM data source * 3. `{int}` Row index ({@link DataTable.models.oSettings.aoData}), which * can be useful to retrieve the `TR` element if you need DOM interaction. * * And the following return is expected: * * * {boolean} Include the row in the searched result set (true) or not * (false) * * Note that as with the main search ability in DataTables, technically this * is "filtering", since it is subtractive. However, for consistency in * naming we call it searching here. * * @type array * @default [] * * @example * // The following example shows custom search being applied to the * // fourth column (i.e. the data[3] index) based on two input values * // from the end-user, matching the data in a certain range. * $.fn.dataTable.ext.search.push( * function( settings, data, dataIndex ) { * var min = document.getElementById('min').value * 1; * var max = document.getElementById('max').value * 1; * var version = data[3] == "-" ? 0 : data[3]*1; * * if ( min == "" && max == "" ) { * return true; * } * else if ( min == "" && version < max ) { * return true; * } * else if ( min < version && "" == max ) { * return true; * } * else if ( min < version && version < max ) { * return true; * } * return false; * } * ); */ search: [], /** * Selector extensions * * The `selector` option can be used to extend the options available for the * selector modifier options (`selector-modifier` object data type) that * each of the three built in selector types offer (row, column and cell + * their plural counterparts). For example the Select extension uses this * mechanism to provide an option to select only rows, columns and cells * that have been marked as selected by the end user (`{selected: true}`), * which can be used in conjunction with the existing built in selector * options. * * Each property is an array to which functions can be pushed. The functions * take three attributes: * * * Settings object for the host table * * Options object (`selector-modifier` object type) * * Array of selected item indexes * * The return is an array of the resulting item indexes after the custom * selector has been applied. * * @type object */ selector: { cell: [], column: [], row: [] }, /** * Legacy configuration options. Enable and disable legacy options that * are available in DataTables. * * @type object */ legacy: { /** * Enable / disable DataTables 1.9 compatible server-side processing * requests * * @type boolean * @default null */ ajax: null }, /** * Pagination plug-in methods. * * Each entry in this object is a function and defines which buttons should * be shown by the pagination rendering method that is used for the table: * {@link DataTable.ext.renderer.pageButton}. The renderer addresses how the * buttons are displayed in the document, while the functions here tell it * what buttons to display. This is done by returning an array of button * descriptions (what each button will do). * * Pagination types (the four built in options and any additional plug-in * options defined here) can be used through the `paginationType` * initialisation parameter. * * The functions defined take two parameters: * * 1. `{int} page` The current page index * 2. `{int} pages` The number of pages in the table * * Each function is expected to return an array where each element of the * array can be one of: * * * `first` - Jump to first page when activated * * `last` - Jump to last page when activated * * `previous` - Show previous page when activated * * `next` - Show next page when activated * * `{int}` - Show page of the index given * * `{array}` - A nested array containing the above elements to add a * containing 'DIV' element (might be useful for styling). * * Note that DataTables v1.9- used this object slightly differently whereby * an object with two functions would be defined for each plug-in. That * ability is still supported by DataTables 1.10+ to provide backwards * compatibility, but this option of use is now decremented and no longer * documented in DataTables 1.10+. * * @type object * @default {} * * @example * // Show previous, next and current page buttons only * $.fn.dataTableExt.oPagination.current = function ( page, pages ) { * return [ 'previous', page, 'next' ]; * }; */ pager: {}, renderer: { pageButton: {}, header: {} }, /** * Ordering plug-ins - custom data source * * The extension options for ordering of data available here is complimentary * to the default type based ordering that DataTables typically uses. It * allows much greater control over the the data that is being used to * order a column, but is necessarily therefore more complex. * * This type of ordering is useful if you want to do ordering based on data * live from the DOM (for example the contents of an 'input' element) rather * than just the static string that DataTables knows of. * * The way these plug-ins work is that you create an array of the values you * wish to be ordering for the column in question and then return that * array. The data in the array much be in the index order of the rows in * the table (not the currently ordering order!). Which order data gathering * function is run here depends on the `dt-init columns.orderDataType` * parameter that is used for the column (if any). * * The functions defined take two parameters: * * 1. `{object}` DataTables settings object: see * {@link DataTable.models.oSettings} * 2. `{int}` Target column index * * Each function is expected to return an array: * * * `{array}` Data for the column to be ordering upon * * @type array * * @example * // Ordering using `input` node values * $.fn.dataTable.ext.order['dom-text'] = function ( settings, col ) * { * return this.api().column( col, {order:'index'} ).nodes().map( function ( td, i ) { * return $('input', td).val(); * } ); * } */ order: {}, /** * Type based plug-ins. * * Each column in DataTables has a type assigned to it, either by automatic * detection or by direct assignment using the `type` option for the column. * The type of a column will effect how it is ordering and search (plug-ins * can also make use of the column type if required). * * @namespace */ type: { /** * Automatic column class assignment */ className: {}, /** * Type detection functions. * * The functions defined in this object are used to automatically detect * a column's type, making initialisation of DataTables super easy, even * when complex data is in the table. * * The functions defined take two parameters: * * 1. `{*}` Data from the column cell to be analysed * 2. `{settings}` DataTables settings object. This can be used to * perform context specific type detection - for example detection * based on language settings such as using a comma for a decimal * place. Generally speaking the options from the settings will not * be required * * Each function is expected to return: * * * `{string|null}` Data type detected, or null if unknown (and thus * pass it on to the other type detection functions. * * @type array * * @example * // Currency type detection plug-in: * $.fn.dataTable.ext.type.detect.push( * function ( data, settings ) { * // Check the numeric part * if ( ! data.substring(1).match(/[0-9]/) ) { * return null; * } * * // Check prefixed by currency * if ( data.charAt(0) == '$' || data.charAt(0) == '£' ) { * return 'currency'; * } * return null; * } * ); */ detect: [], /** * Automatic renderer assignment */ render: {}, /** * Type based search formatting. * * The type based searching functions can be used to pre-format the * data to be search on. For example, it can be used to strip HTML * tags or to de-format telephone numbers for numeric only searching. * * Note that is a search is not defined for a column of a given type, * no search formatting will be performed. * * Pre-processing of searching data plug-ins - When you assign the sType * for a column (or have it automatically detected for you by DataTables * or a type detection plug-in), you will typically be using this for * custom sorting, but it can also be used to provide custom searching * by allowing you to pre-processing the data and returning the data in * the format that should be searched upon. This is done by adding * functions this object with a parameter name which matches the sType * for that target column. This is the corollary of <i>afnSortData</i> * for searching data. * * The functions defined take a single parameter: * * 1. `{*}` Data from the column cell to be prepared for searching * * Each function is expected to return: * * * `{string|null}` Formatted string that will be used for the searching. * * @type object * @default {} * * @example * $.fn.dataTable.ext.type.search['title-numeric'] = function ( d ) { * return d.replace(/\n/g," ").replace( /<.*?>/g, "" ); * } */ search: {}, /** * Type based ordering. * * The column type tells DataTables what ordering to apply to the table * when a column is sorted upon. The order for each type that is defined, * is defined by the functions available in this object. * * Each ordering option can be described by three properties added to * this object: * * * `{type}-pre` - Pre-formatting function * * `{type}-asc` - Ascending order function * * `{type}-desc` - Descending order function * * All three can be used together, only `{type}-pre` or only * `{type}-asc` and `{type}-desc` together. It is generally recommended * that only `{type}-pre` is used, as this provides the optimal * implementation in terms of speed, although the others are provided * for compatibility with existing Javascript sort functions. * * `{type}-pre`: Functions defined take a single parameter: * * 1. `{*}` Data from the column cell to be prepared for ordering * * And return: * * * `{*}` Data to be sorted upon * * `{type}-asc` and `{type}-desc`: Functions are typical Javascript sort * functions, taking two parameters: * * 1. `{*}` Data to compare to the second parameter * 2. `{*}` Data to compare to the first parameter * * And returning: * * * `{*}` Ordering match: <0 if first parameter should be sorted lower * than the second parameter, ===0 if the two parameters are equal and * >0 if the first parameter should be sorted height than the second * parameter. * * @type object * @default {} * * @example * // Numeric ordering of formatted numbers with a pre-formatter * $.extend( $.fn.dataTable.ext.type.order, { * "string-pre": function(x) { * a = (a === "-" || a === "") ? 0 : a.replace( /[^\d\-\.]/g, "" ); * return parseFloat( a ); * } * } ); * * @example * // Case-sensitive string ordering, with no pre-formatting method * $.extend( $.fn.dataTable.ext.order, { * "string-case-asc": function(x,y) { * return ((x < y) ? -1 : ((x > y) ? 1 : 0)); * }, * "string-case-desc": function(x,y) { * return ((x < y) ? 1 : ((x > y) ? -1 : 0)); * } * } ); */ order: {} }, /** * Unique DataTables instance counter * * @type int * @private */ _unique: 0, // // Depreciated // The following properties are retained for backwards compatibility only. // The should not be used in new projects and will be removed in a future // version // /** * Version check function. * @type function * @depreciated Since 1.10 */ fnVersionCheck: DataTable.fnVersionCheck, /** * Index for what 'this' index API functions should use * @type int * @deprecated Since v1.10 */ iApiIndex: 0, /** * Software version * @type string * @deprecated Since v1.10 */ sVersion: DataTable.version }; // // Backwards compatibility. Alias to pre 1.10 Hungarian notation counter parts // $.extend( _ext, { afnFiltering: _ext.search, aTypes: _ext.type.detect, ofnSearch: _ext.type.search, oSort: _ext.type.order, afnSortData: _ext.order, aoFeatures: _ext.feature, oStdClasses: _ext.classes, oPagination: _ext.pager } ); $.extend( DataTable.ext.classes, { container: 'dt-container', empty: { row: 'dt-empty' }, info: { container: 'dt-info' }, length: { container: 'dt-length', select: 'dt-input' }, order: { canAsc: 'dt-orderable-asc', canDesc: 'dt-orderable-desc', isAsc: 'dt-ordering-asc', isDesc: 'dt-ordering-desc', none: 'dt-orderable-none', position: 'sorting_' }, processing: { container: 'dt-processing' }, scrolling: { body: 'dt-scroll-body', container: 'dt-scroll', footer: { self: 'dt-scroll-foot', inner: 'dt-scroll-footInner' }, header: { self: 'dt-scroll-head', inner: 'dt-scroll-headInner' } }, search: { container: 'dt-search', input: 'dt-input' }, table: 'dataTable', tbody: { cell: '', row: '' }, thead: { cell: '', row: '' }, tfoot: { cell: '', row: '' }, paging: { active: 'current', button: 'dt-paging-button', container: 'dt-paging', disabled: 'disabled' } } ); /* * It is useful to have variables which are scoped locally so only the * DataTables functions can access them and they don't leak into global space. * At the same time these functions are often useful over multiple files in the * core and API, so we list, or at least document, all variables which are used * by DataTables as private variables here. This also ensures that there is no * clashing of variable names and that they can easily referenced for reuse. */ // Defined else where // _selector_run // _selector_opts // _selector_row_indexes var _ext; // DataTable.ext var _Api; // DataTable.Api var _api_register; // DataTable.Api.register var _api_registerPlural; // DataTable.Api.registerPlural var _re_dic = {}; var _re_new_lines = /[\r\n\u2028]/g; var _re_html = /<([^>]*>)/g; var _max_str_len = Math.pow(2, 28); // This is not strict ISO8601 - Date.parse() is quite lax, although // implementations differ between browsers. var _re_date = /^\d{2,4}[./-]\d{1,2}[./-]\d{1,2}([T ]{1}\d{1,2}[:.]\d{2}([.:]\d{2})?)?$/; // Escape regular expression special characters var _re_escape_regex = new RegExp( '(\\' + [ '/', '.', '*', '+', '?', '|', '(', ')', '[', ']', '{', '}', '\\', '$', '^', '-' ].join('|\\') + ')', 'g' ); // https://en.wikipedia.org/wiki/Foreign_exchange_market // - \u20BD - Russian ruble. // - \u20a9 - South Korean Won // - \u20BA - Turkish Lira // - \u20B9 - Indian Rupee // - R - Brazil (R$) and South Africa // - fr - Swiss Franc // - kr - Swedish krona, Norwegian krone and Danish krone // - \u2009 is thin space and \u202F is narrow no-break space, both used in many // - Ƀ - Bitcoin // - Ξ - Ethereum // standards as thousands separators. var _re_formatted_numeric = /['\u00A0,$£€¥%\u2009\u202F\u20BD\u20a9\u20BArfkɃΞ]/gi; var _empty = function ( d ) { return !d || d === true || d === '-' ? true : false; }; var _intVal = function ( s ) { var integer = parseInt( s, 10 ); return !isNaN(integer) && isFinite(s) ? integer : null; }; // Convert from a formatted number with characters other than `.` as the // decimal place, to a Javascript number var _numToDecimal = function ( num, decimalPoint ) { // Cache created regular expressions for speed as this function is called often if ( ! _re_dic[ decimalPoint ] ) { _re_dic[ decimalPoint ] = new RegExp( _fnEscapeRegex( decimalPoint ), 'g' ); } return typeof num === 'string' && decimalPoint !== '.' ? num.replace( /\./g, '' ).replace( _re_dic[ decimalPoint ], '.' ) : num; }; var _isNumber = function ( d, decimalPoint, formatted ) { var type = typeof d; var strType = type === 'string'; if ( type === 'number' || type === 'bigint') { return true; } // If empty return immediately so there must be a number if it is a // formatted string (this stops the string "k", or "kr", etc being detected // as a formatted number for currency if ( _empty( d ) ) { return true; } if ( decimalPoint && strType ) { d = _numToDecimal( d, decimalPoint ); } if ( formatted && strType ) { d = d.replace( _re_formatted_numeric, '' ); } return !isNaN( parseFloat(d) ) && isFinite( d ); }; // A string without HTML in it can be considered to be HTML still var _isHtml = function ( d ) { return _empty( d ) || typeof d === 'string'; }; // Is a string a number surrounded by HTML? var _htmlNumeric = function ( d, decimalPoint, formatted ) { if ( _empty( d ) ) { return true; } // input and select strings mean that this isn't just a number if (typeof d === 'string' && d.match(/<(input|select)/i)) { return null; } var html = _isHtml( d ); return ! html ? null : _isNumber( _stripHtml( d ), decimalPoint, formatted ) ? true : null; }; var _pluck = function ( a, prop, prop2 ) { var out = []; var i=0, ien=a.length; // Could have the test in the loop for slightly smaller code, but speed // is essential here if ( prop2 !== undefined ) { for ( ; i<ien ; i++ ) { if ( a[i] && a[i][ prop ] ) { out.push( a[i][ prop ][ prop2 ] ); } } } else { for ( ; i<ien ; i++ ) { if ( a[i] ) { out.push( a[i][ prop ] ); } } } return out; }; // Basically the same as _pluck, but rather than looping over `a` we use `order` // as the indexes to pick from `a` var _pluck_order = function ( a, order, prop, prop2 ) { var out = []; var i=0, ien=order.length; // Could have the test in the loop for slightly smaller code, but speed // is essential here if ( prop2 !== undefined ) { for ( ; i<ien ; i++ ) { if ( a[ order[i] ][ prop ] ) { out.push( a[ order[i] ][ prop ][ prop2 ] ); } } } else { for ( ; i<ien ; i++ ) { if ( a[ order[i] ] ) { out.push( a[ order[i] ][ prop ] ); } } } return out; }; var _range = function ( len, start ) { var out = []; var end; if ( start === undefined ) { start = 0; end = len; } else { end = start; start = len; } for ( var i=start ; i<end ; i++ ) { out.push( i ); } return out; }; var _removeEmpty = function ( a ) { var out = []; for ( var i=0, ien=a.length ; i<ien ; i++ ) { if ( a[i] ) { // careful - will remove all falsy values! out.push( a[i] ); } } return out; }; // Replaceable function in api.util var _stripHtml = function (input) { // Irrelevant check to workaround CodeQL's false positive on the regex if (input.length > _max_str_len) { throw new Error('Exceeded max str len'); } var previous; input = input.replace(_re_html, ''); // Complete tags // Safety for incomplete script tag - use do / while to ensure that // we get all instances do { previous = input; input = input.replace(/<script/i, ''); } while (input !== previous); return previous; }; // Replaceable function in api.util var _escapeHtml = function ( d ) { if (Array.isArray(d)) { d = d.join(','); } return typeof d === 'string' ? d .replace(/&/g, '&') .replace(/</g, '<') .replace(/>/g, '>') .replace(/"/g, '"') : d; }; // Remove diacritics from a string by decomposing it and then removing // non-ascii characters var _normalize = function (str, both) { if (typeof str !== 'string') { return str; } // It is faster to just run `normalize` than it is to check if // we need to with a regex! var res = str.normalize("NFD"); // Equally, here we check if a regex is needed or not return res.length !== str.length ? (both === true ? str + ' ' : '' ) + res.replace(/[\u0300-\u036f]/g, "") : res; } /** * Determine if all values in the array are unique. This means we can short * cut the _unique method at the cost of a single loop. A sorted array is used * to easily check the values. * * @param {array} src Source array * @return {boolean} true if all unique, false otherwise * @ignore */ var _areAllUnique = function ( src ) { if ( src.length < 2 ) { return true; } var sorted = src.slice().sort(); var last = sorted[0]; for ( var i=1, ien=sorted.length ; i<ien ; i++ ) { if ( sorted[i] === last ) { return false; } last = sorted[i]; } return true; }; /** * Find the unique elements in a source array. * * @param {array} src Source array * @return {array} Array of unique items * @ignore */ var _unique = function ( src ) { if (Array.from && Set) { return Array.from(new Set(src)); } if ( _areAllUnique( src ) ) { return src.slice(); } // A faster unique method is to use object keys to identify used values, // but this doesn't work with arrays or objects, which we must also // consider. See jsperf.app/compare-array-unique-versions/4 for more // information. var out = [], val, i, ien=src.length, j, k=0; again: for ( i=0 ; i<ien ; i++ ) { val = src[i]; for ( j=0 ; j<k ; j++ ) { if ( out[j] === val ) { continue again; } } out.push( val ); k++; } return out; }; // Surprisingly this is faster than [].concat.apply // https://jsperf.com/flatten-an-array-loop-vs-reduce/2 var _flatten = function (out, val) { if (Array.isArray(val)) { for (var i=0 ; i<val.length ; i++) { _flatten(out, val[i]); } } else { out.push(val); } return out; } // Similar to jQuery's addClass, but use classList.add function _addClass(el, name) { if (name) { name.split(' ').forEach(function (n) { if (n) { // `add` does deduplication, so no need to check `contains` el.classList.add(n); } }); } } /** * DataTables utility methods * * This namespace provides helper methods that DataTables uses internally to * create a DataTable, but which are not exclusively used only for DataTables. * These methods can be used by extension authors to save the duplication of * code. * * @namespace */ DataTable.util = { /** * Return a string with diacritic characters decomposed * @param {*} mixed Function or string to normalize * @param {*} both Return original string and the normalized string * @returns String or undefined */ diacritics: function (mixed, both) { var type = typeof mixed; if (type !== 'function') { return _normalize(mixed, both); } _normalize = mixed; }, /** * Debounce a function * * @param {function} fn Function to be called * @param {integer} freq Call frequency in mS * @return {function} Wrapped function */ debounce: function ( fn, timeout ) { var timer; return function () { var that = this; var args = arguments; clearTimeout(timer); timer = setTimeout( function () { fn.apply(that, args); }, timeout || 250 ); }; }, /** * Throttle the calls to a function. Arguments and context are maintained * for the throttled function. * * @param {function} fn Function to be called * @param {integer} freq Call frequency in mS * @return {function} Wrapped function */ throttle: function ( fn, freq ) { var frequency = freq !== undefined ? freq : 200, last, timer; return function () { var that = this, now = +new Date(), args = arguments; if ( last && now < last + frequency ) { clearTimeout( timer ); timer = setTimeout( function () { last = undefined; fn.apply( that, args ); }, frequency ); } else { last = now; fn.apply( that, args ); } }; }, /** * Escape a string such that it can be used in a regular expression * * @param {string} val string to escape * @returns {string} escaped string */ escapeRegex: function ( val ) { return val.replace( _re_escape_regex, '\\$1' ); }, /** * Create a function that will write to a nested object or array * @param {*} source JSON notation string * @returns Write function */ set: function ( source ) { if ( $.isPlainObject( source ) ) { /* Unlike get, only the underscore (global) option is used for for * setting data since we don't know the type here. This is why an object * option is not documented for `mData` (which is read/write), but it is * for `mRender` which is read only. */ return DataTable.util.set( source._ ); } else if ( source === null ) { // Nothing to do when the data source is null return function () {}; } else if ( typeof source === 'function' ) { return function (data, val, meta) { source( data, 'set', val, meta ); }; } else if ( typeof source === 'string' && (source.indexOf('.') !== -1 || source.indexOf('[') !== -1 || source.indexOf('(') !== -1) ) { // Like the get, we need to get data from a nested object var setData = function (data, val, src) { var a = _fnSplitObjNotation( src ), b; var aLast = a[a.length-1]; var arrayNotation, funcNotation, o, innerSrc; for ( var i=0, iLen=a.length-1 ; i<iLen ; i++ ) { // Protect against prototype pollution if (a[i] === '__proto__' || a[i] === 'constructor') { throw new Error('Cannot set prototype values'); } // Check if we are dealing with an array notation request arrayNotation = a[i].match(__reArray); funcNotation = a[i].match(__reFn); if ( arrayNotation ) { a[i] = a[i].replace(__reArray, ''); data[ a[i] ] = []; // Get the remainder of the nested object to set so we can recurse b = a.slice(); b.splice( 0, i+1 ); innerSrc = b.join('.'); // Traverse each entry in the array setting the properties requested if ( Array.isArray( val ) ) { for ( var j=0, jLen=val.length ; j<jLen ; j++ ) { o = {}; setData( o, val[j], innerSrc ); data[ a[i] ].push( o ); } } else { // We've been asked to save data to an array, but it // isn't array data to be saved. Best that can be done // is to just save the value. data[ a[i] ] = val; } // The inner call to setData has already traversed through the remainder // of the source and has set the data, thus we can exit here return; } else if ( funcNotation ) { // Function call a[i] = a[i].replace(__reFn, ''); data = data[ a[i] ]( val ); } // If the nested object doesn't currently exist - since we are // trying to set the value - create it if ( data[ a[i] ] === null || data[ a[i] ] === undefined ) { data[ a[i] ] = {}; } data = data[ a[i] ]; } // Last item in the input - i.e, the actual set if ( aLast.match(__reFn ) ) { // Function call data = data[ aLast.replace(__reFn, '') ]( val ); } else { // If array notation is used, we just want to strip it and use the property name // and assign the value. If it isn't used, then we get the result we want anyway data[ aLast.replace(__reArray, '') ] = val; } }; return function (data, val) { // meta is also passed in, but not used return setData( data, val, source ); }; } else { // Array or flat object mapping return function (data, val) { // meta is also passed in, but not used data[source] = val; }; } }, /** * Create a function that will read nested objects from arrays, based on JSON notation * @param {*} source JSON notation string * @returns Value read */ get: function ( source ) { if ( $.isPlainObject( source ) ) { // Build an object of get functions, and wrap them in a single call var o = {}; $.each( source, function (key, val) { if ( val ) { o[key] = DataTable.util.get( val ); } } ); return function (data, type, row, meta) { var t = o[type] || o._; return t !== undefined ? t(data, type, row, meta) : data; }; } else if ( source === null ) { // Give an empty string for rendering / sorting etc return function (data) { // type, row and meta also passed, but not used return data; }; } else if ( typeof source === 'function' ) { return function (data, type, row, meta) { return source( data, type, row, meta ); }; } else if ( typeof source === 'string' && (source.indexOf('.') !== -1 || source.indexOf('[') !== -1 || source.indexOf('(') !== -1) ) { /* If there is a . in the source string then the data source is in a * nested object so we loop over the data for each level to get the next * level down. On each loop we test for undefined, and if found immediately * return. This allows entire objects to be missing and sDefaultContent to * be used if defined, rather than throwing an error */ var fetchData = function (data, type, src) { var arrayNotation, funcNotation, out, innerSrc; if ( src !== "" ) { var a = _fnSplitObjNotation( src ); for ( var i=0, iLen=a.length ; i<iLen ; i++ ) { // Check if we are dealing with special notation arrayNotation = a[i].match(__reArray); funcNotation = a[i].match(__reFn); if ( arrayNotation ) { // Array notation a[i] = a[i].replace(__reArray, ''); // Condition allows simply [] to be passed in if ( a[i] !== "" ) { data = data[ a[i] ]; } out = []; // Get the remainder of the nested object to get a.splice( 0, i+1 ); innerSrc = a.join('.'); // Traverse each entry in the array getting the properties requested if ( Array.isArray( data ) ) { for ( var j=0, jLen=data.length ; j<jLen ; j++ ) { out.push( fetchData( data[j], type, innerSrc ) ); } } // If a string is given in between the array notation indicators, that // is used to join the strings together, otherwise an array is returned var join = arrayNotation[0].substring(1, arrayNotation[0].length-1); data = (join==="") ? out : out.join(join); // The inner call to fetchData has already traversed through the remainder // of the source requested, so we exit from the loop break; } else if ( funcNotation ) { // Function call a[i] = a[i].replace(__reFn, ''); data = data[ a[i] ](); continue; } if (data === null || data[ a[i] ] === null) { return null; } else if ( data === undefined || data[ a[i] ] === undefined ) { return undefined; } data = data[ a[i] ]; } } return data; }; return function (data, type) { // row and meta also passed, but not used return fetchData( data, type, source ); }; } else { // Array or flat object mapping return function (data) { // row and meta also passed, but not used return data[source]; }; } }, stripHtml: function (mixed) { var type = typeof mixed; if (type === 'function') { _stripHtml = mixed; return; } else if (type === 'string') { return _stripHtml(mixed); } return mixed; }, escapeHtml: function (mixed) { var type = typeof mixed; if (type === 'function') { _escapeHtml = mixed; return; } else if (type === 'string' || Array.isArray(mixed)) { return _escapeHtml(mixed); } return mixed; }, unique: _unique }; /** * Create a mapping object that allows camel case parameters to be looked up * for their Hungarian counterparts. The mapping is stored in a private * parameter called `_hungarianMap` which can be accessed on the source object. * @param {object} o * @memberof DataTable#oApi */ function _fnHungarianMap ( o ) { var hungarian = 'a aa ai ao as b fn i m o s ', match, newKey, map = {}; $.each( o, function (key) { match = key.match(/^([^A-Z]+?)([A-Z])/); if ( match && hungarian.indexOf(match[1]+' ') !== -1 ) { newKey = key.replace( match[0], match[2].toLowerCase() ); map[ newKey ] = key; if ( match[1] === 'o' ) { _fnHungarianMap( o[key] ); } } } ); o._hungarianMap = map; } /** * Convert from camel case parameters to Hungarian, based on a Hungarian map * created by _fnHungarianMap. * @param {object} src The model object which holds all parameters that can be * mapped. * @param {object} user The object to convert from camel case to Hungarian. * @param {boolean} force When set to `true`, properties which already have a * Hungarian value in the `user` object will be overwritten. Otherwise they * won't be. * @memberof DataTable#oApi */ function _fnCamelToHungarian ( src, user, force ) { if ( ! src._hungarianMap ) { _fnHungarianMap( src ); } var hungarianKey; $.each( user, function (key) { hungarianKey = src._hungarianMap[ key ]; if ( hungarianKey !== undefined && (force || user[hungarianKey] === undefined) ) { // For objects, we need to buzz down into the object to copy parameters if ( hungarianKey.charAt(0) === 'o' ) { // Copy the camelCase options over to the hungarian if ( ! user[ hungarianKey ] ) { user[ hungarianKey ] = {}; } $.extend( true, user[hungarianKey], user[key] ); _fnCamelToHungarian( src[hungarianKey], user[hungarianKey], force ); } else { user[hungarianKey] = user[ key ]; } } } ); } /** * Map one parameter onto another * @param {object} o Object to map * @param {*} knew The new parameter name * @param {*} old The old parameter name */ var _fnCompatMap = function ( o, knew, old ) { if ( o[ knew ] !== undefined ) { o[ old ] = o[ knew ]; } }; /** * Provide backwards compatibility for the main DT options. Note that the new * options are mapped onto the old parameters, so this is an external interface * change only. * @param {object} init Object to map */ function _fnCompatOpts ( init ) { _fnCompatMap( init, 'ordering', 'bSort' ); _fnCompatMap( init, 'orderMulti', 'bSortMulti' ); _fnCompatMap( init, 'orderClasses', 'bSortClasses' ); _fnCompatMap( init, 'orderCellsTop', 'bSortCellsTop' ); _fnCompatMap( init, 'order', 'aaSorting' ); _fnCompatMap( init, 'orderFixed', 'aaSortingFixed' ); _fnCompatMap( init, 'paging', 'bPaginate' ); _fnCompatMap( init, 'pagingType', 'sPaginationType' ); _fnCompatMap( init, 'pageLength', 'iDisplayLength' ); _fnCompatMap( init, 'searching', 'bFilter' ); // Boolean initialisation of x-scrolling if ( typeof init.sScrollX === 'boolean' ) { init.sScrollX = init.sScrollX ? '100%' : ''; } if ( typeof init.scrollX === 'boolean' ) { init.scrollX = init.scrollX ? '100%' : ''; } // Column search objects are in an array, so it needs to be converted // element by element var searchCols = init.aoSearchCols; if ( searchCols ) { for ( var i=0, ien=searchCols.length ; i<ien ; i++ ) { if ( searchCols[i] ) { _fnCamelToHungarian( DataTable.models.oSearch, searchCols[i] ); } } } // Enable search delay if server-side processing is enabled if (init.serverSide && ! init.searchDelay) { init.searchDelay = 400; } } /** * Provide backwards compatibility for column options. Note that the new options * are mapped onto the old parameters, so this is an external interface change * only. * @param {object} init Object to map */ function _fnCompatCols ( init ) { _fnCompatMap( init, 'orderable', 'bSortable' ); _fnCompatMap( init, 'orderData', 'aDataSort' ); _fnCompatMap( init, 'orderSequence', 'asSorting' ); _fnCompatMap( init, 'orderDataType', 'sortDataType' ); // orderData can be given as an integer var dataSort = init.aDataSort; if ( typeof dataSort === 'number' && ! Array.isArray( dataSort ) ) { init.aDataSort = [ dataSort ]; } } /** * Browser feature detection for capabilities, quirks * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnBrowserDetect( settings ) { // We don't need to do this every time DataTables is constructed, the values // calculated are specific to the browser and OS configuration which we // don't expect to change between initialisations if ( ! DataTable.__browser ) { var browser = {}; DataTable.__browser = browser; // Scrolling feature / quirks detection var n = $('<div/>') .css( { position: 'fixed', top: 0, left: -1 * window.pageXOffset, // allow for scrolling height: 1, width: 1, overflow: 'hidden' } ) .append( $('<div/>') .css( { position: 'absolute', top: 1, left: 1, width: 100, overflow: 'scroll' } ) .append( $('<div/>') .css( { width: '100%', height: 10 } ) ) ) .appendTo( 'body' ); var outer = n.children(); var inner = outer.children(); // Get scrollbar width browser.barWidth = outer[0].offsetWidth - outer[0].clientWidth; // In rtl text layout, some browsers (most, but not all) will place the // scrollbar on the left, rather than the right. browser.bScrollbarLeft = Math.round( inner.offset().left ) !== 1; n.remove(); } $.extend( settings.oBrowser, DataTable.__browser ); settings.oScroll.iBarWidth = DataTable.__browser.barWidth; } /** * Add a column to the list used for the table with default values * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnAddColumn( oSettings ) { // Add column to aoColumns array var oDefaults = DataTable.defaults.column; var iCol = oSettings.aoColumns.length; var oCol = $.extend( {}, DataTable.models.oColumn, oDefaults, { "aDataSort": oDefaults.aDataSort ? oDefaults.aDataSort : [iCol], "mData": oDefaults.mData ? oDefaults.mData : iCol, idx: iCol, searchFixed: {}, colEl: $('<col>').attr('data-dt-column', iCol) } ); oSettings.aoColumns.push( oCol ); // Add search object for column specific search. Note that the `searchCols[ iCol ]` // passed into extend can be undefined. This allows the user to give a default // with only some of the parameters defined, and also not give a default var searchCols = oSettings.aoPreSearchCols; searchCols[ iCol ] = $.extend( {}, DataTable.models.oSearch, searchCols[ iCol ] ); } /** * Apply options for a column * @param {object} oSettings dataTables settings object * @param {int} iCol column index to consider * @param {object} oOptions object with sType, bVisible and bSearchable etc * @memberof DataTable#oApi */ function _fnColumnOptions( oSettings, iCol, oOptions ) { var oCol = oSettings.aoColumns[ iCol ]; /* User specified column options */ if ( oOptions !== undefined && oOptions !== null ) { // Backwards compatibility _fnCompatCols( oOptions ); // Map camel case parameters to their Hungarian counterparts _fnCamelToHungarian( DataTable.defaults.column, oOptions, true ); /* Backwards compatibility for mDataProp */ if ( oOptions.mDataProp !== undefined && !oOptions.mData ) { oOptions.mData = oOptions.mDataProp; } if ( oOptions.sType ) { oCol._sManualType = oOptions.sType; } // `class` is a reserved word in Javascript, so we need to provide // the ability to use a valid name for the camel case input if ( oOptions.className && ! oOptions.sClass ) { oOptions.sClass = oOptions.className; } var origClass = oCol.sClass; $.extend( oCol, oOptions ); _fnMap( oCol, oOptions, "sWidth", "sWidthOrig" ); // Merge class from previously defined classes with this one, rather than just // overwriting it in the extend above if (origClass !== oCol.sClass) { oCol.sClass = origClass + ' ' + oCol.sClass; } /* iDataSort to be applied (backwards compatibility), but aDataSort will take * priority if defined */ if ( oOptions.iDataSort !== undefined ) { oCol.aDataSort = [ oOptions.iDataSort ]; } _fnMap( oCol, oOptions, "aDataSort" ); } /* Cache the data get and set functions for speed */ var mDataSrc = oCol.mData; var mData = _fnGetObjectDataFn( mDataSrc ); // The `render` option can be given as an array to access the helper rendering methods. // The first element is the rendering method to use, the rest are the parameters to pass if ( oCol.mRender && Array.isArray( oCol.mRender ) ) { var copy = oCol.mRender.slice(); var name = copy.shift(); oCol.mRender = DataTable.render[name].apply(window, copy); } oCol._render = oCol.mRender ? _fnGetObjectDataFn( oCol.mRender ) : null; var attrTest = function( src ) { return typeof src === 'string' && src.indexOf('@') !== -1; }; oCol._bAttrSrc = $.isPlainObject( mDataSrc ) && ( attrTest(mDataSrc.sort) || attrTest(mDataSrc.type) || attrTest(mDataSrc.filter) ); oCol._setter = null; oCol.fnGetData = function (rowData, type, meta) { var innerData = mData( rowData, type, undefined, meta ); return oCol._render && type ? oCol._render( innerData, type, rowData, meta ) : innerData; }; oCol.fnSetData = function ( rowData, val, meta ) { return _fnSetObjectDataFn( mDataSrc )( rowData, val, meta ); }; // Indicate if DataTables should read DOM data as an object or array // Used in _fnGetRowElements if ( typeof mDataSrc !== 'number' && ! oCol._isArrayHost ) { oSettings._rowReadObject = true; } /* Feature sorting overrides column specific when off */ if ( !oSettings.oFeatures.bSort ) { oCol.bSortable = false; } } /** * Adjust the table column widths for new data. Note: you would probably want to * do a redraw after calling this function! * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnAdjustColumnSizing ( settings ) { _fnCalculateColumnWidths( settings ); _fnColumnSizes( settings ); var scroll = settings.oScroll; if ( scroll.sY !== '' || scroll.sX !== '') { _fnScrollDraw( settings ); } _fnCallbackFire( settings, null, 'column-sizing', [settings] ); } /** * Apply column sizes * * @param {*} settings DataTables settings object */ function _fnColumnSizes ( settings ) { var cols = settings.aoColumns; for (var i=0 ; i<cols.length ; i++) { var width = _fnColumnsSumWidth(settings, [i], false, false); cols[i].colEl.css('width', width); } } /** * Convert the index of a visible column to the index in the data array (take account * of hidden columns) * @param {object} oSettings dataTables settings object * @param {int} iMatch Visible column index to lookup * @returns {int} i the data index * @memberof DataTable#oApi */ function _fnVisibleToColumnIndex( oSettings, iMatch ) { var aiVis = _fnGetColumns( oSettings, 'bVisible' ); return typeof aiVis[iMatch] === 'number' ? aiVis[iMatch] : null; } /** * Convert the index of an index in the data array and convert it to the visible * column index (take account of hidden columns) * @param {int} iMatch Column index to lookup * @param {object} oSettings dataTables settings object * @returns {int} i the data index * @memberof DataTable#oApi */ function _fnColumnIndexToVisible( oSettings, iMatch ) { var aiVis = _fnGetColumns( oSettings, 'bVisible' ); var iPos = aiVis.indexOf(iMatch); return iPos !== -1 ? iPos : null; } /** * Get the number of visible columns * @param {object} oSettings dataTables settings object * @returns {int} i the number of visible columns * @memberof DataTable#oApi */ function _fnVisbleColumns( settings ) { var layout = settings.aoHeader; var columns = settings.aoColumns; var vis = 0; if ( layout.length ) { for ( var i=0, ien=layout[0].length ; i<ien ; i++ ) { if ( columns[i].bVisible && $(layout[0][i].cell).css('display') !== 'none' ) { vis++; } } } return vis; } /** * Get an array of column indexes that match a given property * @param {object} oSettings dataTables settings object * @param {string} sParam Parameter in aoColumns to look for - typically * bVisible or bSearchable * @returns {array} Array of indexes with matched properties * @memberof DataTable#oApi */ function _fnGetColumns( oSettings, sParam ) { var a = []; oSettings.aoColumns.map( function(val, i) { if ( val[sParam] ) { a.push( i ); } } ); return a; } /** * Calculate the 'type' of a column * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnColumnTypes ( settings ) { var columns = settings.aoColumns; var data = settings.aoData; var types = DataTable.ext.type.detect; var i, ien, j, jen, k, ken; var col, detectedType, cache; // For each column, spin over the for ( i=0, ien=columns.length ; i<ien ; i++ ) { col = columns[i]; cache = []; if ( ! col.sType && col._sManualType ) { col.sType = col._sManualType; } else if ( ! col.sType ) { for ( j=0, jen=types.length ; j<jen ; j++ ) { for ( k=0, ken=data.length ; k<ken ; k++ ) { if (! data[k]) { continue; } // Use a cache array so we only need to get the type data // from the formatter once (when using multiple detectors) if ( cache[k] === undefined ) { cache[k] = _fnGetCellData( settings, k, i, 'type' ); } detectedType = types[j]( cache[k], settings ); // If null, then this type can't apply to this column, so // rather than testing all cells, break out. There is an // exception for the last type which is `html`. We need to // scan all rows since it is possible to mix string and HTML // types if ( ! detectedType && j !== types.length-2 ) { break; } // Only a single match is needed for html type since it is // bottom of the pile and very similar to string - but it // must not be empty if ( detectedType === 'html' && ! _empty(cache[k]) ) { break; } } // Type is valid for all data points in the column - use this // type if ( detectedType ) { col.sType = detectedType; break; } } // Fall back - if no type was detected, always use string if ( ! col.sType ) { col.sType = 'string'; } } // Set class names for header / footer for auto type classes var autoClass = _ext.type.className[col.sType]; if (autoClass) { _columnAutoClass(settings.aoHeader, i, autoClass); _columnAutoClass(settings.aoFooter, i, autoClass); } var renderer = _ext.type.render[col.sType]; // This can only happen once! There is no way to remover // a renderer. After the first time the renderer has // already been set so createTr will run the renderer itself. if (renderer && ! col._render) { col._render = DataTable.util.get(renderer); _columnAutoRender(settings, i); } } } /** * Apply an auto detected renderer to data which doesn't yet have * a renderer */ function _columnAutoRender(settings, colIdx) { var data = settings.aoData; for (var i=0 ; i<data.length ; i++) { if (data[i].nTr) { // We have to update the display here since there is no // invalidation check for the data var display = _fnGetCellData( settings, i, colIdx, 'display' ); data[i].displayData[colIdx] = display; _fnWriteCell(data[i].anCells[colIdx], display); // No need to update sort / filter data since it has // been invalidated and will be re-read with the // renderer now applied } } } /** * Apply a class name to a column's header cells */ function _columnAutoClass(container, colIdx, className) { container.forEach(function (row) { if (row[colIdx] && row[colIdx].unique) { _addClass(row[colIdx].cell, className); } }); } /** * Take the column definitions and static columns arrays and calculate how * they relate to column indexes. The callback function will then apply the * definition found for a column to a suitable configuration object. * @param {object} oSettings dataTables settings object * @param {array} aoColDefs The aoColumnDefs array that is to be applied * @param {array} aoCols The aoColumns array that defines columns individually * @param {array} headerLayout Layout for header as it was loaded * @param {function} fn Callback function - takes two parameters, the calculated * column index and the definition for that column. * @memberof DataTable#oApi */ function _fnApplyColumnDefs( oSettings, aoColDefs, aoCols, headerLayout, fn ) { var i, iLen, j, jLen, k, kLen, def; var columns = oSettings.aoColumns; if ( aoCols ) { for ( i=0, iLen=aoCols.length ; i<iLen ; i++ ) { if (aoCols[i] && aoCols[i].name) { columns[i].sName = aoCols[i].name; } } } // Column definitions with aTargets if ( aoColDefs ) { /* Loop over the definitions array - loop in reverse so first instance has priority */ for ( i=aoColDefs.length-1 ; i>=0 ; i-- ) { def = aoColDefs[i]; /* Each definition can target multiple columns, as it is an array */ var aTargets = def.target !== undefined ? def.target : def.targets !== undefined ? def.targets : def.aTargets; if ( ! Array.isArray( aTargets ) ) { aTargets = [ aTargets ]; } for ( j=0, jLen=aTargets.length ; j<jLen ; j++ ) { var target = aTargets[j]; if ( typeof target === 'number' && target >= 0 ) { /* Add columns that we don't yet know about */ while( columns.length <= target ) { _fnAddColumn( oSettings ); } /* Integer, basic index */ fn( target, def ); } else if ( typeof target === 'number' && target < 0 ) { /* Negative integer, right to left column counting */ fn( columns.length+target, def ); } else if ( typeof target === 'string' ) { for ( k=0, kLen=columns.length ; k<kLen ; k++ ) { if (target === '_all') { // Apply to all columns fn( k, def ); } else if (target.indexOf(':name') !== -1) { // Column selector if (columns[k].sName === target.replace(':name', '')) { fn( k, def ); } } else { // Cell selector headerLayout.forEach(function (row) { if (row[k]) { var cell = $(row[k].cell); // Legacy support. Note that it means that we don't support // an element name selector only, since they are treated as // class names for 1.x compat. if (target.match(/^[a-z][\w-]*$/i)) { target = '.' + target; } if (cell.is( target )) { fn( k, def ); } } }); } } } } } } // Statically defined columns array if ( aoCols ) { for ( i=0, iLen=aoCols.length ; i<iLen ; i++ ) { fn( i, aoCols[i] ); } } } /** * Get the width for a given set of columns * * @param {*} settings DataTables settings object * @param {*} targets Columns - comma separated string or array of numbers * @param {*} original Use the original width (true) or calculated (false) * @param {*} incVisible Include visible columns (true) or not (false) * @returns Combined CSS value */ function _fnColumnsSumWidth( settings, targets, original, incVisible ) { if ( ! Array.isArray( targets ) ) { targets = _fnColumnsFromHeader( targets ); } var sum = 0; var unit; var columns = settings.aoColumns; for ( var i=0, ien=targets.length ; i<ien ; i++ ) { var column = columns[ targets[i] ]; var definedWidth = original ? column.sWidthOrig : column.sWidth; if ( ! incVisible && column.bVisible === false ) { continue; } if ( definedWidth === null || definedWidth === undefined ) { return null; // can't determine a defined width - browser defined } else if ( typeof definedWidth === 'number' ) { unit = 'px'; sum += definedWidth; } else { var matched = definedWidth.match(/([\d\.]+)([^\d]*)/); if ( matched ) { sum += matched[1] * 1; unit = matched.length === 3 ? matched[2] : 'px'; } } } return sum + unit; } function _fnColumnsFromHeader( cell ) { var attr = $(cell).closest('[data-dt-column]').attr('data-dt-column'); if ( ! attr ) { return []; } return attr.split(',').map( function (val) { return val * 1; } ); } /** * Add a data array to the table, creating DOM node etc. This is the parallel to * _fnGatherData, but for adding rows from a Javascript source, rather than a * DOM source. * @param {object} settings dataTables settings object * @param {array} data data array to be added * @param {node} [tr] TR element to add to the table - optional. If not given, * DataTables will create a row automatically * @param {array} [tds] Array of TD|TH elements for the row - must be given * if nTr is. * @returns {int} >=0 if successful (index of new aoData entry), -1 if failed * @memberof DataTable#oApi */ function _fnAddData ( settings, dataIn, tr, tds ) { /* Create the object for storing information about this new row */ var rowIdx = settings.aoData.length; var rowModel = $.extend( true, {}, DataTable.models.oRow, { src: tr ? 'dom' : 'data', idx: rowIdx } ); rowModel._aData = dataIn; settings.aoData.push( rowModel ); var columns = settings.aoColumns; for ( var i=0, iLen=columns.length ; i<iLen ; i++ ) { // Invalidate the column types as the new data needs to be revalidated columns[i].sType = null; } /* Add to the display array */ settings.aiDisplayMaster.push( rowIdx ); var id = settings.rowIdFn( dataIn ); if ( id !== undefined ) { settings.aIds[ id ] = rowModel; } /* Create the DOM information, or register it if already present */ if ( tr || ! settings.oFeatures.bDeferRender ) { _fnCreateTr( settings, rowIdx, tr, tds ); } return rowIdx; } /** * Add one or more TR elements to the table. Generally we'd expect to * use this for reading data from a DOM sourced table, but it could be * used for an TR element. Note that if a TR is given, it is used (i.e. * it is not cloned). * @param {object} settings dataTables settings object * @param {array|node|jQuery} trs The TR element(s) to add to the table * @returns {array} Array of indexes for the added rows * @memberof DataTable#oApi */ function _fnAddTr( settings, trs ) { var row; // Allow an individual node to be passed in if ( ! (trs instanceof $) ) { trs = $(trs); } return trs.map( function (i, el) { row = _fnGetRowElements( settings, el ); return _fnAddData( settings, row.data, el, row.cells ); } ); } /** * Get the data for a given cell from the internal cache, taking into account data mapping * @param {object} settings dataTables settings object * @param {int} rowIdx aoData row id * @param {int} colIdx Column index * @param {string} type data get type ('display', 'type' 'filter|search' 'sort|order') * @returns {*} Cell data * @memberof DataTable#oApi */ function _fnGetCellData( settings, rowIdx, colIdx, type ) { if (type === 'search') { type = 'filter'; } else if (type === 'order') { type = 'sort'; } var row = settings.aoData[rowIdx]; if (! row) { return undefined; } var draw = settings.iDraw; var col = settings.aoColumns[colIdx]; var rowData = row._aData; var defaultContent = col.sDefaultContent; var cellData = col.fnGetData( rowData, type, { settings: settings, row: rowIdx, col: colIdx } ); // Allow for a node being returned for non-display types if (type !== 'display' && cellData && typeof cellData === 'object' && cellData.nodeName) { cellData = cellData.innerHTML; } if ( cellData === undefined ) { if ( settings.iDrawError != draw && defaultContent === null ) { _fnLog( settings, 0, "Requested unknown parameter "+ (typeof col.mData=='function' ? '{function}' : "'"+col.mData+"'")+ " for row "+rowIdx+", column "+colIdx, 4 ); settings.iDrawError = draw; } return defaultContent; } // When the data source is null and a specific data type is requested (i.e. // not the original data), we can use default column data if ( (cellData === rowData || cellData === null) && defaultContent !== null && type !== undefined ) { cellData = defaultContent; } else if ( typeof cellData === 'function' ) { // If the data source is a function, then we run it and use the return, // executing in the scope of the data object (for instances) return cellData.call( rowData ); } if ( cellData === null && type === 'display' ) { return ''; } if ( type === 'filter' ) { var fomatters = DataTable.ext.type.search; if ( fomatters[ col.sType ] ) { cellData = fomatters[ col.sType ]( cellData ); } } return cellData; } /** * Set the value for a specific cell, into the internal data cache * @param {object} settings dataTables settings object * @param {int} rowIdx aoData row id * @param {int} colIdx Column index * @param {*} val Value to set * @memberof DataTable#oApi */ function _fnSetCellData( settings, rowIdx, colIdx, val ) { var col = settings.aoColumns[colIdx]; var rowData = settings.aoData[rowIdx]._aData; col.fnSetData( rowData, val, { settings: settings, row: rowIdx, col: colIdx } ); } /** * Write a value to a cell * @param {*} td Cell * @param {*} val Value */ function _fnWriteCell(td, val) { if (val && typeof val === 'object' && val.nodeName) { $(td) .empty() .append(val); } else { td.innerHTML = val; } } // Private variable that is used to match action syntax in the data property object var __reArray = /\[.*?\]$/; var __reFn = /\(\)$/; /** * Split string on periods, taking into account escaped periods * @param {string} str String to split * @return {array} Split string */ function _fnSplitObjNotation( str ) { var parts = str.match(/(\\.|[^.])+/g) || ['']; return parts.map( function ( s ) { return s.replace(/\\\./g, '.'); } ); } /** * Return a function that can be used to get data from a source object, taking * into account the ability to use nested objects as a source * @param {string|int|function} mSource The data source for the object * @returns {function} Data get function * @memberof DataTable#oApi */ var _fnGetObjectDataFn = DataTable.util.get; /** * Return a function that can be used to set data from a source object, taking * into account the ability to use nested objects as a source * @param {string|int|function} mSource The data source for the object * @returns {function} Data set function * @memberof DataTable#oApi */ var _fnSetObjectDataFn = DataTable.util.set; /** * Return an array with the full table data * @param {object} oSettings dataTables settings object * @returns array {array} aData Master data array * @memberof DataTable#oApi */ function _fnGetDataMaster ( settings ) { return _pluck( settings.aoData, '_aData' ); } /** * Nuke the table * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnClearTable( settings ) { settings.aoData.length = 0; settings.aiDisplayMaster.length = 0; settings.aiDisplay.length = 0; settings.aIds = {}; } /** * Mark cached data as invalid such that a re-read of the data will occur when * the cached data is next requested. Also update from the data source object. * * @param {object} settings DataTables settings object * @param {int} rowIdx Row index to invalidate * @param {string} [src] Source to invalidate from: undefined, 'auto', 'dom' * or 'data' * @param {int} [colIdx] Column index to invalidate. If undefined the whole * row will be invalidated * @memberof DataTable#oApi * * @todo For the modularisation of v1.11 this will need to become a callback, so * the sort and filter methods can subscribe to it. That will required * initialisation options for sorting, which is why it is not already baked in */ function _fnInvalidate( settings, rowIdx, src, colIdx ) { var row = settings.aoData[ rowIdx ]; var i, ien; // Remove the cached data for the row row._aSortData = null; row._aFilterData = null; row.displayData = null; // Are we reading last data from DOM or the data object? if ( src === 'dom' || ((! src || src === 'auto') && row.src === 'dom') ) { // Read the data from the DOM row._aData = _fnGetRowElements( settings, row, colIdx, colIdx === undefined ? undefined : row._aData ) .data; } else { // Reading from data object, update the DOM var cells = row.anCells; var display = _fnGetRowDisplay(settings, rowIdx); if ( cells ) { if ( colIdx !== undefined ) { _fnWriteCell(cells[colIdx], display[colIdx]); } else { for ( i=0, ien=cells.length ; i<ien ; i++ ) { _fnWriteCell(cells[i], display[i]); } } } } // Column specific invalidation var cols = settings.aoColumns; if ( colIdx !== undefined ) { // Type - the data might have changed cols[ colIdx ].sType = null; // Max length string. Its a fairly cheep recalculation, so not worth // something more complicated cols[ colIdx ].maxLenString = null; } else { for ( i=0, ien=cols.length ; i<ien ; i++ ) { cols[i].sType = null; cols[i].maxLenString = null; } // Update DataTables special `DT_*` attributes for the row _fnRowAttributes( settings, row ); } } /** * Build a data source object from an HTML row, reading the contents of the * cells that are in the row. * * @param {object} settings DataTables settings object * @param {node|object} TR element from which to read data or existing row * object from which to re-read the data from the cells * @param {int} [colIdx] Optional column index * @param {array|object} [d] Data source object. If `colIdx` is given then this * parameter should also be given and will be used to write the data into. * Only the column in question will be written * @returns {object} Object with two parameters: `data` the data read, in * document order, and `cells` and array of nodes (they can be useful to the * caller, so rather than needing a second traversal to get them, just return * them from here). * @memberof DataTable#oApi */ function _fnGetRowElements( settings, row, colIdx, d ) { var tds = [], td = row.firstChild, name, col, i=0, contents, columns = settings.aoColumns, objectRead = settings._rowReadObject; // Allow the data object to be passed in, or construct d = d !== undefined ? d : objectRead ? {} : []; var attr = function ( str, td ) { if ( typeof str === 'string' ) { var idx = str.indexOf('@'); if ( idx !== -1 ) { var attr = str.substring( idx+1 ); var setter = _fnSetObjectDataFn( str ); setter( d, td.getAttribute( attr ) ); } } }; // Read data from a cell and store into the data object var cellProcess = function ( cell ) { if ( colIdx === undefined || colIdx === i ) { col = columns[i]; contents = (cell.innerHTML).trim(); if ( col && col._bAttrSrc ) { var setter = _fnSetObjectDataFn( col.mData._ ); setter( d, contents ); attr( col.mData.sort, cell ); attr( col.mData.type, cell ); attr( col.mData.filter, cell ); } else { // Depending on the `data` option for the columns the data can // be read to either an object or an array. if ( objectRead ) { if ( ! col._setter ) { // Cache the setter function col._setter = _fnSetObjectDataFn( col.mData ); } col._setter( d, contents ); } else { d[i] = contents; } } } i++; }; if ( td ) { // `tr` element was passed in while ( td ) { name = td.nodeName.toUpperCase(); if ( name == "TD" || name == "TH" ) { cellProcess( td ); tds.push( td ); } td = td.nextSibling; } } else { // Existing row object passed in tds = row.anCells; for ( var j=0, jen=tds.length ; j<jen ; j++ ) { cellProcess( tds[j] ); } } // Read the ID from the DOM if present var rowNode = row.firstChild ? row : row.nTr; if ( rowNode ) { var id = rowNode.getAttribute( 'id' ); if ( id ) { _fnSetObjectDataFn( settings.rowId )( d, id ); } } return { data: d, cells: tds }; } /** * Render and cache a row's display data for the columns, if required * @returns */ function _fnGetRowDisplay (settings, rowIdx) { let rowModal = settings.aoData[rowIdx]; let columns = settings.aoColumns; if (! rowModal.displayData) { // Need to render and cache rowModal.displayData = []; for ( var colIdx=0, len=columns.length ; colIdx<len ; colIdx++ ) { rowModal.displayData.push( _fnGetCellData( settings, rowIdx, colIdx, 'display' ) ); } } return rowModal.displayData; } /** * Create a new TR element (and it's TD children) for a row * @param {object} oSettings dataTables settings object * @param {int} iRow Row to consider * @param {node} [nTrIn] TR element to add to the table - optional. If not given, * DataTables will create a row automatically * @param {array} [anTds] Array of TD|TH elements for the row - must be given * if nTr is. * @memberof DataTable#oApi */ function _fnCreateTr ( oSettings, iRow, nTrIn, anTds ) { var row = oSettings.aoData[iRow], rowData = row._aData, cells = [], nTr, nTd, oCol, i, iLen, create, trClass = oSettings.oClasses.tbody.row; if ( row.nTr === null ) { nTr = nTrIn || document.createElement('tr'); row.nTr = nTr; row.anCells = cells; _addClass(nTr, trClass); /* Use a private property on the node to allow reserve mapping from the node * to the aoData array for fast look up */ nTr._DT_RowIndex = iRow; /* Special parameters can be given by the data source to be used on the row */ _fnRowAttributes( oSettings, row ); /* Process each column */ for ( i=0, iLen=oSettings.aoColumns.length ; i<iLen ; i++ ) { oCol = oSettings.aoColumns[i]; create = nTrIn && anTds[i] ? false : true; nTd = create ? document.createElement( oCol.sCellType ) : anTds[i]; if (! nTd) { _fnLog( oSettings, 0, 'Incorrect column count', 18 ); } nTd._DT_CellIndex = { row: iRow, column: i }; cells.push( nTd ); var display = _fnGetRowDisplay(oSettings, iRow); // Need to create the HTML if new, or if a rendering function is defined if ( create || ( (oCol.mRender || oCol.mData !== i) && (!$.isPlainObject(oCol.mData) || oCol.mData._ !== i+'.display') ) ) { _fnWriteCell(nTd, display[i]); } // Visibility - add or remove as required if ( oCol.bVisible && create ) { nTr.appendChild( nTd ); } else if ( ! oCol.bVisible && ! create ) { nTd.parentNode.removeChild( nTd ); } if ( oCol.fnCreatedCell ) { oCol.fnCreatedCell.call( oSettings.oInstance, nTd, _fnGetCellData( oSettings, iRow, i ), rowData, iRow, i ); } } _fnCallbackFire( oSettings, 'aoRowCreatedCallback', 'row-created', [nTr, rowData, iRow, cells] ); } else { _addClass(row.nTr, trClass); } } /** * Add attributes to a row based on the special `DT_*` parameters in a data * source object. * @param {object} settings DataTables settings object * @param {object} DataTables row object for the row to be modified * @memberof DataTable#oApi */ function _fnRowAttributes( settings, row ) { var tr = row.nTr; var data = row._aData; if ( tr ) { var id = settings.rowIdFn( data ); if ( id ) { tr.id = id; } if ( data.DT_RowClass ) { // Remove any classes added by DT_RowClass before var a = data.DT_RowClass.split(' '); row.__rowc = row.__rowc ? _unique( row.__rowc.concat( a ) ) : a; $(tr) .removeClass( row.__rowc.join(' ') ) .addClass( data.DT_RowClass ); } if ( data.DT_RowAttr ) { $(tr).attr( data.DT_RowAttr ); } if ( data.DT_RowData ) { $(tr).data( data.DT_RowData ); } } } /** * Create the HTML header for the table * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnBuildHead( settings, side ) { var classes = settings.oClasses; var columns = settings.aoColumns; var i, ien, row; var target = side === 'header' ? settings.nTHead : settings.nTFoot; var titleProp = side === 'header' ? 'sTitle' : side; // Footer might be defined if (! target) { return; } // If no cells yet and we have content for them, then create if (side === 'header' || _pluck(settings.aoColumns, titleProp).join('')) { row = $('tr', target); // Add a row if needed if (! row.length) { row = $('<tr/>').appendTo(target) } // Add the number of cells needed to make up to the number of columns if (row.length === 1) { var cells = $('td, th', row); for ( i=cells.length, ien=columns.length ; i<ien ; i++ ) { $('<th/>') .html( columns[i][titleProp] || '' ) .appendTo( row ); } } } var detected = _fnDetectHeader( settings, target, true ); if (side === 'header') { settings.aoHeader = detected; } else { settings.aoFooter = detected; } // ARIA role for the rows $(target).children('tr').attr('role', 'row'); // Every cell needs to be passed through the renderer $(target).children('tr').children('th, td') .each( function () { _fnRenderer( settings, side )( settings, $(this), classes ); } ); } /** * Build a layout structure for a header or footer * * @param {*} settings DataTables settings * @param {*} source Source layout array * @param {*} incColumns What columns should be included * @returns Layout array */ function _fnHeaderLayout( settings, source, incColumns ) { var row, column, cell; var local = []; var structure = []; var columns = settings.aoColumns; var columnCount = columns.length; var rowspan, colspan; if ( ! source ) { return; } // Default is to work on only visible columns if ( ! incColumns ) { incColumns = _range(columnCount) .filter(function (idx) { return columns[idx].bVisible; }); } // Make a copy of the master layout array, but with only the columns we want for ( row=0 ; row<source.length ; row++ ) { // Remove any columns we haven't selected local[row] = source[row].slice().filter(function (cell, i) { return incColumns.includes(i); }); // Prep the structure array - it needs an element for each row structure.push( [] ); } for ( row=0 ; row<local.length ; row++ ) { for ( column=0 ; column<local[row].length ; column++ ) { rowspan = 1; colspan = 1; // Check to see if there is already a cell (row/colspan) covering our target // insert point. If there is, then there is nothing to do. if ( structure[row][column] === undefined ) { cell = local[row][column].cell; // Expand for rowspan while ( local[row+rowspan] !== undefined && local[row][column].cell == local[row+rowspan][column].cell ) { structure[row+rowspan][column] = null; rowspan++; } // And for colspan while ( local[row][column+colspan] !== undefined && local[row][column].cell == local[row][column+colspan].cell ) { // Which also needs to go over rows for ( var k=0 ; k<rowspan ; k++ ) { structure[row+k][column+colspan] = null; } colspan++; } var titleSpan = $('span.dt-column-title', cell); structure[row][column] = { cell: cell, colspan: colspan, rowspan: rowspan, title: titleSpan.length ? titleSpan.html() : $(cell).html() }; } } } return structure; } /** * Draw the header (or footer) element based on the column visibility states. * * @param object oSettings dataTables settings object * @param array aoSource Layout array from _fnDetectHeader * @memberof DataTable#oApi */ function _fnDrawHead( settings, source ) { var layout = _fnHeaderLayout(settings, source); var tr, n; for ( var row=0 ; row<source.length ; row++ ) { tr = source[row].row; // All cells are going to be replaced, so empty out the row // Can't use $().empty() as that kills event handlers if (tr) { while( (n = tr.firstChild) ) { tr.removeChild( n ); } } for ( var column=0 ; column<layout[row].length ; column++ ) { var point = layout[row][column]; if (point) { $(point.cell) .appendTo(tr) .attr('rowspan', point.rowspan) .attr('colspan', point.colspan); } } } } /** * Insert the required TR nodes into the table for display * @param {object} oSettings dataTables settings object * @param ajaxComplete true after ajax call to complete rendering * @memberof DataTable#oApi */ function _fnDraw( oSettings, ajaxComplete ) { // Allow for state saving and a custom start position _fnStart( oSettings ); /* Provide a pre-callback function which can be used to cancel the draw is false is returned */ var aPreDraw = _fnCallbackFire( oSettings, 'aoPreDrawCallback', 'preDraw', [oSettings] ); if ( aPreDraw.indexOf(false) !== -1 ) { _fnProcessingDisplay( oSettings, false ); return; } var anRows = []; var iRowCount = 0; var bServerSide = _fnDataSource( oSettings ) == 'ssp'; var aiDisplay = oSettings.aiDisplay; var iDisplayStart = oSettings._iDisplayStart; var iDisplayEnd = oSettings.fnDisplayEnd(); var columns = oSettings.aoColumns; var body = $(oSettings.nTBody); oSettings.bDrawing = true; /* Server-side processing draw intercept */ if ( !bServerSide ) { oSettings.iDraw++; } else if ( !oSettings.bDestroying && !ajaxComplete) { // Show loading message for server-side processing if (oSettings.iDraw === 0) { body.empty().append(_emptyRow(oSettings)); } _fnAjaxUpdate( oSettings ); return; } if ( aiDisplay.length !== 0 ) { var iStart = bServerSide ? 0 : iDisplayStart; var iEnd = bServerSide ? oSettings.aoData.length : iDisplayEnd; for ( var j=iStart ; j<iEnd ; j++ ) { var iDataIndex = aiDisplay[j]; var aoData = oSettings.aoData[ iDataIndex ]; if ( aoData.nTr === null ) { _fnCreateTr( oSettings, iDataIndex ); } var nRow = aoData.nTr; // Add various classes as needed for (var i=0 ; i<columns.length ; i++) { var col = columns[i]; var td = aoData.anCells[i]; _addClass(td, _ext.type.className[col.sType]); // auto class _addClass(td, col.sClass); // column class _addClass(td, oSettings.oClasses.tbody.cell); // all cells } // Row callback functions - might want to manipulate the row // iRowCount and j are not currently documented. Are they at all // useful? _fnCallbackFire( oSettings, 'aoRowCallback', null, [nRow, aoData._aData, iRowCount, j, iDataIndex] ); anRows.push( nRow ); iRowCount++; } } else { anRows[ 0 ] = _emptyRow(oSettings); } /* Header and footer callbacks */ _fnCallbackFire( oSettings, 'aoHeaderCallback', 'header', [ $(oSettings.nTHead).children('tr')[0], _fnGetDataMaster( oSettings ), iDisplayStart, iDisplayEnd, aiDisplay ] ); _fnCallbackFire( oSettings, 'aoFooterCallback', 'footer', [ $(oSettings.nTFoot).children('tr')[0], _fnGetDataMaster( oSettings ), iDisplayStart, iDisplayEnd, aiDisplay ] ); // replaceChildren is faster, but only became widespread in 2020, // so a fall back in jQuery is provided for older browsers. if (body[0].replaceChildren) { body[0].replaceChildren.apply(body[0], anRows); } else { body.children().detach(); body.append( $(anRows) ); } // Empty table needs a specific class $(oSettings.nTableWrapper).toggleClass('dt-empty-footer', $('tr', oSettings.nTFoot).length === 0); /* Call all required callback functions for the end of a draw */ _fnCallbackFire( oSettings, 'aoDrawCallback', 'draw', [oSettings], true ); /* Draw is complete, sorting and filtering must be as well */ oSettings.bSorted = false; oSettings.bFiltered = false; oSettings.bDrawing = false; } /** * Redraw the table - taking account of the various features which are enabled * @param {object} oSettings dataTables settings object * @param {boolean} [holdPosition] Keep the current paging position. By default * the paging is reset to the first page * @memberof DataTable#oApi */ function _fnReDraw( settings, holdPosition, recompute ) { var features = settings.oFeatures, sort = features.bSort, filter = features.bFilter; if (recompute === undefined || recompute === true) { if ( sort ) { _fnSort( settings ); } if ( filter ) { _fnFilterComplete( settings, settings.oPreviousSearch ); } else { // No filtering, so we want to just use the display master settings.aiDisplay = settings.aiDisplayMaster.slice(); } } if ( holdPosition !== true ) { settings._iDisplayStart = 0; } // Let any modules know about the draw hold position state (used by // scrolling internally) settings._drawHold = holdPosition; _fnDraw( settings ); settings._drawHold = false; } /* * Table is empty - create a row with an empty message in it */ function _emptyRow ( settings ) { var oLang = settings.oLanguage; var zero = oLang.sZeroRecords; var dataSrc = _fnDataSource( settings ); if ( (settings.iDraw < 1 && dataSrc === 'ssp') || (settings.iDraw <= 1 && dataSrc === 'ajax') ) { zero = oLang.sLoadingRecords; } else if ( oLang.sEmptyTable && settings.fnRecordsTotal() === 0 ) { zero = oLang.sEmptyTable; } return $( '<tr/>' ) .append( $('<td />', { 'colSpan': _fnVisbleColumns( settings ), 'class': settings.oClasses.empty.row } ).html( zero ) )[0]; } /** * Convert a `layout` object given by a user to the object structure needed * for the renderer. This is done twice, once for above and once for below * the table. Ordering must also be considered. * * @param {*} settings DataTables settings object * @param {*} layout Layout object to convert * @param {string} side `top` or `bottom` * @returns Converted array structure - one item for each row. */ function _layoutArray ( settings, layout, side ) { var groups = {}; // Combine into like groups (e.g. `top`, `top2`, etc) $.each( layout, function ( pos, val ) { if (val === null) { return; } var splitPos = pos.replace(/([A-Z])/g, ' $1').split(' '); if ( ! groups[ splitPos[0] ] ) { groups[ splitPos[0] ] = {}; } var align = splitPos.length === 1 ? 'full' : splitPos[1].toLowerCase(); var group = groups[ splitPos[0] ]; var groupRun = function (contents, innerVal) { // If it is an object, then there can be multiple features contained in it if ( $.isPlainObject( innerVal ) ) { Object.keys(innerVal).map(function (key) { contents.push( { feature: key, opts: innerVal[key] }); }); } else { contents.push(innerVal); } } // Transform to an object with a contents property if (! group[align] || ! group[align].contents) { group[align] = { contents: [] }; } // Allow for an array or just a single object if ( Array.isArray(val)) { for (var i=0 ; i<val.length ; i++) { groupRun(group[align].contents, val[i]); } } else { groupRun(group[ align ].contents, val); } // And make contents an array if ( ! Array.isArray( group[ align ].contents ) ) { group[ align ].contents = [ group[ align ].contents ]; } } ); var filtered = Object.keys(groups) .map( function ( pos ) { // Filter to only the side we need if ( pos.indexOf(side) !== 0 ) { return null; } return { name: pos, val: groups[pos] }; } ) .filter( function (item) { return item !== null; }); // Order by item identifier filtered.sort( function ( a, b ) { var order1 = a.name.replace(/[^0-9]/g, '') * 1; var order2 = b.name.replace(/[^0-9]/g, '') * 1; return order2 - order1; } ); if ( side === 'bottom' ) { filtered.reverse(); } // Split into rows var rows = []; for ( var i=0, ien=filtered.length ; i<ien ; i++ ) { if ( filtered[i].val.full ) { rows.push( { full: filtered[i].val.full } ); _layoutResolve( settings, rows[ rows.length - 1 ] ); delete filtered[i].val.full; } if ( Object.keys(filtered[i].val).length ) { rows.push( filtered[i].val ); _layoutResolve( settings, rows[ rows.length - 1 ] ); } } return rows; } /** * Convert the contents of a row's layout object to nodes that can be inserted * into the document by a renderer. Execute functions, look up plug-ins, etc. * * @param {*} settings DataTables settings object * @param {*} row Layout object for this row */ function _layoutResolve( settings, row ) { var getFeature = function (feature, opts) { if ( ! _ext.features[ feature ] ) { _fnLog( settings, 0, 'Unknown feature: '+ feature ); } return _ext.features[ feature ].apply( this, [settings, opts] ); }; var resolve = function ( item ) { var line = row[ item ].contents; for ( var i=0, ien=line.length ; i<ien ; i++ ) { if ( ! line[i] ) { continue; } else if ( typeof line[i] === 'string' ) { line[i] = getFeature( line[i], null ); } else if ( $.isPlainObject(line[i]) ) { // If it's an object, it just has feature and opts properties from // the transform in _layoutArray line[i] = getFeature(line[i].feature, line[i].opts); } else if ( typeof line[i].node === 'function' ) { line[i] = line[i].node( settings ); } else if ( typeof line[i] === 'function' ) { var inst = line[i]( settings ); line[i] = typeof inst.node === 'function' ? inst.node() : inst; } } }; $.each( row, function ( key ) { resolve( key ); } ); } /** * Add the options to the page HTML for the table * @param {object} settings DataTables settings object * @memberof DataTable#oApi */ function _fnAddOptionsHtml ( settings ) { var classes = settings.oClasses; var table = $(settings.nTable); // Wrapper div around everything DataTables controls var insert = $('<div/>') .attr({ id: settings.sTableId+'_wrapper', 'class': classes.container }) .insertBefore(table); settings.nTableWrapper = insert[0]; if (settings.sDom) { // Legacy _fnLayoutDom(settings, settings.sDom, insert); } else { var top = _layoutArray( settings, settings.layout, 'top' ); var bottom = _layoutArray( settings, settings.layout, 'bottom' ); var renderer = _fnRenderer( settings, 'layout' ); // Everything above - the renderer will actually insert the contents into the document top.forEach(function (item) { renderer( settings, insert, item ); }); // The table - always the center of attention renderer( settings, insert, { full: { table: true, contents: [ _fnFeatureHtmlTable(settings) ] } } ); // Everything below bottom.forEach(function (item) { renderer( settings, insert, item ); }); } // Processing floats on top, so it isn't an inserted feature _processingHtml( settings ); } /** * Draw the table with the legacy DOM property * @param {*} settings DT settings object * @param {*} dom DOM string * @param {*} insert Insert point */ function _fnLayoutDom( settings, dom, insert ) { var parts = dom.match(/(".*?")|('.*?')|./g); var featureNode, option, newNode, next, attr; for ( var i=0 ; i<parts.length ; i++ ) { featureNode = null; option = parts[i]; if ( option == '<' ) { // New container div newNode = $('<div/>'); // Check to see if we should append an id and/or a class name to the container next = parts[i+1]; if ( next[0] == "'" || next[0] == '"' ) { attr = next.replace(/['"]/g, ''); var id = '', className; /* The attribute can be in the format of "#id.class", "#id" or "class" This logic * breaks the string into parts and applies them as needed */ if ( attr.indexOf('.') != -1 ) { var split = attr.split('.'); id = split[0]; className = split[1]; } else if ( attr[0] == "#" ) { id = attr; } else { className = attr; } newNode .attr('id', id.substring(1)) .addClass(className); i++; // Move along the position array } insert.append( newNode ); insert = newNode; } else if ( option == '>' ) { // End container div insert = insert.parent(); } else if ( option == 't' ) { // Table featureNode = _fnFeatureHtmlTable( settings ); } else { DataTable.ext.feature.forEach(function(feature) { if ( option == feature.cFeature ) { featureNode = feature.fnInit( settings ); } }); } // Add to the display if ( featureNode ) { insert.append( featureNode ); } } } /** * Use the DOM source to create up an array of header cells. The idea here is to * create a layout grid (array) of rows x columns, which contains a reference * to the cell that that point in the grid (regardless of col/rowspan), such that * any column / row could be removed and the new grid constructed * @param {node} thead The header/footer element for the table * @returns {array} Calculated layout array * @memberof DataTable#oApi */ function _fnDetectHeader ( settings, thead, write ) { var columns = settings.aoColumns; var rows = $(thead).children('tr'); var row, cell; var i, k, l, iLen, shifted, column, colspan, rowspan; var isHeader = thead && thead.nodeName.toLowerCase() === 'thead'; var layout = []; var unique; var shift = function ( a, i, j ) { var k = a[i]; while ( k[j] ) { j++; } return j; }; // We know how many rows there are in the layout - so prep it for ( i=0, iLen=rows.length ; i<iLen ; i++ ) { layout.push( [] ); } for ( i=0, iLen=rows.length ; i<iLen ; i++ ) { row = rows[i]; column = 0; // For every cell in the row.. cell = row.firstChild; while ( cell ) { if ( cell.nodeName.toUpperCase() == 'TD' || cell.nodeName.toUpperCase() == 'TH' ) { var cols = []; // Get the col and rowspan attributes from the DOM and sanitise them colspan = cell.getAttribute('colspan') * 1; rowspan = cell.getAttribute('rowspan') * 1; colspan = (!colspan || colspan===0 || colspan===1) ? 1 : colspan; rowspan = (!rowspan || rowspan===0 || rowspan===1) ? 1 : rowspan; // There might be colspan cells already in this row, so shift our target // accordingly shifted = shift( layout, i, column ); // Cache calculation for unique columns unique = colspan === 1 ? true : false; // Perform header setup if ( write ) { if (unique) { // Allow column options to be set from HTML attributes _fnColumnOptions( settings, shifted, $(cell).data() ); // Get the width for the column. This can be defined from the // width attribute, style attribute or `columns.width` option var columnDef = columns[shifted]; var width = cell.getAttribute('width') || null; var t = cell.style.width.match(/width:\s*(\d+[pxem%]+)/); if ( t ) { width = t[1]; } columnDef.sWidthOrig = columnDef.sWidth || width; if (isHeader) { // Column title handling - can be user set, or read from the DOM // This happens before the render, so the original is still in place if ( columnDef.sTitle !== null && ! columnDef.autoTitle ) { cell.innerHTML = columnDef.sTitle; } if (! columnDef.sTitle && unique) { columnDef.sTitle = _stripHtml(cell.innerHTML); columnDef.autoTitle = true; } } else { // Footer specific operations if (columnDef.footer) { cell.innerHTML = columnDef.footer; } } // Fall back to the aria-label attribute on the table header if no ariaTitle is // provided. if (! columnDef.ariaTitle) { columnDef.ariaTitle = $(cell).attr("aria-label") || columnDef.sTitle; } // Column specific class names if ( columnDef.className ) { $(cell).addClass( columnDef.className ); } } // Wrap the column title so we can write to it in future if ( $('span.dt-column-title', cell).length === 0) { $('<span>') .addClass('dt-column-title') .append(cell.childNodes) .appendTo(cell); } if ( isHeader && $('span.dt-column-order', cell).length === 0) { $('<span>') .addClass('dt-column-order') .appendTo(cell); } } // If there is col / rowspan, copy the information into the layout grid for ( l=0 ; l<colspan ; l++ ) { for ( k=0 ; k<rowspan ; k++ ) { layout[i+k][shifted+l] = { cell: cell, unique: unique }; layout[i+k].row = row; } cols.push( shifted+l ); } // Assign an attribute so spanning cells can still be identified // as belonging to a column cell.setAttribute('data-dt-column', _unique(cols).join(',')); } cell = cell.nextSibling; } } return layout; } /** * Set the start position for draw * @param {object} oSettings dataTables settings object */ function _fnStart( oSettings ) { var bServerSide = _fnDataSource( oSettings ) == 'ssp'; var iInitDisplayStart = oSettings.iInitDisplayStart; // Check and see if we have an initial draw position from state saving if ( iInitDisplayStart !== undefined && iInitDisplayStart !== -1 ) { oSettings._iDisplayStart = bServerSide ? iInitDisplayStart : iInitDisplayStart >= oSettings.fnRecordsDisplay() ? 0 : iInitDisplayStart; oSettings.iInitDisplayStart = -1; } } /** * Create an Ajax call based on the table's settings, taking into account that * parameters can have multiple forms, and backwards compatibility. * * @param {object} oSettings dataTables settings object * @param {array} data Data to send to the server, required by * DataTables - may be augmented by developer callbacks * @param {function} fn Callback function to run when data is obtained */ function _fnBuildAjax( oSettings, data, fn ) { var ajaxData; var ajax = oSettings.ajax; var instance = oSettings.oInstance; var callback = function ( json ) { var status = oSettings.jqXHR ? oSettings.jqXHR.status : null; if ( json === null || (typeof status === 'number' && status == 204 ) ) { json = {}; _fnAjaxDataSrc( oSettings, json, [] ); } var error = json.error || json.sError; if ( error ) { _fnLog( oSettings, 0, error ); } oSettings.json = json; _fnCallbackFire( oSettings, null, 'xhr', [oSettings, json, oSettings.jqXHR], true ); fn( json ); }; if ( $.isPlainObject( ajax ) && ajax.data ) { ajaxData = ajax.data; var newData = typeof ajaxData === 'function' ? ajaxData( data, oSettings ) : // fn can manipulate data or return ajaxData; // an object object or array to merge // If the function returned something, use that alone data = typeof ajaxData === 'function' && newData ? newData : $.extend( true, data, newData ); // Remove the data property as we've resolved it already and don't want // jQuery to do it again (it is restored at the end of the function) delete ajax.data; } var baseAjax = { "url": typeof ajax === 'string' ? ajax : '', "data": data, "success": callback, "dataType": "json", "cache": false, "type": oSettings.sServerMethod, "error": function (xhr, error) { var ret = _fnCallbackFire( oSettings, null, 'xhr', [oSettings, null, oSettings.jqXHR], true ); if ( ret.indexOf(true) === -1 ) { if ( error == "parsererror" ) { _fnLog( oSettings, 0, 'Invalid JSON response', 1 ); } else if ( xhr.readyState === 4 ) { _fnLog( oSettings, 0, 'Ajax error', 7 ); } } _fnProcessingDisplay( oSettings, false ); } }; // If `ajax` option is an object, extend and override our default base if ( $.isPlainObject( ajax ) ) { $.extend( baseAjax, ajax ) } // Store the data submitted for the API oSettings.oAjaxData = data; // Allow plug-ins and external processes to modify the data _fnCallbackFire( oSettings, null, 'preXhr', [oSettings, data, baseAjax], true ); if ( typeof ajax === 'function' ) { // Is a function - let the caller define what needs to be done oSettings.jqXHR = ajax.call( instance, data, callback, oSettings ); } else if (ajax.url === '') { // No url, so don't load any data. Just apply an empty data array // to the object for the callback. var empty = {}; DataTable.util.set(ajax.dataSrc)(empty, []); callback(empty); } else { // Object to extend the base settings oSettings.jqXHR = $.ajax( baseAjax ); // Restore for next time around if ( ajaxData ) { ajax.data = ajaxData; } } } /** * Update the table using an Ajax call * @param {object} settings dataTables settings object * @returns {boolean} Block the table drawing or not * @memberof DataTable#oApi */ function _fnAjaxUpdate( settings ) { settings.iDraw++; _fnProcessingDisplay( settings, true ); _fnBuildAjax( settings, _fnAjaxParameters( settings ), function(json) { _fnAjaxUpdateDraw( settings, json ); } ); } /** * Build up the parameters in an object needed for a server-side processing * request. * @param {object} oSettings dataTables settings object * @returns {bool} block the table drawing or not * @memberof DataTable#oApi */ function _fnAjaxParameters( settings ) { var columns = settings.aoColumns, features = settings.oFeatures, preSearch = settings.oPreviousSearch, preColSearch = settings.aoPreSearchCols, colData = function ( idx, prop ) { return typeof columns[idx][prop] === 'function' ? 'function' : columns[idx][prop]; }; return { draw: settings.iDraw, columns: columns.map( function ( column, i ) { return { data: colData(i, 'mData'), name: column.sName, searchable: column.bSearchable, orderable: column.bSortable, search: { value: preColSearch[i].search, regex: preColSearch[i].regex, fixed: Object.keys(column.searchFixed).map( function(name) { return { name: name, term: column.searchFixed[name].toString() } }) } }; } ), order: _fnSortFlatten( settings ).map( function ( val ) { return { column: val.col, dir: val.dir, name: colData(val.col, 'sName') }; } ), start: settings._iDisplayStart, length: features.bPaginate ? settings._iDisplayLength : -1, search: { value: preSearch.search, regex: preSearch.regex, fixed: Object.keys(settings.searchFixed).map( function(name) { return { name: name, term: settings.searchFixed[name].toString() } }) } }; } /** * Data the data from the server (nuking the old) and redraw the table * @param {object} oSettings dataTables settings object * @param {object} json json data return from the server. * @param {string} json.sEcho Tracking flag for DataTables to match requests * @param {int} json.iTotalRecords Number of records in the data set, not accounting for filtering * @param {int} json.iTotalDisplayRecords Number of records in the data set, accounting for filtering * @param {array} json.aaData The data to display on this page * @param {string} [json.sColumns] Column ordering (sName, comma separated) * @memberof DataTable#oApi */ function _fnAjaxUpdateDraw ( settings, json ) { var data = _fnAjaxDataSrc(settings, json); var draw = _fnAjaxDataSrcParam(settings, 'draw', json); var recordsTotal = _fnAjaxDataSrcParam(settings, 'recordsTotal', json); var recordsFiltered = _fnAjaxDataSrcParam(settings, 'recordsFiltered', json); if ( draw !== undefined ) { // Protect against out of sequence returns if ( draw*1 < settings.iDraw ) { return; } settings.iDraw = draw * 1; } // No data in returned object, so rather than an array, we show an empty table if ( ! data ) { data = []; } _fnClearTable( settings ); settings._iRecordsTotal = parseInt(recordsTotal, 10); settings._iRecordsDisplay = parseInt(recordsFiltered, 10); for ( var i=0, ien=data.length ; i<ien ; i++ ) { _fnAddData( settings, data[i] ); } settings.aiDisplay = settings.aiDisplayMaster.slice(); _fnDraw( settings, true ); _fnInitComplete( settings ); _fnProcessingDisplay( settings, false ); } /** * Get the data from the JSON data source to use for drawing a table. Using * `_fnGetObjectDataFn` allows the data to be sourced from a property of the * source object, or from a processing function. * @param {object} settings dataTables settings object * @param {object} json Data source object / array from the server * @return {array} Array of data to use */ function _fnAjaxDataSrc ( settings, json, write ) { var dataProp = 'data'; if ($.isPlainObject( settings.ajax ) && settings.ajax.dataSrc !== undefined) { // Could in inside a `dataSrc` object, or not! var dataSrc = settings.ajax.dataSrc; // string, function and object are valid types if (typeof dataSrc === 'string' || typeof dataSrc === 'function') { dataProp = dataSrc; } else if (dataSrc.data !== undefined) { dataProp = dataSrc.data; } } if ( ! write ) { if ( dataProp === 'data' ) { // If the default, then we still want to support the old style, and safely ignore // it if possible return json.aaData || json[dataProp]; } return dataProp !== "" ? _fnGetObjectDataFn( dataProp )( json ) : json; } // set _fnSetObjectDataFn( dataProp )( json, write ); } /** * Very similar to _fnAjaxDataSrc, but for the other SSP properties * @param {*} settings DataTables settings object * @param {*} param Target parameter * @param {*} json JSON data * @returns Resolved value */ function _fnAjaxDataSrcParam (settings, param, json) { var dataSrc = $.isPlainObject( settings.ajax ) ? settings.ajax.dataSrc : null; if (dataSrc && dataSrc[param]) { // Get from custom location return _fnGetObjectDataFn( dataSrc[param] )( json ); } // else - Default behaviour var old = ''; // Legacy support if (param === 'draw') { old = 'sEcho'; } else if (param === 'recordsTotal') { old = 'iTotalRecords'; } else if (param === 'recordsFiltered') { old = 'iTotalDisplayRecords'; } return json[old] !== undefined ? json[old] : json[param]; } /** * Filter the table using both the global filter and column based filtering * @param {object} settings dataTables settings object * @param {object} input search information * @memberof DataTable#oApi */ function _fnFilterComplete ( settings, input ) { var columnsSearch = settings.aoPreSearchCols; // Resolve any column types that are unknown due to addition or invalidation // @todo As per sort - can this be moved into an event handler? _fnColumnTypes( settings ); // In server-side processing all filtering is done by the server, so no point hanging around here if ( _fnDataSource( settings ) != 'ssp' ) { // Check if any of the rows were invalidated _fnFilterData( settings ); // Start from the full data set settings.aiDisplay = settings.aiDisplayMaster.slice(); // Global filter first _fnFilter( settings.aiDisplay, settings, input.search, input ); $.each(settings.searchFixed, function (name, term) { _fnFilter(settings.aiDisplay, settings, term, {}); }); // Then individual column filters for ( var i=0 ; i<columnsSearch.length ; i++ ) { var col = columnsSearch[i]; _fnFilter( settings.aiDisplay, settings, col.search, col, i ); $.each(settings.aoColumns[i].searchFixed, function (name, term) { _fnFilter(settings.aiDisplay, settings, term, {}, i); }); } // And finally global filtering _fnFilterCustom( settings ); } // Tell the draw function we have been filtering settings.bFiltered = true; _fnCallbackFire( settings, null, 'search', [settings] ); } /** * Apply custom filtering functions * * This is legacy now that we have named functions, but it is widely used * from 1.x, so it is not yet deprecated. * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnFilterCustom( settings ) { var filters = DataTable.ext.search; var displayRows = settings.aiDisplay; var row, rowIdx; for ( var i=0, ien=filters.length ; i<ien ; i++ ) { var rows = []; // Loop over each row and see if it should be included for ( var j=0, jen=displayRows.length ; j<jen ; j++ ) { rowIdx = displayRows[ j ]; row = settings.aoData[ rowIdx ]; if ( filters[i]( settings, row._aFilterData, rowIdx, row._aData, j ) ) { rows.push( rowIdx ); } } // So the array reference doesn't break set the results into the // existing array displayRows.length = 0; displayRows.push.apply(displayRows, rows); } } /** * Filter the data table based on user input and draw the table */ function _fnFilter( searchRows, settings, input, options, column ) { if ( input === '' ) { return; } var i = 0; var matched = []; // Search term can be a function, regex or string - if a string we apply our // smart filtering regex (assuming the options require that) var searchFunc = typeof input === 'function' ? input : null; var rpSearch = input instanceof RegExp ? input : searchFunc ? null : _fnFilterCreateSearch( input, options ); // Then for each row, does the test pass. If not, lop the row from the array for (i=0 ; i<searchRows.length ; i++) { var row = settings.aoData[ searchRows[i] ]; var data = column === undefined ? row._sFilterRow : row._aFilterData[ column ]; if ( (searchFunc && searchFunc(data, row._aData, searchRows[i], column)) || (rpSearch && rpSearch.test(data)) ) { matched.push(searchRows[i]); } } // Mutate the searchRows array searchRows.length = matched.length; for (i=0 ; i<matched.length ; i++) { searchRows[i] = matched[i]; } } /** * Build a regular expression object suitable for searching a table * @param {string} sSearch string to search for * @param {bool} bRegex treat as a regular expression or not * @param {bool} bSmart perform smart filtering or not * @param {bool} bCaseInsensitive Do case insensitive matching or not * @returns {RegExp} constructed object * @memberof DataTable#oApi */ function _fnFilterCreateSearch( search, inOpts ) { var not = []; var options = $.extend({}, { boundary: false, caseInsensitive: true, exact: false, regex: false, smart: true }, inOpts); if (typeof search !== 'string') { search = search.toString(); } // Remove diacritics if normalize is set up to do so search = _normalize(search); if (options.exact) { return new RegExp( '^'+_fnEscapeRegex(search)+'$', options.caseInsensitive ? 'i' : '' ); } search = options.regex ? search : _fnEscapeRegex( search ); if ( options.smart ) { /* For smart filtering we want to allow the search to work regardless of * word order. We also want double quoted text to be preserved, so word * order is important - a la google. And a negative look around for * finding rows which don't contain a given string. * * So this is the sort of thing we want to generate: * * ^(?=.*?\bone\b)(?=.*?\btwo three\b)(?=.*?\bfour\b).*$ */ var parts = search.match( /!?["\u201C][^"\u201D]+["\u201D]|[^ ]+/g ) || ['']; var a = parts.map( function ( word ) { var negative = false; var m; // Determine if it is a "does not include" if ( word.charAt(0) === '!' ) { negative = true; word = word.substring(1); } // Strip the quotes from around matched phrases if ( word.charAt(0) === '"' ) { m = word.match( /^"(.*)"$/ ); word = m ? m[1] : word; } else if ( word.charAt(0) === '\u201C' ) { // Smart quote match (iPhone users) m = word.match( /^\u201C(.*)\u201D$/ ); word = m ? m[1] : word; } // For our "not" case, we need to modify the string that is // allowed to match at the end of the expression. if (negative) { if (word.length > 1) { not.push('(?!'+word+')'); } word = ''; } return word.replace(/"/g, ''); } ); var match = not.length ? not.join('') : ''; var boundary = options.boundary ? '\\b' : ''; search = '^(?=.*?'+boundary+a.join( ')(?=.*?'+boundary )+')('+match+'.)*$'; } return new RegExp( search, options.caseInsensitive ? 'i' : '' ); } /** * Escape a string such that it can be used in a regular expression * @param {string} sVal string to escape * @returns {string} escaped string * @memberof DataTable#oApi */ var _fnEscapeRegex = DataTable.util.escapeRegex; var __filter_div = $('<div>')[0]; var __filter_div_textContent = __filter_div.textContent !== undefined; // Update the filtering data for each row if needed (by invalidation or first run) function _fnFilterData ( settings ) { var columns = settings.aoColumns; var data = settings.aoData; var column; var j, jen, filterData, cellData, row; var wasInvalidated = false; for ( var rowIdx=0 ; rowIdx<data.length ; rowIdx++ ) { if (! data[rowIdx]) { continue; } row = data[rowIdx]; if ( ! row._aFilterData ) { filterData = []; for ( j=0, jen=columns.length ; j<jen ; j++ ) { column = columns[j]; if ( column.bSearchable ) { cellData = _fnGetCellData( settings, rowIdx, j, 'filter' ); // Search in DataTables is string based if ( cellData === null ) { cellData = ''; } if ( typeof cellData !== 'string' && cellData.toString ) { cellData = cellData.toString(); } } else { cellData = ''; } // If it looks like there is an HTML entity in the string, // attempt to decode it so sorting works as expected. Note that // we could use a single line of jQuery to do this, but the DOM // method used here is much faster https://jsperf.com/html-decode if ( cellData.indexOf && cellData.indexOf('&') !== -1 ) { __filter_div.innerHTML = cellData; cellData = __filter_div_textContent ? __filter_div.textContent : __filter_div.innerText; } if ( cellData.replace ) { cellData = cellData.replace(/[\r\n\u2028]/g, ''); } filterData.push( cellData ); } row._aFilterData = filterData; row._sFilterRow = filterData.join(' '); wasInvalidated = true; } } return wasInvalidated; } /** * Draw the table for the first time, adding all required features * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnInitialise ( settings ) { var i, iAjaxStart=settings.iInitDisplayStart; /* Ensure that the table data is fully initialised */ if ( ! settings.bInitialised ) { setTimeout( function(){ _fnInitialise( settings ); }, 200 ); return; } /* Build and draw the header / footer for the table */ _fnBuildHead( settings, 'header' ); _fnBuildHead( settings, 'footer' ); _fnDrawHead( settings, settings.aoHeader ); _fnDrawHead( settings, settings.aoFooter ); // Enable features _fnAddOptionsHtml( settings ); _fnSortInit( settings ); _colGroup( settings ); /* Okay to show that something is going on now */ _fnProcessingDisplay( settings, true ); _fnCallbackFire( settings, null, 'preInit', [settings], true ); // If there is default sorting required - let's do it. The sort function // will do the drawing for us. Otherwise we draw the table regardless of the // Ajax source - this allows the table to look initialised for Ajax sourcing // data (show 'loading' message possibly) _fnReDraw( settings ); var dataSrc = _fnDataSource( settings ); // Server-side processing init complete is done by _fnAjaxUpdateDraw if ( dataSrc != 'ssp' ) { // if there is an ajax source load the data if ( dataSrc == 'ajax' ) { _fnBuildAjax( settings, {}, function(json) { var aData = _fnAjaxDataSrc( settings, json ); // Got the data - add it to the table for ( i=0 ; i<aData.length ; i++ ) { _fnAddData( settings, aData[i] ); } // Reset the init display for cookie saving. We've already done // a filter, and therefore cleared it before. So we need to make // it appear 'fresh' settings.iInitDisplayStart = iAjaxStart; _fnReDraw( settings ); _fnProcessingDisplay( settings, false ); _fnInitComplete( settings ); }, settings ); } else { _fnInitComplete( settings ); _fnProcessingDisplay( settings, false ); } } } /** * Draw the table for the first time, adding all required features * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnInitComplete ( settings ) { if (settings._bInitComplete) { return; } var args = [settings, settings.json]; settings._bInitComplete = true; // Table is fully set up and we have data, so calculate the // column widths _fnAdjustColumnSizing( settings ); _fnCallbackFire( settings, null, 'plugin-init', args, true ); _fnCallbackFire( settings, 'aoInitComplete', 'init', args, true ); } function _fnLengthChange ( settings, val ) { var len = parseInt( val, 10 ); settings._iDisplayLength = len; _fnLengthOverflow( settings ); // Fire length change event _fnCallbackFire( settings, null, 'length', [settings, len] ); } /** * Alter the display settings to change the page * @param {object} settings DataTables settings object * @param {string|int} action Paging action to take: "first", "previous", * "next" or "last" or page number to jump to (integer) * @param [bool] redraw Automatically draw the update or not * @returns {bool} true page has changed, false - no change * @memberof DataTable#oApi */ function _fnPageChange ( settings, action, redraw ) { var start = settings._iDisplayStart, len = settings._iDisplayLength, records = settings.fnRecordsDisplay(); if ( records === 0 || len === -1 ) { start = 0; } else if ( typeof action === "number" ) { start = action * len; if ( start > records ) { start = 0; } } else if ( action == "first" ) { start = 0; } else if ( action == "previous" ) { start = len >= 0 ? start - len : 0; if ( start < 0 ) { start = 0; } } else if ( action == "next" ) { if ( start + len < records ) { start += len; } } else if ( action == "last" ) { start = Math.floor( (records-1) / len) * len; } else if ( action === 'ellipsis' ) { return; } else { _fnLog( settings, 0, "Unknown paging action: "+action, 5 ); } var changed = settings._iDisplayStart !== start; settings._iDisplayStart = start; _fnCallbackFire( settings, null, changed ? 'page' : 'page-nc', [settings] ); if ( changed && redraw ) { _fnDraw( settings ); } return changed; } /** * Generate the node required for the processing node * @param {object} settings DataTables settings object */ function _processingHtml ( settings ) { var table = settings.nTable; var scrolling = settings.oScroll.sX !== '' || settings.oScroll.sY !== ''; if ( settings.oFeatures.bProcessing ) { var n = $('<div/>', { 'id': settings.sTableId + '_processing', 'class': settings.oClasses.processing.container, 'role': 'status' } ) .html( settings.oLanguage.sProcessing ) .append('<div><div></div><div></div><div></div><div></div></div>'); // Different positioning depending on if scrolling is enabled or not if (scrolling) { n.prependTo( $('div.dt-scroll', settings.nTableWrapper) ); } else { n.insertBefore( table ); } $(table).on( 'processing.dt.DT', function (e, s, show) { n.css( 'display', show ? 'block' : 'none' ); } ); } } /** * Display or hide the processing indicator * @param {object} settings DataTables settings object * @param {bool} show Show the processing indicator (true) or not (false) */ function _fnProcessingDisplay ( settings, show ) { _fnCallbackFire( settings, null, 'processing', [settings, show] ); } /** * Add any control elements for the table - specifically scrolling * @param {object} settings dataTables settings object * @returns {node} Node to add to the DOM * @memberof DataTable#oApi */ function _fnFeatureHtmlTable ( settings ) { var table = $(settings.nTable); // Scrolling from here on in var scroll = settings.oScroll; if ( scroll.sX === '' && scroll.sY === '' ) { return settings.nTable; } var scrollX = scroll.sX; var scrollY = scroll.sY; var classes = settings.oClasses.scrolling; var caption = settings.captionNode; var captionSide = caption ? caption._captionSide : null; var headerClone = $( table[0].cloneNode(false) ); var footerClone = $( table[0].cloneNode(false) ); var footer = table.children('tfoot'); var _div = '<div/>'; var size = function ( s ) { return !s ? null : _fnStringToCss( s ); }; if ( ! footer.length ) { footer = null; } /* * The HTML structure that we want to generate in this function is: * div - scroller * div - scroll head * div - scroll head inner * table - scroll head table * thead - thead * div - scroll body * table - table (master table) * thead - thead clone for sizing * tbody - tbody * div - scroll foot * div - scroll foot inner * table - scroll foot table * tfoot - tfoot */ var scroller = $( _div, { 'class': classes.container } ) .append( $(_div, { 'class': classes.header.self } ) .css( { overflow: 'hidden', position: 'relative', border: 0, width: scrollX ? size(scrollX) : '100%' } ) .append( $(_div, { 'class': classes.header.inner } ) .css( { 'box-sizing': 'content-box', width: scroll.sXInner || '100%' } ) .append( headerClone .removeAttr('id') .css( 'margin-left', 0 ) .append( captionSide === 'top' ? caption : null ) .append( table.children('thead') ) ) ) ) .append( $(_div, { 'class': classes.body } ) .css( { position: 'relative', overflow: 'auto', width: size( scrollX ) } ) .append( table ) ); if ( footer ) { scroller.append( $(_div, { 'class': classes.footer.self } ) .css( { overflow: 'hidden', border: 0, width: scrollX ? size(scrollX) : '100%' } ) .append( $(_div, { 'class': classes.footer.inner } ) .append( footerClone .removeAttr('id') .css( 'margin-left', 0 ) .append( captionSide === 'bottom' ? caption : null ) .append( table.children('tfoot') ) ) ) ); } var children = scroller.children(); var scrollHead = children[0]; var scrollBody = children[1]; var scrollFoot = footer ? children[2] : null; // When the body is scrolled, then we also want to scroll the headers $(scrollBody).on( 'scroll.DT', function () { var scrollLeft = this.scrollLeft; scrollHead.scrollLeft = scrollLeft; if ( footer ) { scrollFoot.scrollLeft = scrollLeft; } } ); // When focus is put on the header cells, we might need to scroll the body $('th, td', scrollHead).on('focus', function () { var scrollLeft = scrollHead.scrollLeft; scrollBody.scrollLeft = scrollLeft; if ( footer ) { scrollBody.scrollLeft = scrollLeft; } }); $(scrollBody).css('max-height', scrollY); if (! scroll.bCollapse) { $(scrollBody).css('height', scrollY); } settings.nScrollHead = scrollHead; settings.nScrollBody = scrollBody; settings.nScrollFoot = scrollFoot; // On redraw - align columns settings.aoDrawCallback.push(_fnScrollDraw); return scroller[0]; } /** * Update the header, footer and body tables for resizing - i.e. column * alignment. * * Welcome to the most horrible function DataTables. The process that this * function follows is basically: * 1. Re-create the table inside the scrolling div * 2. Correct colgroup > col values if needed * 3. Copy colgroup > col over to header and footer * 4. Clean up * * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnScrollDraw ( settings ) { // Given that this is such a monster function, a lot of variables are use // to try and keep the minimised size as small as possible var scroll = settings.oScroll, barWidth = scroll.iBarWidth, divHeader = $(settings.nScrollHead), divHeaderInner = divHeader.children('div'), divHeaderTable = divHeaderInner.children('table'), divBodyEl = settings.nScrollBody, divBody = $(divBodyEl), divFooter = $(settings.nScrollFoot), divFooterInner = divFooter.children('div'), divFooterTable = divFooterInner.children('table'), header = $(settings.nTHead), table = $(settings.nTable), footer = settings.nTFoot && $('th, td', settings.nTFoot).length ? $(settings.nTFoot) : null, browser = settings.oBrowser, headerCopy, footerCopy; // If the scrollbar visibility has changed from the last draw, we need to // adjust the column sizes as the table width will have changed to account // for the scrollbar var scrollBarVis = divBodyEl.scrollHeight > divBodyEl.clientHeight; if ( settings.scrollBarVis !== scrollBarVis && settings.scrollBarVis !== undefined ) { settings.scrollBarVis = scrollBarVis; _fnAdjustColumnSizing( settings ); return; // adjust column sizing will call this function again } else { settings.scrollBarVis = scrollBarVis; } // 1. Re-create the table inside the scrolling div // Remove the old minimised thead and tfoot elements in the inner table table.children('thead, tfoot').remove(); // Clone the current header and footer elements and then place it into the inner table headerCopy = header.clone().prependTo( table ); headerCopy.find('th, td').removeAttr('tabindex'); headerCopy.find('[id]').removeAttr('id'); if ( footer ) { footerCopy = footer.clone().prependTo( table ); footerCopy.find('[id]').removeAttr('id'); } // 2. Correct colgroup > col values if needed // It is possible that the cell sizes are smaller than the content, so we need to // correct colgroup>col for such cases. This can happen if the auto width detection // uses a cell which has a longer string, but isn't the widest! For example // "Chief Executive Officer (CEO)" is the longest string in the demo, but // "Systems Administrator" is actually the widest string since it doesn't collapse. // Note the use of translating into a column index to get the `col` element. This // is because of Responsive which might remove `col` elements, knocking the alignment // of the indexes out. if (settings.aiDisplay.length) { // Get the column sizes from the first row in the table var colSizes = table.children('tbody').eq(0).children('tr').eq(0).children('th, td').map(function (vis) { return { idx: _fnVisibleToColumnIndex(settings, vis), width: $(this).outerWidth() } }); // Check against what the colgroup > col is set to and correct if needed for (var i=0 ; i<colSizes.length ; i++) { var colEl = settings.aoColumns[ colSizes[i].idx ].colEl[0]; var colWidth = colEl.style.width.replace('px', ''); if (colWidth !== colSizes[i].width) { colEl.style.width = colSizes[i].width + 'px'; } } } // 3. Copy the colgroup over to the header and footer divHeaderTable .find('colgroup') .remove(); divHeaderTable.append(settings.colgroup.clone()); if ( footer ) { divFooterTable .find('colgroup') .remove(); divFooterTable.append(settings.colgroup.clone()); } // "Hide" the header and footer that we used for the sizing. We need to keep // the content of the cell so that the width applied to the header and body // both match, but we want to hide it completely. $('th, td', headerCopy).each(function () { $(this.childNodes).wrapAll('<div class="dt-scroll-sizing">'); }); if ( footer ) { $('th, td', footerCopy).each(function () { $(this.childNodes).wrapAll('<div class="dt-scroll-sizing">'); }); } // 4. Clean up // Figure out if there are scrollbar present - if so then we need a the header and footer to // provide a bit more space to allow "overflow" scrolling (i.e. past the scrollbar) var isScrolling = Math.floor(table.height()) > divBodyEl.clientHeight || divBody.css('overflow-y') == "scroll"; var paddingSide = 'padding' + (browser.bScrollbarLeft ? 'Left' : 'Right' ); // Set the width's of the header and footer tables var outerWidth = table.outerWidth(); divHeaderTable.css('width', _fnStringToCss( outerWidth )); divHeaderInner .css('width', _fnStringToCss( outerWidth )) .css(paddingSide, isScrolling ? barWidth+"px" : "0px"); if ( footer ) { divFooterTable.css('width', _fnStringToCss( outerWidth )); divFooterInner .css('width', _fnStringToCss( outerWidth )) .css(paddingSide, isScrolling ? barWidth+"px" : "0px"); } // Correct DOM ordering for colgroup - comes before the thead table.children('colgroup').prependTo(table); // Adjust the position of the header in case we loose the y-scrollbar divBody.trigger('scroll'); // If sorting or filtering has occurred, jump the scrolling back to the top // only if we aren't holding the position if ( (settings.bSorted || settings.bFiltered) && ! settings._drawHold ) { divBodyEl.scrollTop = 0; } } /** * Calculate the width of columns for the table * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnCalculateColumnWidths ( settings ) { // Not interested in doing column width calculation if auto-width is disabled if (! settings.oFeatures.bAutoWidth) { return; } var table = settings.nTable, columns = settings.aoColumns, scroll = settings.oScroll, scrollY = scroll.sY, scrollX = scroll.sX, scrollXInner = scroll.sXInner, visibleColumns = _fnGetColumns( settings, 'bVisible' ), tableWidthAttr = table.getAttribute('width'), // from DOM element tableContainer = table.parentNode, i, column, columnIdx; var styleWidth = table.style.width; if ( styleWidth && styleWidth.indexOf('%') !== -1 ) { tableWidthAttr = styleWidth; } // Let plug-ins know that we are doing a recalc, in case they have changed any of the // visible columns their own way (e.g. Responsive uses display:none). _fnCallbackFire( settings, null, 'column-calc', {visible: visibleColumns}, false ); // Construct a single row, worst case, table with the widest // node in the data, assign any user defined widths, then insert it into // the DOM and allow the browser to do all the hard work of calculating // table widths var tmpTable = $(table.cloneNode()) .css( 'visibility', 'hidden' ) .removeAttr( 'id' ); // Clean up the table body tmpTable.append('<tbody>') var tr = $('<tr/>').appendTo( tmpTable.find('tbody') ); // Clone the table header and footer - we can't use the header / footer // from the cloned table, since if scrolling is active, the table's // real header and footer are contained in different table tags tmpTable .append( $(settings.nTHead).clone() ) .append( $(settings.nTFoot).clone() ); // Remove any assigned widths from the footer (from scrolling) tmpTable.find('tfoot th, tfoot td').css('width', ''); // Apply custom sizing to the cloned header tmpTable.find('thead th, thead td').each( function () { // Get the `width` from the header layout var width = _fnColumnsSumWidth( settings, this, true, false ); if ( width ) { this.style.width = width; // For scrollX we need to force the column width otherwise the // browser will collapse it. If this width is smaller than the // width the column requires, then it will have no effect if ( scrollX ) { $( this ).append( $('<div/>').css( { width: width, margin: 0, padding: 0, border: 0, height: 1 } ) ); } } else { this.style.width = ''; } } ); // Find the widest piece of data for each column and put it into the table for ( i=0 ; i<visibleColumns.length ; i++ ) { columnIdx = visibleColumns[i]; column = columns[ columnIdx ]; var longest = _fnGetMaxLenString(settings, columnIdx); var autoClass = _ext.type.className[column.sType]; var text = longest + column.sContentPadding; var insert = longest.indexOf('<') === -1 ? document.createTextNode(text) : text $('<td/>') .addClass(autoClass) .addClass(column.sClass) .append(insert) .appendTo(tr); } // Tidy the temporary table - remove name attributes so there aren't // duplicated in the dom (radio elements for example) $('[name]', tmpTable).removeAttr('name'); // Table has been built, attach to the document so we can work with it. // A holding element is used, positioned at the top of the container // with minimal height, so it has no effect on if the container scrolls // or not. Otherwise it might trigger scrolling when it actually isn't // needed var holder = $('<div/>').css( scrollX || scrollY ? { position: 'absolute', top: 0, left: 0, height: 1, right: 0, overflow: 'hidden' } : {} ) .append( tmpTable ) .appendTo( tableContainer ); // When scrolling (X or Y) we want to set the width of the table as // appropriate. However, when not scrolling leave the table width as it // is. This results in slightly different, but I think correct behaviour if ( scrollX && scrollXInner ) { tmpTable.width( scrollXInner ); } else if ( scrollX ) { tmpTable.css( 'width', 'auto' ); tmpTable.removeAttr('width'); // If there is no width attribute or style, then allow the table to // collapse if ( tmpTable.width() < tableContainer.clientWidth && tableWidthAttr ) { tmpTable.width( tableContainer.clientWidth ); } } else if ( scrollY ) { tmpTable.width( tableContainer.clientWidth ); } else if ( tableWidthAttr ) { tmpTable.width( tableWidthAttr ); } // Get the width of each column in the constructed table var total = 0; var bodyCells = tmpTable.find('tbody tr').eq(0).children(); for ( i=0 ; i<visibleColumns.length ; i++ ) { // Use getBounding for sub-pixel accuracy, which we then want to round up! var bounding = bodyCells[i].getBoundingClientRect().width; // Total is tracked to remove any sub-pixel errors as the outerWidth // of the table might not equal the total given here total += bounding; // Width for each column to use columns[ visibleColumns[i] ].sWidth = _fnStringToCss( bounding ); } table.style.width = _fnStringToCss( total ); // Finished with the table - ditch it holder.remove(); // If there is a width attr, we want to attach an event listener which // allows the table sizing to automatically adjust when the window is // resized. Use the width attr rather than CSS, since we can't know if the // CSS is a relative value or absolute - DOM read is always px. if ( tableWidthAttr ) { table.style.width = _fnStringToCss( tableWidthAttr ); } if ( (tableWidthAttr || scrollX) && ! settings._reszEvt ) { var bindResize = function () { $(window).on('resize.DT-'+settings.sInstance, DataTable.util.throttle( function () { if (! settings.bDestroying) { _fnAdjustColumnSizing( settings ); } } ) ); }; bindResize(); settings._reszEvt = true; } } /** * Get the maximum strlen for each data column * @param {object} settings dataTables settings object * @param {int} colIdx column of interest * @returns {string} string of the max length * @memberof DataTable#oApi */ function _fnGetMaxLenString( settings, colIdx ) { var column = settings.aoColumns[colIdx]; if (! column.maxLenString) { var s, max='', maxLen = -1; for ( var i=0, ien=settings.aiDisplayMaster.length ; i<ien ; i++ ) { var rowIdx = settings.aiDisplayMaster[i]; var data = _fnGetRowDisplay(settings, rowIdx)[colIdx]; var cellString = data && typeof data === 'object' && data.nodeType ? data.innerHTML : data+''; // Remove id / name attributes from elements so they // don't interfere with existing elements cellString = cellString .replace(/id=".*?"/g, '') .replace(/name=".*?"/g, ''); s = _stripHtml(cellString) .replace( / /g, ' ' ); if ( s.length > maxLen ) { // We want the HTML in the string, but the length that // is important is the stripped string max = cellString; maxLen = s.length; } } column.maxLenString = max; } return column.maxLenString; } /** * Append a CSS unit (only if required) to a string * @param {string} value to css-ify * @returns {string} value with css unit * @memberof DataTable#oApi */ function _fnStringToCss( s ) { if ( s === null ) { return '0px'; } if ( typeof s == 'number' ) { return s < 0 ? '0px' : s+'px'; } // Check it has a unit character already return s.match(/\d$/) ? s+'px' : s; } /** * Re-insert the `col` elements for current visibility * * @param {*} settings DT settings */ function _colGroup( settings ) { var cols = settings.aoColumns; settings.colgroup.empty(); for (i=0 ; i<cols.length ; i++) { if (cols[i].bVisible) { settings.colgroup.append(cols[i].colEl); } } } function _fnSortInit( settings ) { var target = settings.nTHead; var headerRows = target.querySelectorAll('tr'); var legacyTop = settings.bSortCellsTop; var notSelector = ':not([data-dt-order="disable"]):not([data-dt-order="icon-only"])'; // Legacy support for `orderCellsTop` if (legacyTop === true) { target = headerRows[0]; } else if (legacyTop === false) { target = headerRows[ headerRows.length - 1 ]; } _fnSortAttachListener( settings, target, target === settings.nTHead ? 'tr'+notSelector+' th'+notSelector+', tr'+notSelector+' td'+notSelector : 'th'+notSelector+', td'+notSelector ); // Need to resolve the user input array into our internal structure var order = []; _fnSortResolve( settings, order, settings.aaSorting ); settings.aaSorting = order; } function _fnSortAttachListener(settings, node, selector, column, callback) { _fnBindAction( node, selector, function (e) { var run = false; var columns = column === undefined ? _fnColumnsFromHeader( e.target ) : [column]; if ( columns.length ) { for ( var i=0, ien=columns.length ; i<ien ; i++ ) { var ret = _fnSortAdd( settings, columns[i], i, e.shiftKey ); if (ret !== false) { run = true; } // If the first entry is no sort, then subsequent // sort columns are ignored if (settings.aaSorting.length === 1 && settings.aaSorting[0][1] === '') { break; } } if (run) { _fnProcessingDisplay( settings, true ); // Allow the processing display to show setTimeout( function () { _fnSort( settings ); _fnSortDisplay( settings, settings.aiDisplay ); // Sort processing done - redraw has its own processing display _fnProcessingDisplay( settings, false ); _fnReDraw( settings, false, false ); if (callback) { callback(); } }, 0); } } } ); } /** * Sort the display array to match the master's order * @param {*} settings */ function _fnSortDisplay(settings, display) { if (display.length < 2) { return; } var master = settings.aiDisplayMaster; var masterMap = {}; var map = {}; var i; // Rather than needing an `indexOf` on master array, we can create a map for (i=0 ; i<master.length ; i++) { masterMap[master[i]] = i; } // And then cache what would be the indexOf fom the display for (i=0 ; i<display.length ; i++) { map[display[i]] = masterMap[display[i]]; } display.sort(function(a, b){ // Short version of this function is simply `master.indexOf(a) - master.indexOf(b);` return map[a] - map[b]; }); } function _fnSortResolve (settings, nestedSort, sort) { var push = function ( a ) { if ($.isPlainObject(a)) { if (a.idx !== undefined) { // Index based ordering nestedSort.push([a.idx, a.dir]); } else if (a.name) { // Name based ordering var cols = _pluck( settings.aoColumns, 'sName'); var idx = cols.indexOf(a.name); if (idx !== -1) { nestedSort.push([idx, a.dir]); } } } else { // Plain column index and direction pair nestedSort.push(a); } }; if ( $.isPlainObject(sort) ) { // Object push(sort); } else if ( sort.length && typeof sort[0] === 'number' ) { // 1D array push(sort); } else if ( sort.length ) { // 2D array for (var z=0; z<sort.length; z++) { push(sort[z]); // Object or array } } } function _fnSortFlatten ( settings ) { var i, k, kLen, aSort = [], extSort = DataTable.ext.type.order, aoColumns = settings.aoColumns, aDataSort, iCol, sType, srcCol, fixed = settings.aaSortingFixed, fixedObj = $.isPlainObject( fixed ), nestedSort = []; if ( ! settings.oFeatures.bSort ) { return aSort; } // Build the sort array, with pre-fix and post-fix options if they have been // specified if ( Array.isArray( fixed ) ) { _fnSortResolve( settings, nestedSort, fixed ); } if ( fixedObj && fixed.pre ) { _fnSortResolve( settings, nestedSort, fixed.pre ); } _fnSortResolve( settings, nestedSort, settings.aaSorting ); if (fixedObj && fixed.post ) { _fnSortResolve( settings, nestedSort, fixed.post ); } for ( i=0 ; i<nestedSort.length ; i++ ) { srcCol = nestedSort[i][0]; if ( aoColumns[ srcCol ] ) { aDataSort = aoColumns[ srcCol ].aDataSort; for ( k=0, kLen=aDataSort.length ; k<kLen ; k++ ) { iCol = aDataSort[k]; sType = aoColumns[ iCol ].sType || 'string'; if ( nestedSort[i]._idx === undefined ) { nestedSort[i]._idx = aoColumns[iCol].asSorting.indexOf(nestedSort[i][1]); } if ( nestedSort[i][1] ) { aSort.push( { src: srcCol, col: iCol, dir: nestedSort[i][1], index: nestedSort[i]._idx, type: sType, formatter: extSort[ sType+"-pre" ], sorter: extSort[ sType+"-"+nestedSort[i][1] ] } ); } } } } return aSort; } /** * Change the order of the table * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnSort ( oSettings, col, dir ) { var i, ien, iLen, aiOrig = [], extSort = DataTable.ext.type.order, aoData = oSettings.aoData, sortCol, displayMaster = oSettings.aiDisplayMaster, aSort; // Resolve any column types that are unknown due to addition or invalidation // @todo Can this be moved into a 'data-ready' handler which is called when // data is going to be used in the table? _fnColumnTypes( oSettings ); // Allow a specific column to be sorted, which will _not_ alter the display // master if (col !== undefined) { var srcCol = oSettings.aoColumns[col]; aSort = [{ src: col, col: col, dir: dir, index: 0, type: srcCol.sType, formatter: extSort[ srcCol.sType+"-pre" ], sorter: extSort[ srcCol.sType+"-"+dir ] }]; displayMaster = displayMaster.slice(); } else { aSort = _fnSortFlatten( oSettings ); } for ( i=0, ien=aSort.length ; i<ien ; i++ ) { sortCol = aSort[i]; // Load the data needed for the sort, for each cell _fnSortData( oSettings, sortCol.col ); } /* No sorting required if server-side or no sorting array */ if ( _fnDataSource( oSettings ) != 'ssp' && aSort.length !== 0 ) { // Reset the initial positions on each pass so we get a stable sort for ( i=0, iLen=displayMaster.length ; i<iLen ; i++ ) { aiOrig[ i ] = i; } // If the first sort is desc, then reverse the array to preserve original // order, just in reverse if (aSort.length && aSort[0].dir === 'desc') { aiOrig.reverse(); } /* Do the sort - here we want multi-column sorting based on a given data source (column) * and sorting function (from oSort) in a certain direction. It's reasonably complex to * follow on it's own, but this is what we want (example two column sorting): * fnLocalSorting = function(a,b){ * var test; * test = oSort['string-asc']('data11', 'data12'); * if (test !== 0) * return test; * test = oSort['numeric-desc']('data21', 'data22'); * if (test !== 0) * return test; * return oSort['numeric-asc']( aiOrig[a], aiOrig[b] ); * } * Basically we have a test for each sorting column, if the data in that column is equal, * test the next column. If all columns match, then we use a numeric sort on the row * positions in the original data array to provide a stable sort. */ displayMaster.sort( function ( a, b ) { var x, y, k, test, sort, len=aSort.length, dataA = aoData[a]._aSortData, dataB = aoData[b]._aSortData; for ( k=0 ; k<len ; k++ ) { sort = aSort[k]; // Data, which may have already been through a `-pre` function x = dataA[ sort.col ]; y = dataB[ sort.col ]; if (sort.sorter) { // If there is a custom sorter (`-asc` or `-desc`) for this // data type, use it test = sort.sorter(x, y); if ( test !== 0 ) { return test; } } else { // Otherwise, use generic sorting test = x<y ? -1 : x>y ? 1 : 0; if ( test !== 0 ) { return sort.dir === 'asc' ? test : -test; } } } x = aiOrig[a]; y = aiOrig[b]; return x<y ? -1 : x>y ? 1 : 0; } ); } else if ( aSort.length === 0 ) { // Apply index order displayMaster.sort(function (x, y) { return x<y ? -1 : x>y ? 1 : 0; }); } if (col === undefined) { // Tell the draw function that we have sorted the data oSettings.bSorted = true; _fnCallbackFire( oSettings, null, 'order', [oSettings, aSort] ); } return displayMaster; } /** * Function to run on user sort request * @param {object} settings dataTables settings object * @param {node} attachTo node to attach the handler to * @param {int} colIdx column sorting index * @param {int} addIndex Counter * @param {boolean} [shift=false] Shift click add * @param {function} [callback] callback function * @memberof DataTable#oApi */ function _fnSortAdd ( settings, colIdx, addIndex, shift ) { var col = settings.aoColumns[ colIdx ]; var sorting = settings.aaSorting; var asSorting = col.asSorting; var nextSortIdx; var next = function ( a, overflow ) { var idx = a._idx; if ( idx === undefined ) { idx = asSorting.indexOf(a[1]); } return idx+1 < asSorting.length ? idx+1 : overflow ? null : 0; }; if ( ! col.bSortable ) { return false; } // Convert to 2D array if needed if ( typeof sorting[0] === 'number' ) { sorting = settings.aaSorting = [ sorting ]; } // If appending the sort then we are multi-column sorting if ( (shift || addIndex) && settings.oFeatures.bSortMulti ) { // Are we already doing some kind of sort on this column? var sortIdx = _pluck(sorting, '0').indexOf(colIdx); if ( sortIdx !== -1 ) { // Yes, modify the sort nextSortIdx = next( sorting[sortIdx], true ); if ( nextSortIdx === null && sorting.length === 1 ) { nextSortIdx = 0; // can't remove sorting completely } if ( nextSortIdx === null ) { sorting.splice( sortIdx, 1 ); } else { sorting[sortIdx][1] = asSorting[ nextSortIdx ]; sorting[sortIdx]._idx = nextSortIdx; } } else if (shift) { // No sort on this column yet, being added by shift click // add it as itself sorting.push( [ colIdx, asSorting[0], 0 ] ); sorting[sorting.length-1]._idx = 0; } else { // No sort on this column yet, being added from a colspan // so add with same direction as first column sorting.push( [ colIdx, sorting[0][1], 0 ] ); sorting[sorting.length-1]._idx = 0; } } else if ( sorting.length && sorting[0][0] == colIdx ) { // Single column - already sorting on this column, modify the sort nextSortIdx = next( sorting[0] ); sorting.length = 1; sorting[0][1] = asSorting[ nextSortIdx ]; sorting[0]._idx = nextSortIdx; } else { // Single column - sort only on this column sorting.length = 0; sorting.push( [ colIdx, asSorting[0] ] ); sorting[0]._idx = 0; } } /** * Set the sorting classes on table's body, Note: it is safe to call this function * when bSort and bSortClasses are false * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnSortingClasses( settings ) { var oldSort = settings.aLastSort; var sortClass = settings.oClasses.order.position; var sort = _fnSortFlatten( settings ); var features = settings.oFeatures; var i, ien, colIdx; if ( features.bSort && features.bSortClasses ) { // Remove old sorting classes for ( i=0, ien=oldSort.length ; i<ien ; i++ ) { colIdx = oldSort[i].src; // Remove column sorting $( _pluck( settings.aoData, 'anCells', colIdx ) ) .removeClass( sortClass + (i<2 ? i+1 : 3) ); } // Add new column sorting for ( i=0, ien=sort.length ; i<ien ; i++ ) { colIdx = sort[i].src; $( _pluck( settings.aoData, 'anCells', colIdx ) ) .addClass( sortClass + (i<2 ? i+1 : 3) ); } } settings.aLastSort = sort; } // Get the data to sort a column, be it from cache, fresh (populating the // cache), or from a sort formatter function _fnSortData( settings, colIdx ) { // Custom sorting function - provided by the sort data type var column = settings.aoColumns[ colIdx ]; var customSort = DataTable.ext.order[ column.sSortDataType ]; var customData; if ( customSort ) { customData = customSort.call( settings.oInstance, settings, colIdx, _fnColumnIndexToVisible( settings, colIdx ) ); } // Use / populate cache var row, cellData; var formatter = DataTable.ext.type.order[ column.sType+"-pre" ]; var data = settings.aoData; for ( var rowIdx=0 ; rowIdx<data.length ; rowIdx++ ) { // Sparse array if (! data[rowIdx]) { continue; } row = data[rowIdx]; if ( ! row._aSortData ) { row._aSortData = []; } if ( ! row._aSortData[colIdx] || customSort ) { cellData = customSort ? customData[rowIdx] : // If there was a custom sort function, use data from there _fnGetCellData( settings, rowIdx, colIdx, 'sort' ); row._aSortData[ colIdx ] = formatter ? formatter( cellData, settings ) : cellData; } } } /** * State information for a table * * @param {*} settings * @returns State object */ function _fnSaveState ( settings ) { if (settings._bLoadingState) { return; } /* Store the interesting variables */ var state = { time: +new Date(), start: settings._iDisplayStart, length: settings._iDisplayLength, order: $.extend( true, [], settings.aaSorting ), search: $.extend({}, settings.oPreviousSearch), columns: settings.aoColumns.map( function ( col, i ) { return { visible: col.bVisible, search: $.extend({}, settings.aoPreSearchCols[i]) }; } ) }; settings.oSavedState = state; _fnCallbackFire( settings, "aoStateSaveParams", 'stateSaveParams', [settings, state] ); if ( settings.oFeatures.bStateSave && !settings.bDestroying ) { settings.fnStateSaveCallback.call( settings.oInstance, settings, state ); } } /** * Attempt to load a saved table state * @param {object} oSettings dataTables settings object * @param {object} oInit DataTables init object so we can override settings * @param {function} callback Callback to execute when the state has been loaded * @memberof DataTable#oApi */ function _fnLoadState ( settings, init, callback ) { if ( ! settings.oFeatures.bStateSave ) { callback(); return; } var loaded = function(state) { _fnImplementState(settings, state, callback); } var state = settings.fnStateLoadCallback.call( settings.oInstance, settings, loaded ); if ( state !== undefined ) { _fnImplementState( settings, state, callback ); } // otherwise, wait for the loaded callback to be executed return true; } function _fnImplementState ( settings, s, callback) { var i, ien; var columns = settings.aoColumns; settings._bLoadingState = true; // When StateRestore was introduced the state could now be implemented at any time // Not just initialisation. To do this an api instance is required in some places var api = settings._bInitComplete ? new DataTable.Api(settings) : null; if ( ! s || ! s.time ) { settings._bLoadingState = false; callback(); return; } // Reject old data var duration = settings.iStateDuration; if ( duration > 0 && s.time < +new Date() - (duration*1000) ) { settings._bLoadingState = false; callback(); return; } // Allow custom and plug-in manipulation functions to alter the saved data set and // cancelling of loading by returning false var abStateLoad = _fnCallbackFire( settings, 'aoStateLoadParams', 'stateLoadParams', [settings, s] ); if ( abStateLoad.indexOf(false) !== -1 ) { settings._bLoadingState = false; callback(); return; } // Number of columns have changed - all bets are off, no restore of settings if ( s.columns && columns.length !== s.columns.length ) { settings._bLoadingState = false; callback(); return; } // Store the saved state so it might be accessed at any time settings.oLoadedState = $.extend( true, {}, s ); // This is needed for ColReorder, which has to happen first to allow all // the stored indexes to be usable. It is not publicly documented. _fnCallbackFire( settings, null, 'stateLoadInit', [settings, s], true ); // Page Length if ( s.length !== undefined ) { // If already initialised just set the value directly so that the select element is also updated if (api) { api.page.len(s.length) } else { settings._iDisplayLength = s.length; } } // Restore key features - todo - for 1.11 this needs to be done by // subscribed events if ( s.start !== undefined ) { if(api === null) { settings._iDisplayStart = s.start; settings.iInitDisplayStart = s.start; } else { _fnPageChange(settings, s.start/settings._iDisplayLength); } } // Order if ( s.order !== undefined ) { settings.aaSorting = []; $.each( s.order, function ( i, col ) { settings.aaSorting.push( col[0] >= columns.length ? [ 0, col[1] ] : col ); } ); } // Search if ( s.search !== undefined ) { $.extend( settings.oPreviousSearch, s.search ); } // Columns if ( s.columns ) { for ( i=0, ien=s.columns.length ; i<ien ; i++ ) { var col = s.columns[i]; // Visibility if ( col.visible !== undefined ) { // If the api is defined, the table has been initialised so we need to use it rather than internal settings if (api) { // Don't redraw the columns on every iteration of this loop, we will do this at the end instead api.column(i).visible(col.visible, false); } else { columns[i].bVisible = col.visible; } } // Search if ( col.search !== undefined ) { $.extend( settings.aoPreSearchCols[i], col.search ); } } // If the api is defined then we need to adjust the columns once the visibility has been changed if (api) { api.columns.adjust(); } } settings._bLoadingState = false; _fnCallbackFire( settings, 'aoStateLoaded', 'stateLoaded', [settings, s] ); callback(); } /** * Log an error message * @param {object} settings dataTables settings object * @param {int} level log error messages, or display them to the user * @param {string} msg error message * @param {int} tn Technical note id to get more information about the error. * @memberof DataTable#oApi */ function _fnLog( settings, level, msg, tn ) { msg = 'DataTables warning: '+ (settings ? 'table id='+settings.sTableId+' - ' : '')+msg; if ( tn ) { msg += '. For more information about this error, please see '+ 'https://datatables.net/tn/'+tn; } if ( ! level ) { // Backwards compatibility pre 1.10 var ext = DataTable.ext; var type = ext.sErrMode || ext.errMode; if ( settings ) { _fnCallbackFire( settings, null, 'dt-error', [ settings, tn, msg ], true ); } if ( type == 'alert' ) { alert( msg ); } else if ( type == 'throw' ) { throw new Error(msg); } else if ( typeof type == 'function' ) { type( settings, tn, msg ); } } else if ( window.console && console.log ) { console.log( msg ); } } /** * See if a property is defined on one object, if so assign it to the other object * @param {object} ret target object * @param {object} src source object * @param {string} name property * @param {string} [mappedName] name to map too - optional, name used if not given * @memberof DataTable#oApi */ function _fnMap( ret, src, name, mappedName ) { if ( Array.isArray( name ) ) { $.each( name, function (i, val) { if ( Array.isArray( val ) ) { _fnMap( ret, src, val[0], val[1] ); } else { _fnMap( ret, src, val ); } } ); return; } if ( mappedName === undefined ) { mappedName = name; } if ( src[name] !== undefined ) { ret[mappedName] = src[name]; } } /** * Extend objects - very similar to jQuery.extend, but deep copy objects, and * shallow copy arrays. The reason we need to do this, is that we don't want to * deep copy array init values (such as aaSorting) since the dev wouldn't be * able to override them, but we do want to deep copy arrays. * @param {object} out Object to extend * @param {object} extender Object from which the properties will be applied to * out * @param {boolean} breakRefs If true, then arrays will be sliced to take an * independent copy with the exception of the `data` or `aaData` parameters * if they are present. This is so you can pass in a collection to * DataTables and have that used as your data source without breaking the * references * @returns {object} out Reference, just for convenience - out === the return. * @memberof DataTable#oApi * @todo This doesn't take account of arrays inside the deep copied objects. */ function _fnExtend( out, extender, breakRefs ) { var val; for ( var prop in extender ) { if ( Object.prototype.hasOwnProperty.call(extender, prop) ) { val = extender[prop]; if ( $.isPlainObject( val ) ) { if ( ! $.isPlainObject( out[prop] ) ) { out[prop] = {}; } $.extend( true, out[prop], val ); } else if ( breakRefs && prop !== 'data' && prop !== 'aaData' && Array.isArray(val) ) { out[prop] = val.slice(); } else { out[prop] = val; } } } return out; } /** * Bind an event handers to allow a click or return key to activate the callback. * This is good for accessibility since a return on the keyboard will have the * same effect as a click, if the element has focus. * @param {element} n Element to bind the action to * @param {object|string} selector Selector (for delegated events) or data object * to pass to the triggered function * @param {function} fn Callback function for when the event is triggered * @memberof DataTable#oApi */ function _fnBindAction( n, selector, fn ) { $(n) .on( 'click.DT', selector, function (e) { fn(e); } ) .on( 'keypress.DT', selector, function (e){ if ( e.which === 13 ) { e.preventDefault(); fn(e); } } ) .on( 'selectstart.DT', selector, function () { // Don't want a double click resulting in text selection return false; } ); } /** * Register a callback function. Easily allows a callback function to be added to * an array store of callback functions that can then all be called together. * @param {object} settings dataTables settings object * @param {string} store Name of the array storage for the callbacks in oSettings * @param {function} fn Function to be called back * @memberof DataTable#oApi */ function _fnCallbackReg( settings, store, fn ) { if ( fn ) { settings[store].push(fn); } } /** * Fire callback functions and trigger events. Note that the loop over the * callback array store is done backwards! Further note that you do not want to * fire off triggers in time sensitive applications (for example cell creation) * as its slow. * @param {object} settings dataTables settings object * @param {string} callbackArr Name of the array storage for the callbacks in * oSettings * @param {string} eventName Name of the jQuery custom event to trigger. If * null no trigger is fired * @param {array} args Array of arguments to pass to the callback function / * trigger * @param {boolean} [bubbles] True if the event should bubble * @memberof DataTable#oApi */ function _fnCallbackFire( settings, callbackArr, eventName, args, bubbles ) { var ret = []; if ( callbackArr ) { ret = settings[callbackArr].slice().reverse().map( function (val) { return val.apply( settings.oInstance, args ); } ); } if ( eventName !== null) { var e = $.Event( eventName+'.dt' ); var table = $(settings.nTable); // Expose the DataTables API on the event object for easy access e.dt = settings.api; table[bubbles ? 'trigger' : 'triggerHandler']( e, args ); // If not yet attached to the document, trigger the event // on the body directly to sort of simulate the bubble if (bubbles && table.parents('body').length === 0) { $('body').trigger( e, args ); } ret.push( e.result ); } return ret; } function _fnLengthOverflow ( settings ) { var start = settings._iDisplayStart, end = settings.fnDisplayEnd(), len = settings._iDisplayLength; /* If we have space to show extra rows (backing up from the end point - then do so */ if ( start >= end ) { start = end - len; } // Keep the start record on the current page start -= (start % len); if ( len === -1 || start < 0 ) { start = 0; } settings._iDisplayStart = start; } function _fnRenderer( settings, type ) { var renderer = settings.renderer; var host = DataTable.ext.renderer[type]; if ( $.isPlainObject( renderer ) && renderer[type] ) { // Specific renderer for this type. If available use it, otherwise use // the default. return host[renderer[type]] || host._; } else if ( typeof renderer === 'string' ) { // Common renderer - if there is one available for this type use it, // otherwise use the default return host[renderer] || host._; } // Use the default return host._; } /** * Detect the data source being used for the table. Used to simplify the code * a little (ajax) and to make it compress a little smaller. * * @param {object} settings dataTables settings object * @returns {string} Data source * @memberof DataTable#oApi */ function _fnDataSource ( settings ) { if ( settings.oFeatures.bServerSide ) { return 'ssp'; } else if ( settings.ajax ) { return 'ajax'; } return 'dom'; } /** * Common replacement for language strings * * @param {*} settings DT settings object * @param {*} str String with values to replace * @param {*} entries Plural number for _ENTRIES_ - can be undefined * @returns String */ function _fnMacros ( settings, str, entries ) { // When infinite scrolling, we are always starting at 1. _iDisplayStart is // used only internally var formatter = settings.fnFormatNumber, start = settings._iDisplayStart+1, len = settings._iDisplayLength, vis = settings.fnRecordsDisplay(), max = settings.fnRecordsTotal(), all = len === -1; return str. replace(/_START_/g, formatter.call( settings, start ) ). replace(/_END_/g, formatter.call( settings, settings.fnDisplayEnd() ) ). replace(/_MAX_/g, formatter.call( settings, max ) ). replace(/_TOTAL_/g, formatter.call( settings, vis ) ). replace(/_PAGE_/g, formatter.call( settings, all ? 1 : Math.ceil( start / len ) ) ). replace(/_PAGES_/g, formatter.call( settings, all ? 1 : Math.ceil( vis / len ) ) ). replace(/_ENTRIES_/g, settings.api.i18n('entries', '', entries) ). replace(/_ENTRIES-MAX_/g, settings.api.i18n('entries', '', max) ). replace(/_ENTRIES-TOTAL_/g, settings.api.i18n('entries', '', vis) ); } /** * Computed structure of the DataTables API, defined by the options passed to * `DataTable.Api.register()` when building the API. * * The structure is built in order to speed creation and extension of the Api * objects since the extensions are effectively pre-parsed. * * The array is an array of objects with the following structure, where this * base array represents the Api prototype base: * * [ * { * name: 'data' -- string - Property name * val: function () {}, -- function - Api method (or undefined if just an object * methodExt: [ ... ], -- array - Array of Api object definitions to extend the method result * propExt: [ ... ] -- array - Array of Api object definitions to extend the property * }, * { * name: 'row' * val: {}, * methodExt: [ ... ], * propExt: [ * { * name: 'data' * val: function () {}, * methodExt: [ ... ], * propExt: [ ... ] * }, * ... * ] * } * ] * * @type {Array} * @ignore */ var __apiStruct = []; /** * `Array.prototype` reference. * * @type object * @ignore */ var __arrayProto = Array.prototype; /** * Abstraction for `context` parameter of the `Api` constructor to allow it to * take several different forms for ease of use. * * Each of the input parameter types will be converted to a DataTables settings * object where possible. * * @param {string|node|jQuery|object} mixed DataTable identifier. Can be one * of: * * * `string` - jQuery selector. Any DataTables' matching the given selector * with be found and used. * * `node` - `TABLE` node which has already been formed into a DataTable. * * `jQuery` - A jQuery object of `TABLE` nodes. * * `object` - DataTables settings object * * `DataTables.Api` - API instance * @return {array|null} Matching DataTables settings objects. `null` or * `undefined` is returned if no matching DataTable is found. * @ignore */ var _toSettings = function ( mixed ) { var idx, jq; var settings = DataTable.settings; var tables = _pluck(settings, 'nTable'); if ( ! mixed ) { return []; } else if ( mixed.nTable && mixed.oFeatures ) { // DataTables settings object return [ mixed ]; } else if ( mixed.nodeName && mixed.nodeName.toLowerCase() === 'table' ) { // Table node idx = tables.indexOf(mixed); return idx !== -1 ? [ settings[idx] ] : null; } else if ( mixed && typeof mixed.settings === 'function' ) { return mixed.settings().toArray(); } else if ( typeof mixed === 'string' ) { // jQuery selector jq = $(mixed).get(); } else if ( mixed instanceof $ ) { // jQuery object (also DataTables instance) jq = mixed.get(); } if ( jq ) { return settings.filter(function (v, idx) { return jq.includes(tables[idx]); }); } }; /** * DataTables API class - used to control and interface with one or more * DataTables enhanced tables. * * The API class is heavily based on jQuery, presenting a chainable interface * that you can use to interact with tables. Each instance of the API class has * a "context" - i.e. the tables that it will operate on. This could be a single * table, all tables on a page or a sub-set thereof. * * Additionally the API is designed to allow you to easily work with the data in * the tables, retrieving and manipulating it as required. This is done by * presenting the API class as an array like interface. The contents of the * array depend upon the actions requested by each method (for example * `rows().nodes()` will return an array of nodes, while `rows().data()` will * return an array of objects or arrays depending upon your table's * configuration). The API object has a number of array like methods (`push`, * `pop`, `reverse` etc) as well as additional helper methods (`each`, `pluck`, * `unique` etc) to assist your working with the data held in a table. * * Most methods (those which return an Api instance) are chainable, which means * the return from a method call also has all of the methods available that the * top level object had. For example, these two calls are equivalent: * * // Not chained * api.row.add( {...} ); * api.draw(); * * // Chained * api.row.add( {...} ).draw(); * * @class DataTable.Api * @param {array|object|string|jQuery} context DataTable identifier. This is * used to define which DataTables enhanced tables this API will operate on. * Can be one of: * * * `string` - jQuery selector. Any DataTables' matching the given selector * with be found and used. * * `node` - `TABLE` node which has already been formed into a DataTable. * * `jQuery` - A jQuery object of `TABLE` nodes. * * `object` - DataTables settings object * @param {array} [data] Data to initialise the Api instance with. * * @example * // Direct initialisation during DataTables construction * var api = $('#example').DataTable(); * * @example * // Initialisation using a DataTables jQuery object * var api = $('#example').dataTable().api(); * * @example * // Initialisation as a constructor * var api = new DataTable.Api( 'table.dataTable' ); */ _Api = function ( context, data ) { if ( ! (this instanceof _Api) ) { return new _Api( context, data ); } var settings = []; var ctxSettings = function ( o ) { var a = _toSettings( o ); if ( a ) { settings.push.apply( settings, a ); } }; if ( Array.isArray( context ) ) { for ( var i=0, ien=context.length ; i<ien ; i++ ) { ctxSettings( context[i] ); } } else { ctxSettings( context ); } // Remove duplicates this.context = settings.length > 1 ? _unique( settings ) : settings; // Initial data if ( data ) { this.push.apply(this, data); } // selector this.selector = { rows: null, cols: null, opts: null }; _Api.extend( this, this, __apiStruct ); }; DataTable.Api = _Api; // Don't destroy the existing prototype, just extend it. Required for jQuery 2's // isPlainObject. $.extend( _Api.prototype, { any: function () { return this.count() !== 0; }, context: [], // array of table settings objects count: function () { return this.flatten().length; }, each: function ( fn ) { for ( var i=0, ien=this.length ; i<ien; i++ ) { fn.call( this, this[i], i, this ); } return this; }, eq: function ( idx ) { var ctx = this.context; return ctx.length > idx ? new _Api( ctx[idx], this[idx] ) : null; }, filter: function ( fn ) { var a = __arrayProto.filter.call( this, fn, this ); return new _Api( this.context, a ); }, flatten: function () { var a = []; return new _Api( this.context, a.concat.apply( a, this.toArray() ) ); }, get: function ( idx ) { return this[ idx ]; }, join: __arrayProto.join, includes: function ( find ) { return this.indexOf( find ) === -1 ? false : true; }, indexOf: __arrayProto.indexOf, iterator: function ( flatten, type, fn, alwaysNew ) { var a = [], ret, i, ien, j, jen, context = this.context, rows, items, item, selector = this.selector; // Argument shifting if ( typeof flatten === 'string' ) { alwaysNew = fn; fn = type; type = flatten; flatten = false; } for ( i=0, ien=context.length ; i<ien ; i++ ) { var apiInst = new _Api( context[i] ); if ( type === 'table' ) { ret = fn.call( apiInst, context[i], i ); if ( ret !== undefined ) { a.push( ret ); } } else if ( type === 'columns' || type === 'rows' ) { // this has same length as context - one entry for each table ret = fn.call( apiInst, context[i], this[i], i ); if ( ret !== undefined ) { a.push( ret ); } } else if ( type === 'every' || type === 'column' || type === 'column-rows' || type === 'row' || type === 'cell' ) { // columns and rows share the same structure. // 'this' is an array of column indexes for each context items = this[i]; if ( type === 'column-rows' ) { rows = _selector_row_indexes( context[i], selector.opts ); } for ( j=0, jen=items.length ; j<jen ; j++ ) { item = items[j]; if ( type === 'cell' ) { ret = fn.call( apiInst, context[i], item.row, item.column, i, j ); } else { ret = fn.call( apiInst, context[i], item, i, j, rows ); } if ( ret !== undefined ) { a.push( ret ); } } } } if ( a.length || alwaysNew ) { var api = new _Api( context, flatten ? a.concat.apply( [], a ) : a ); var apiSelector = api.selector; apiSelector.rows = selector.rows; apiSelector.cols = selector.cols; apiSelector.opts = selector.opts; return api; } return this; }, lastIndexOf: __arrayProto.lastIndexOf, length: 0, map: function ( fn ) { var a = __arrayProto.map.call( this, fn, this ); return new _Api( this.context, a ); }, pluck: function ( prop ) { var fn = DataTable.util.get(prop); return this.map( function ( el ) { return fn(el); } ); }, pop: __arrayProto.pop, push: __arrayProto.push, reduce: __arrayProto.reduce, reduceRight: __arrayProto.reduceRight, reverse: __arrayProto.reverse, // Object with rows, columns and opts selector: null, shift: __arrayProto.shift, slice: function () { return new _Api( this.context, this ); }, sort: __arrayProto.sort, splice: __arrayProto.splice, toArray: function () { return __arrayProto.slice.call( this ); }, to$: function () { return $( this ); }, toJQuery: function () { return $( this ); }, unique: function () { return new _Api( this.context, _unique(this.toArray()) ); }, unshift: __arrayProto.unshift } ); function _api_scope( scope, fn, struc ) { return function () { var ret = fn.apply( scope || this, arguments ); // Method extension _Api.extend( ret, ret, struc.methodExt ); return ret; }; } function _api_find( src, name ) { for ( var i=0, ien=src.length ; i<ien ; i++ ) { if ( src[i].name === name ) { return src[i]; } } return null; } window.__apiStruct = __apiStruct; _Api.extend = function ( scope, obj, ext ) { // Only extend API instances and static properties of the API if ( ! ext.length || ! obj || ( ! (obj instanceof _Api) && ! obj.__dt_wrapper ) ) { return; } var i, ien, struct; for ( i=0, ien=ext.length ; i<ien ; i++ ) { struct = ext[i]; if (struct.name === '__proto__') { continue; } // Value obj[ struct.name ] = struct.type === 'function' ? _api_scope( scope, struct.val, struct ) : struct.type === 'object' ? {} : struct.val; obj[ struct.name ].__dt_wrapper = true; // Property extension _Api.extend( scope, obj[ struct.name ], struct.propExt ); } }; // [ // { // name: 'data' -- string - Property name // val: function () {}, -- function - Api method (or undefined if just an object // methodExt: [ ... ], -- array - Array of Api object definitions to extend the method result // propExt: [ ... ] -- array - Array of Api object definitions to extend the property // }, // { // name: 'row' // val: {}, // methodExt: [ ... ], // propExt: [ // { // name: 'data' // val: function () {}, // methodExt: [ ... ], // propExt: [ ... ] // }, // ... // ] // } // ] _Api.register = _api_register = function ( name, val ) { if ( Array.isArray( name ) ) { for ( var j=0, jen=name.length ; j<jen ; j++ ) { _Api.register( name[j], val ); } return; } var i, ien, heir = name.split('.'), struct = __apiStruct, key, method; for ( i=0, ien=heir.length ; i<ien ; i++ ) { method = heir[i].indexOf('()') !== -1; key = method ? heir[i].replace('()', '') : heir[i]; var src = _api_find( struct, key ); if ( ! src ) { src = { name: key, val: {}, methodExt: [], propExt: [], type: 'object' }; struct.push( src ); } if ( i === ien-1 ) { src.val = val; src.type = typeof val === 'function' ? 'function' : $.isPlainObject( val ) ? 'object' : 'other'; } else { struct = method ? src.methodExt : src.propExt; } } }; _Api.registerPlural = _api_registerPlural = function ( pluralName, singularName, val ) { _Api.register( pluralName, val ); _Api.register( singularName, function () { var ret = val.apply( this, arguments ); if ( ret === this ) { // Returned item is the API instance that was passed in, return it return this; } else if ( ret instanceof _Api ) { // New API instance returned, want the value from the first item // in the returned array for the singular result. return ret.length ? Array.isArray( ret[0] ) ? new _Api( ret.context, ret[0] ) : // Array results are 'enhanced' ret[0] : undefined; } // Non-API return - just fire it back return ret; } ); }; /** * Selector for HTML tables. Apply the given selector to the give array of * DataTables settings objects. * * @param {string|integer} [selector] jQuery selector string or integer * @param {array} Array of DataTables settings objects to be filtered * @return {array} * @ignore */ var __table_selector = function ( selector, a ) { if ( Array.isArray(selector) ) { var result = []; selector.forEach(function (sel) { var inner = __table_selector(sel, a); result.push.apply(result, inner); }); return result.filter( function (item) { return item; }); } // Integer is used to pick out a table by index if ( typeof selector === 'number' ) { return [ a[ selector ] ]; } // Perform a jQuery selector on the table nodes var nodes = a.map( function (el) { return el.nTable; } ); return $(nodes) .filter( selector ) .map( function () { // Need to translate back from the table node to the settings var idx = nodes.indexOf(this); return a[ idx ]; } ) .toArray(); }; /** * Context selector for the API's context (i.e. the tables the API instance * refers to. * * @name DataTable.Api#tables * @param {string|integer} [selector] Selector to pick which tables the iterator * should operate on. If not given, all tables in the current context are * used. This can be given as a jQuery selector (for example `':gt(0)'`) to * select multiple tables or as an integer to select a single table. * @returns {DataTable.Api} Returns a new API instance if a selector is given. */ _api_register( 'tables()', function ( selector ) { // A new instance is created if there was a selector specified return selector !== undefined && selector !== null ? new _Api( __table_selector( selector, this.context ) ) : this; } ); _api_register( 'table()', function ( selector ) { var tables = this.tables( selector ); var ctx = tables.context; // Truncate to the first matched table return ctx.length ? new _Api( ctx[0] ) : tables; } ); // Common methods, combined to reduce size [ ['nodes', 'node', 'nTable'], ['body', 'body', 'nTBody'], ['header', 'header', 'nTHead'], ['footer', 'footer', 'nTFoot'], ].forEach(function (item) { _api_registerPlural( 'tables().' + item[0] + '()', 'table().' + item[1] + '()' , function () { return this.iterator( 'table', function ( ctx ) { return ctx[item[2]]; }, 1 ); } ); }); // Structure methods [ ['header', 'aoHeader'], ['footer', 'aoFooter'], ].forEach(function (item) { _api_register( 'table().' + item[0] + '.structure()' , function (selector) { var indexes = this.columns(selector).indexes().flatten(); var ctx = this.context[0]; return _fnHeaderLayout(ctx, ctx[item[1]], indexes); } ); }) _api_registerPlural( 'tables().containers()', 'table().container()' , function () { return this.iterator( 'table', function ( ctx ) { return ctx.nTableWrapper; }, 1 ); } ); _api_register( 'tables().every()', function ( fn ) { var that = this; return this.iterator('table', function (s, i) { fn.call(that.table(i), i); }); }); _api_register( 'caption()', function ( value, side ) { var context = this.context; // Getter - return existing node's content if ( value === undefined ) { var caption = context[0].captionNode; return caption && context.length ? caption.innerHTML : null; } return this.iterator( 'table', function ( ctx ) { var table = $(ctx.nTable); var caption = $(ctx.captionNode); var container = $(ctx.nTableWrapper); // Create the node if it doesn't exist yet if ( ! caption.length ) { caption = $('<caption/>').html( value ); ctx.captionNode = caption[0]; // If side isn't set, we need to insert into the document to let the // CSS decide so we can read it back, otherwise there is no way to // know if the CSS would put it top or bottom for scrolling if (! side) { table.prepend(caption); side = caption.css('caption-side'); } } caption.html( value ); if ( side ) { caption.css( 'caption-side', side ); caption[0]._captionSide = side; } if (container.find('div.dataTables_scroll').length) { var selector = (side === 'top' ? 'Head' : 'Foot'); container.find('div.dataTables_scroll'+ selector +' table').prepend(caption); } else { table.prepend(caption); } }, 1 ); } ); _api_register( 'caption.node()', function () { var ctx = this.context; return ctx.length ? ctx[0].captionNode : null; } ); /** * Redraw the tables in the current context. */ _api_register( 'draw()', function ( paging ) { return this.iterator( 'table', function ( settings ) { if ( paging === 'page' ) { _fnDraw( settings ); } else { if ( typeof paging === 'string' ) { paging = paging === 'full-hold' ? false : true; } _fnReDraw( settings, paging===false ); } } ); } ); /** * 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.iterator( 'table', function ( settings ) { _fnPageChange( settings, action ); } ); } ); /** * 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 () { if ( this.context.length === 0 ) { return undefined; } var settings = this.context[0], start = settings._iDisplayStart, len = settings.oFeatures.bPaginate ? settings._iDisplayLength : -1, visRecords = settings.fnRecordsDisplay(), all = len === -1; return { "page": all ? 0 : Math.floor( start / len ), "pages": all ? 1 : Math.ceil( visRecords / len ), "start": start, "end": settings.fnDisplayEnd(), "length": len, "recordsTotal": settings.fnRecordsTotal(), "recordsDisplay": visRecords, "serverSide": _fnDataSource( settings ) === 'ssp' }; } ); /** * 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.iterator( 'table', function ( settings ) { _fnLengthChange( settings, len ); } ); } ); var __reload = function ( settings, holdPosition, callback ) { // Use the draw event to trigger a callback if ( callback ) { var api = new _Api( settings ); api.one( 'draw', function () { callback( api.ajax.json() ); } ); } if ( _fnDataSource( settings ) == 'ssp' ) { _fnReDraw( settings, holdPosition ); } else { _fnProcessingDisplay( settings, true ); // Cancel an existing request var xhr = settings.jqXHR; if ( xhr && xhr.readyState !== 4 ) { xhr.abort(); } // Trigger xhr _fnBuildAjax( settings, {}, function( json ) { _fnClearTable( settings ); var data = _fnAjaxDataSrc( settings, json ); for ( var i=0, ien=data.length ; i<ien ; i++ ) { _fnAddData( settings, data[i] ); } _fnReDraw( settings, holdPosition ); _fnInitComplete( settings ); _fnProcessingDisplay( settings, false ); } ); } }; /** * Get the JSON response from the last Ajax request that DataTables made to the * server. Note that this returns the JSON from the first table in the current * context. * * @return {object} JSON received from the server. */ _api_register( 'ajax.json()', function () { var ctx = this.context; if ( ctx.length > 0 ) { return ctx[0].json; } // else return undefined; } ); /** * Get the data submitted in the last Ajax request */ _api_register( 'ajax.params()', function () { var ctx = this.context; if ( ctx.length > 0 ) { return ctx[0].oAjaxData; } // else return undefined; } ); /** * Reload tables from the Ajax data source. Note that this function will * automatically re-draw the table when the remote data has been loaded. * * @param {boolean} [reset=true] Reset (default) or hold the current paging * position. A full re-sort and re-filter is performed when this method is * called, which is why the pagination reset is the default action. * @returns {DataTables.Api} this */ _api_register( 'ajax.reload()', function ( callback, resetPaging ) { return this.iterator( 'table', function (settings) { __reload( settings, resetPaging===false, callback ); } ); } ); /** * Get the current Ajax URL. Note that this returns the URL from the first * table in the current context. * * @return {string} Current Ajax source URL *//** * Set the Ajax URL. Note that this will set the URL for all tables in the * current context. * * @param {string} url URL to set. * @returns {DataTables.Api} this */ _api_register( 'ajax.url()', function ( url ) { var ctx = this.context; if ( url === undefined ) { // get if ( ctx.length === 0 ) { return undefined; } ctx = ctx[0]; return $.isPlainObject( ctx.ajax ) ? ctx.ajax.url : ctx.ajax; } // set return this.iterator( 'table', function ( settings ) { if ( $.isPlainObject( settings.ajax ) ) { settings.ajax.url = url; } else { settings.ajax = url; } } ); } ); /** * Load data from the newly set Ajax URL. Note that this method is only * available when `ajax.url()` is used to set a URL. Additionally, this method * has the same effect as calling `ajax.reload()` but is provided for * convenience when setting a new URL. Like `ajax.reload()` it will * automatically redraw the table once the remote data has been loaded. * * @returns {DataTables.Api} this */ _api_register( 'ajax.url().load()', function ( callback, resetPaging ) { // Same as a reload, but makes sense to present it for easy access after a // url change return this.iterator( 'table', function ( ctx ) { __reload( ctx, resetPaging===false, callback ); } ); } ); var _selector_run = function ( type, selector, selectFn, settings, opts ) { var out = [], res, a, i, ien, j, jen, selectorType = typeof selector; // Can't just check for isArray here, as an API or jQuery instance might be // given with their array like look if ( ! selector || selectorType === 'string' || selectorType === 'function' || selector.length === undefined ) { selector = [ selector ]; } for ( i=0, ien=selector.length ; i<ien ; i++ ) { // Only split on simple strings - complex expressions will be jQuery selectors a = selector[i] && selector[i].split && ! selector[i].match(/[[(:]/) ? selector[i].split(',') : [ selector[i] ]; for ( j=0, jen=a.length ; j<jen ; j++ ) { res = selectFn( typeof a[j] === 'string' ? (a[j]).trim() : a[j] ); // Remove empty items res = res.filter( function (item) { return item !== null && item !== undefined; }); if ( res && res.length ) { out = out.concat( res ); } } } // selector extensions var ext = _ext.selector[ type ]; if ( ext.length ) { for ( i=0, ien=ext.length ; i<ien ; i++ ) { out = ext[i]( settings, opts, out ); } } return _unique( out ); }; var _selector_opts = function ( opts ) { if ( ! opts ) { opts = {}; } // Backwards compatibility for 1.9- which used the terminology filter rather // than search if ( opts.filter && opts.search === undefined ) { opts.search = opts.filter; } return $.extend( { search: 'none', order: 'current', page: 'all' }, opts ); }; // Reduce the API instance to the first item found var _selector_first = function ( old ) { let inst = new _Api(old.context[0]); // Use a push rather than passing to the constructor, since it will // merge arrays down automatically, which isn't what is wanted here if (old.length) { inst.push( old[0] ); } inst.selector = old.selector; // Limit to a single row / column / cell if (inst.length && inst[0].length > 1) { inst[0].splice(1); } return inst; }; var _selector_row_indexes = function ( settings, opts ) { var i, ien, tmp, a=[], displayFiltered = settings.aiDisplay, displayMaster = settings.aiDisplayMaster; var search = opts.search, // none, applied, removed order = opts.order, // applied, current, index (original - compatibility with 1.9) page = opts.page; // all, current if ( _fnDataSource( settings ) == 'ssp' ) { // In server-side processing mode, most options are irrelevant since // rows not shown don't exist and the index order is the applied order // Removed is a special case - for consistency just return an empty // array return search === 'removed' ? [] : _range( 0, displayMaster.length ); } if ( page == 'current' ) { // Current page implies that order=current and filter=applied, since it is // fairly senseless otherwise, regardless of what order and search actually // are for ( i=settings._iDisplayStart, ien=settings.fnDisplayEnd() ; i<ien ; i++ ) { a.push( displayFiltered[i] ); } } else if ( order == 'current' || order == 'applied' ) { if ( search == 'none') { a = displayMaster.slice(); } else if ( search == 'applied' ) { a = displayFiltered.slice(); } else if ( search == 'removed' ) { // O(n+m) solution by creating a hash map var displayFilteredMap = {}; for ( i=0, ien=displayFiltered.length ; i<ien ; i++ ) { displayFilteredMap[displayFiltered[i]] = null; } displayMaster.forEach(function (item) { if (! Object.prototype.hasOwnProperty.call(displayFilteredMap, item)) { a.push(item); } }); } } else if ( order == 'index' || order == 'original' ) { for ( i=0, ien=settings.aoData.length ; i<ien ; i++ ) { if (! settings.aoData[i]) { continue; } if ( search == 'none' ) { a.push( i ); } else { // applied | removed tmp = displayFiltered.indexOf(i); if ((tmp === -1 && search == 'removed') || (tmp >= 0 && search == 'applied') ) { a.push( i ); } } } } else if ( typeof order === 'number' ) { // Order the rows by the given column var ordered = _fnSort(settings, order, 'asc'); if (search === 'none') { a = ordered; } else { // applied | removed for (i=0; i<ordered.length; i++) { tmp = displayFiltered.indexOf(ordered[i]); if ((tmp === -1 && search == 'removed') || (tmp >= 0 && search == 'applied') ) { a.push( ordered[i] ); } } } } return a; }; /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Rows * * {} - no selector - use all available rows * {integer} - row aoData index * {node} - TR node * {string} - jQuery selector to apply to the TR elements * {array} - jQuery array of nodes, or simply an array of TR nodes * */ var __row_selector = function ( settings, selector, opts ) { var rows; var run = function ( sel ) { var selInt = _intVal( sel ); var aoData = settings.aoData; // Short cut - selector is a number and no options provided (default is // all records, so no need to check if the index is in there, since it // must be - dev error if the index doesn't exist). if ( selInt !== null && ! opts ) { return [ selInt ]; } if ( ! rows ) { rows = _selector_row_indexes( settings, opts ); } if ( selInt !== null && rows.indexOf(selInt) !== -1 ) { // Selector - integer return [ selInt ]; } else if ( sel === null || sel === undefined || sel === '' ) { // Selector - none return rows; } // Selector - function if ( typeof sel === 'function' ) { return rows.map( function (idx) { var row = aoData[ idx ]; return sel( idx, row._aData, row.nTr ) ? idx : null; } ); } // Selector - node if ( sel.nodeName ) { var rowIdx = sel._DT_RowIndex; // Property added by DT for fast lookup var cellIdx = sel._DT_CellIndex; if ( rowIdx !== undefined ) { // Make sure that the row is actually still present in the table return aoData[ rowIdx ] && aoData[ rowIdx ].nTr === sel ? [ rowIdx ] : []; } else if ( cellIdx ) { return aoData[ cellIdx.row ] && aoData[ cellIdx.row ].nTr === sel.parentNode ? [ cellIdx.row ] : []; } else { var host = $(sel).closest('*[data-dt-row]'); return host.length ? [ host.data('dt-row') ] : []; } } // ID selector. Want to always be able to select rows by id, regardless // of if the tr element has been created or not, so can't rely upon // jQuery here - hence a custom implementation. This does not match // Sizzle's fast selector or HTML4 - in HTML5 the ID can be anything, // but to select it using a CSS selector engine (like Sizzle or // querySelect) it would need to need to be escaped for some characters. // DataTables simplifies this for row selectors since you can select // only a row. A # indicates an id any anything that follows is the id - // unescaped. if ( typeof sel === 'string' && sel.charAt(0) === '#' ) { // get row index from id var rowObj = settings.aIds[ sel.replace( /^#/, '' ) ]; if ( rowObj !== undefined ) { return [ rowObj.idx ]; } // need to fall through to jQuery in case there is DOM id that // matches } // Get nodes in the order from the `rows` array with null values removed var nodes = _removeEmpty( _pluck_order( settings.aoData, rows, 'nTr' ) ); // Selector - jQuery selector string, array of nodes or jQuery object/ // As jQuery's .filter() allows jQuery objects to be passed in filter, // it also allows arrays, so this will cope with all three options return $(nodes) .filter( sel ) .map( function () { return this._DT_RowIndex; } ) .toArray(); }; var matched = _selector_run( 'row', selector, run, settings, opts ); if (opts.order === 'current' || opts.order === 'applied') { _fnSortDisplay(settings, matched); } return matched; }; _api_register( 'rows()', function ( selector, opts ) { // argument shifting if ( selector === undefined ) { selector = ''; } else if ( $.isPlainObject( selector ) ) { opts = selector; selector = ''; } opts = _selector_opts( opts ); var inst = this.iterator( 'table', function ( settings ) { return __row_selector( settings, selector, opts ); }, 1 ); // Want argument shifting here and in __row_selector? inst.selector.rows = selector; inst.selector.opts = opts; return inst; } ); _api_register( 'rows().nodes()', function () { return this.iterator( 'row', function ( settings, row ) { return settings.aoData[ row ].nTr || undefined; }, 1 ); } ); _api_register( 'rows().data()', function () { return this.iterator( true, 'rows', function ( settings, rows ) { return _pluck_order( settings.aoData, rows, '_aData' ); }, 1 ); } ); _api_registerPlural( 'rows().cache()', 'row().cache()', function ( type ) { return this.iterator( 'row', function ( settings, row ) { var r = settings.aoData[ row ]; return type === 'search' ? r._aFilterData : r._aSortData; }, 1 ); } ); _api_registerPlural( 'rows().invalidate()', 'row().invalidate()', function ( src ) { return this.iterator( 'row', function ( settings, row ) { _fnInvalidate( settings, row, src ); } ); } ); _api_registerPlural( 'rows().indexes()', 'row().index()', function () { return this.iterator( 'row', function ( settings, row ) { return row; }, 1 ); } ); _api_registerPlural( 'rows().ids()', 'row().id()', function ( hash ) { var a = []; var context = this.context; // `iterator` will drop undefined values, but in this case we want them for ( var i=0, ien=context.length ; i<ien ; i++ ) { for ( var j=0, jen=this[i].length ; j<jen ; j++ ) { var id = context[i].rowIdFn( context[i].aoData[ this[i][j] ]._aData ); a.push( (hash === true ? '#' : '' )+ id ); } } return new _Api( context, a ); } ); _api_registerPlural( 'rows().remove()', 'row().remove()', function () { this.iterator( 'row', function ( settings, row ) { var data = settings.aoData; var rowData = data[ row ]; // Delete from the display arrays var idx = settings.aiDisplayMaster.indexOf(row); if (idx !== -1) { settings.aiDisplayMaster.splice(idx, 1); } // For server-side processing tables - subtract the deleted row from the count if ( settings._iRecordsDisplay > 0 ) { settings._iRecordsDisplay--; } // Check for an 'overflow' they case for displaying the table _fnLengthOverflow( settings ); // Remove the row's ID reference if there is one var id = settings.rowIdFn( rowData._aData ); if ( id !== undefined ) { delete settings.aIds[ id ]; } data[row] = null; } ); return this; } ); _api_register( 'rows.add()', function ( rows ) { var newRows = this.iterator( 'table', function ( settings ) { var row, i, ien; var out = []; for ( i=0, ien=rows.length ; i<ien ; i++ ) { row = rows[i]; if ( row.nodeName && row.nodeName.toUpperCase() === 'TR' ) { out.push( _fnAddTr( settings, row )[0] ); } else { out.push( _fnAddData( settings, row ) ); } } return out; }, 1 ); // Return an Api.rows() extended instance, so rows().nodes() etc can be used var modRows = this.rows( -1 ); modRows.pop(); modRows.push.apply(modRows, newRows); return modRows; } ); /** * */ _api_register( 'row()', function ( selector, opts ) { return _selector_first( this.rows( selector, opts ) ); } ); _api_register( 'row().data()', function ( data ) { var ctx = this.context; if ( data === undefined ) { // Get return ctx.length && this.length && this[0].length ? ctx[0].aoData[ this[0] ]._aData : undefined; } // Set var row = ctx[0].aoData[ this[0] ]; row._aData = data; // If the DOM has an id, and the data source is an array if ( Array.isArray( data ) && row.nTr && row.nTr.id ) { _fnSetObjectDataFn( ctx[0].rowId )( data, row.nTr.id ); } // Automatically invalidate _fnInvalidate( ctx[0], this[0], 'data' ); return this; } ); _api_register( 'row().node()', function () { var ctx = this.context; if (ctx.length && this.length && this[0].length) { var row = ctx[0].aoData[ this[0] ]; if (row && row.nTr) { return row.nTr; } } return null; } ); _api_register( 'row.add()', function ( row ) { // Allow a jQuery object to be passed in - only a single row is added from // it though - the first element in the set if ( row instanceof $ && row.length ) { row = row[0]; } var rows = this.iterator( 'table', function ( settings ) { if ( row.nodeName && row.nodeName.toUpperCase() === 'TR' ) { return _fnAddTr( settings, row )[0]; } return _fnAddData( settings, row ); } ); // Return an Api.rows() extended instance, with the newly added row selected return this.row( rows[0] ); } ); $(document).on('plugin-init.dt', function (e, context) { var api = new _Api( context ); api.on( 'stateSaveParams.DT', function ( e, settings, d ) { // This could be more compact with the API, but it is a lot faster as a simple // internal loop var idFn = settings.rowIdFn; var rows = settings.aiDisplayMaster; var ids = []; for (var i=0 ; i<rows.length ; i++) { var rowIdx = rows[i]; var data = settings.aoData[rowIdx]; if (data._detailsShow) { ids.push( '#' + idFn(data._aData) ); } } d.childRows = ids; }); // For future state loads (e.g. with StateRestore) api.on( 'stateLoaded.DT', function (e, settings, state) { __details_state_load( api, state ); }); // And the initial load state __details_state_load( api, api.state.loaded() ); }); var __details_state_load = function (api, state) { if ( state && state.childRows ) { api .rows( state.childRows.map(function (id) { // Escape any `:` characters from the row id. Accounts for // already escaped characters. return id.replace(/([^:\\]*(?:\\.[^:\\]*)*):/g, "$1\\:"); }) ) .every( function () { _fnCallbackFire( api.settings()[0], null, 'requestChild', [ this ] ) }); } } var __details_add = function ( ctx, row, data, klass ) { // Convert to array of TR elements var rows = []; var addRow = function ( r, k ) { // Recursion to allow for arrays of jQuery objects if ( Array.isArray( r ) || r instanceof $ ) { for ( var i=0, ien=r.length ; i<ien ; i++ ) { addRow( r[i], k ); } return; } // If we get a TR element, then just add it directly - up to the dev // to add the correct number of columns etc if ( r.nodeName && r.nodeName.toLowerCase() === 'tr' ) { r.setAttribute( 'data-dt-row', row.idx ); rows.push( r ); } else { // Otherwise create a row with a wrapper var created = $('<tr><td></td></tr>') .attr( 'data-dt-row', row.idx ) .addClass( k ); $('td', created) .addClass( k ) .html( r )[0].colSpan = _fnVisbleColumns( ctx ); rows.push( created[0] ); } }; addRow( data, klass ); if ( row._details ) { row._details.detach(); } row._details = $(rows); // If the children were already shown, that state should be retained if ( row._detailsShow ) { row._details.insertAfter( row.nTr ); } }; // Make state saving of child row details async to allow them to be batch processed var __details_state = DataTable.util.throttle( function (ctx) { _fnSaveState( ctx[0] ) }, 500 ); var __details_remove = function ( api, idx ) { var ctx = api.context; if ( ctx.length ) { var row = ctx[0].aoData[ idx !== undefined ? idx : api[0] ]; if ( row && row._details ) { row._details.remove(); row._detailsShow = undefined; row._details = undefined; $( row.nTr ).removeClass( 'dt-hasChild' ); __details_state( ctx ); } } }; var __details_display = function ( api, show ) { var ctx = api.context; if ( ctx.length && api.length ) { var row = ctx[0].aoData[ api[0] ]; if ( row._details ) { row._detailsShow = show; if ( show ) { row._details.insertAfter( row.nTr ); $( row.nTr ).addClass( 'dt-hasChild' ); } else { row._details.detach(); $( row.nTr ).removeClass( 'dt-hasChild' ); } _fnCallbackFire( ctx[0], null, 'childRow', [ show, api.row( api[0] ) ] ) __details_events( ctx[0] ); __details_state( ctx ); } } }; var __details_events = function ( settings ) { var api = new _Api( settings ); var namespace = '.dt.DT_details'; var drawEvent = 'draw'+namespace; var colvisEvent = 'column-sizing'+namespace; var destroyEvent = 'destroy'+namespace; var data = settings.aoData; api.off( drawEvent +' '+ colvisEvent +' '+ destroyEvent ); if ( _pluck( data, '_details' ).length > 0 ) { // On each draw, insert the required elements into the document api.on( drawEvent, function ( e, ctx ) { if ( settings !== ctx ) { return; } api.rows( {page:'current'} ).eq(0).each( function (idx) { // Internal data grab var row = data[ idx ]; if ( row._detailsShow ) { row._details.insertAfter( row.nTr ); } } ); } ); // Column visibility change - update the colspan api.on( colvisEvent, function ( e, ctx ) { if ( settings !== ctx ) { return; } // Update the colspan for the details rows (note, only if it already has // a colspan) var row, visible = _fnVisbleColumns( ctx ); for ( var i=0, ien=data.length ; i<ien ; i++ ) { row = data[i]; if ( row && row._details ) { row._details.each(function () { var el = $(this).children('td'); if (el.length == 1) { el.attr('colspan', visible); } }); } } } ); // Table destroyed - nuke any child rows api.on( destroyEvent, function ( e, ctx ) { if ( settings !== ctx ) { return; } for ( var i=0, ien=data.length ; i<ien ; i++ ) { if ( data[i] && data[i]._details ) { __details_remove( api, i ); } } } ); } }; // Strings for the method names to help minification var _emp = ''; var _child_obj = _emp+'row().child'; var _child_mth = _child_obj+'()'; // data can be: // tr // string // jQuery or array of any of the above _api_register( _child_mth, function ( data, klass ) { var ctx = this.context; if ( data === undefined ) { // get return ctx.length && this.length && ctx[0].aoData[ this[0] ] ? ctx[0].aoData[ this[0] ]._details : undefined; } else if ( data === true ) { // show this.child.show(); } else if ( data === false ) { // remove __details_remove( this ); } else if ( ctx.length && this.length ) { // set __details_add( ctx[0], ctx[0].aoData[ this[0] ], data, klass ); } return this; } ); _api_register( [ _child_obj+'.show()', _child_mth+'.show()' // only when `child()` was called with parameters (without ], function () { // it returns an object and this method is not executed) __details_display( this, true ); return this; } ); _api_register( [ _child_obj+'.hide()', _child_mth+'.hide()' // only when `child()` was called with parameters (without ], function () { // it returns an object and this method is not executed) __details_display( this, false ); return this; } ); _api_register( [ _child_obj+'.remove()', _child_mth+'.remove()' // only when `child()` was called with parameters (without ], function () { // it returns an object and this method is not executed) __details_remove( this ); return this; } ); _api_register( _child_obj+'.isShown()', function () { var ctx = this.context; if ( ctx.length && this.length && ctx[0].aoData[ this[0] ] ) { // _detailsShown as false or undefined will fall through to return false return ctx[0].aoData[ this[0] ]._detailsShow || false; } return false; } ); /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Columns * * {integer} - column index (>=0 count from left, <0 count from right) * "{integer}:visIdx" - visible column index (i.e. translate to column index) (>=0 count from left, <0 count from right) * "{integer}:visible" - alias for {integer}:visIdx (>=0 count from left, <0 count from right) * "{string}:name" - column name * "{string}" - jQuery selector on column header nodes * */ // can be an array of these items, comma separated list, or an array of comma // separated lists var __re_column_selector = /^([^:]+)?:(name|title|visIdx|visible)$/; // r1 and r2 are redundant - but it means that the parameters match for the // iterator callback in columns().data() var __columnData = function ( settings, column, r1, r2, rows, type ) { var a = []; for ( var row=0, ien=rows.length ; row<ien ; row++ ) { a.push( _fnGetCellData( settings, rows[row], column, type ) ); } return a; }; var __column_header = function ( settings, column, row ) { var header = settings.aoHeader; var target = row !== undefined ? row : settings.bSortCellsTop // legacy support ? 0 : header.length - 1; return header[target][column].cell; }; var __column_selector = function ( settings, selector, opts ) { var columns = settings.aoColumns, names = _pluck( columns, 'sName' ), titles = _pluck( columns, 'sTitle' ), cells = DataTable.util.get('[].[].cell')(settings.aoHeader), nodes = _unique( _flatten([], cells) ); var run = function ( s ) { var selInt = _intVal( s ); // Selector - all if ( s === '' ) { return _range( columns.length ); } // Selector - index if ( selInt !== null ) { return [ selInt >= 0 ? selInt : // Count from left columns.length + selInt // Count from right (+ because its a negative value) ]; } // Selector = function if ( typeof s === 'function' ) { var rows = _selector_row_indexes( settings, opts ); return columns.map(function (col, idx) { return s( idx, __columnData( settings, idx, 0, 0, rows ), __column_header( settings, idx ) ) ? idx : null; }); } // jQuery or string selector var match = typeof s === 'string' ? s.match( __re_column_selector ) : ''; if ( match ) { switch( match[2] ) { case 'visIdx': case 'visible': if (match[1]) { var idx = parseInt( match[1], 10 ); // Visible index given, convert to column index if ( idx < 0 ) { // Counting from the right var visColumns = columns.map( function (col,i) { return col.bVisible ? i : null; } ); return [ visColumns[ visColumns.length + idx ] ]; } // Counting from the left return [ _fnVisibleToColumnIndex( settings, idx ) ]; } // `:visible` on its own return columns.map( function (col, i) { return col.bVisible ? i : null; } ); case 'name': // match by name. `names` is column index complete and in order return names.map( function (name, i) { return name === match[1] ? i : null; } ); case 'title': // match by column title return titles.map( function (title, i) { return title === match[1] ? i : null; } ); default: return []; } } // Cell in the table body if ( s.nodeName && s._DT_CellIndex ) { return [ s._DT_CellIndex.column ]; } // jQuery selector on the TH elements for the columns var jqResult = $( nodes ) .filter( s ) .map( function () { return _fnColumnsFromHeader( this ); // `nodes` is column index complete and in order } ) .toArray(); if ( jqResult.length || ! s.nodeName ) { return jqResult; } // Otherwise a node which might have a `dt-column` data attribute, or be // a child or such an element var host = $(s).closest('*[data-dt-column]'); return host.length ? [ host.data('dt-column') ] : []; }; return _selector_run( 'column', selector, run, settings, opts ); }; var __setColumnVis = function ( settings, column, vis ) { var cols = settings.aoColumns, col = cols[ column ], data = settings.aoData, cells, i, ien, tr; // Get if ( vis === undefined ) { return col.bVisible; } // Set // No change if ( col.bVisible === vis ) { return false; } if ( vis ) { // Insert column // Need to decide if we should use appendChild or insertBefore var insertBefore = _pluck(cols, 'bVisible').indexOf(true, column+1); for ( i=0, ien=data.length ; i<ien ; i++ ) { if (data[i]) { tr = data[i].nTr; cells = data[i].anCells; if ( tr ) { // insertBefore can act like appendChild if 2nd arg is null tr.insertBefore( cells[ column ], cells[ insertBefore ] || null ); } } } } else { // Remove column $( _pluck( settings.aoData, 'anCells', column ) ).detach(); } // Common actions col.bVisible = vis; _colGroup(settings); return true; }; _api_register( 'columns()', function ( selector, opts ) { // argument shifting if ( selector === undefined ) { selector = ''; } else if ( $.isPlainObject( selector ) ) { opts = selector; selector = ''; } opts = _selector_opts( opts ); var inst = this.iterator( 'table', function ( settings ) { return __column_selector( settings, selector, opts ); }, 1 ); // Want argument shifting here and in _row_selector? inst.selector.cols = selector; inst.selector.opts = opts; return inst; } ); _api_registerPlural( 'columns().header()', 'column().header()', function ( row ) { return this.iterator( 'column', function (settings, column) { return __column_header(settings, column, row); }, 1 ); } ); _api_registerPlural( 'columns().footer()', 'column().footer()', function ( row ) { return this.iterator( 'column', function ( settings, column ) { var footer = settings.aoFooter; if (! footer.length) { return null; } return settings.aoFooter[row !== undefined ? row : 0][column].cell; }, 1 ); } ); _api_registerPlural( 'columns().data()', 'column().data()', function () { return this.iterator( 'column-rows', __columnData, 1 ); } ); _api_registerPlural( 'columns().render()', 'column().render()', function ( type ) { return this.iterator( 'column-rows', function ( settings, column, i, j, rows ) { return __columnData( settings, column, i, j, rows, type ); }, 1 ); } ); _api_registerPlural( 'columns().dataSrc()', 'column().dataSrc()', function () { return this.iterator( 'column', function ( settings, column ) { return settings.aoColumns[column].mData; }, 1 ); } ); _api_registerPlural( 'columns().cache()', 'column().cache()', function ( type ) { return this.iterator( 'column-rows', function ( settings, column, i, j, rows ) { return _pluck_order( settings.aoData, rows, type === 'search' ? '_aFilterData' : '_aSortData', column ); }, 1 ); } ); _api_registerPlural( 'columns().init()', 'column().init()', function () { return this.iterator( 'column', function ( settings, column ) { return settings.aoColumns[column]; }, 1 ); } ); _api_registerPlural( 'columns().nodes()', 'column().nodes()', function () { return this.iterator( 'column-rows', function ( settings, column, i, j, rows ) { return _pluck_order( settings.aoData, rows, 'anCells', column ) ; }, 1 ); } ); _api_registerPlural( 'columns().titles()', 'column().title()', function (title, row) { return this.iterator( 'column', function ( settings, column ) { // Argument shifting if (typeof title === 'number') { row = title; title = undefined; } var span = $('span.dt-column-title', this.column(column).header(row)); if (title !== undefined) { span.html(title); return this; } return span.html(); }, 1 ); } ); _api_registerPlural( 'columns().types()', 'column().type()', function () { return this.iterator( 'column', function ( settings, column ) { var type = settings.aoColumns[column].sType; // If the type was invalidated, then resolve it. This actually does // all columns at the moment. Would only happen once if getting all // column's data types. if (! type) { _fnColumnTypes(settings); } return type; }, 1 ); } ); _api_registerPlural( 'columns().visible()', 'column().visible()', function ( vis, calc ) { var that = this; var changed = []; var ret = this.iterator( 'column', function ( settings, column ) { if ( vis === undefined ) { return settings.aoColumns[ column ].bVisible; } // else if (__setColumnVis( settings, column, vis )) { changed.push(column); } } ); // Group the column visibility changes if ( vis !== undefined ) { this.iterator( 'table', function ( settings ) { // Redraw the header after changes _fnDrawHead( settings, settings.aoHeader ); _fnDrawHead( settings, settings.aoFooter ); // Update colspan for no records display. Child rows and extensions will use their own // listeners to do this - only need to update the empty table item here if ( ! settings.aiDisplay.length ) { $(settings.nTBody).find('td[colspan]').attr('colspan', _fnVisbleColumns(settings)); } _fnSaveState( settings ); // Second loop once the first is done for events that.iterator( 'column', function ( settings, column ) { if (changed.includes(column)) { _fnCallbackFire( settings, null, 'column-visibility', [settings, column, vis, calc] ); } } ); if ( changed.length && (calc === undefined || calc) ) { that.columns.adjust(); } }); } return ret; } ); _api_registerPlural( 'columns().widths()', 'column().width()', function () { // Injects a fake row into the table for just a moment so the widths can // be read, regardless of colspan in the header and rows being present in // the body var columns = this.columns(':visible').count(); var row = $('<tr>').html('<td>' + Array(columns).join('</td><td>') + '</td>'); $(this.table().body()).append(row); var widths = row.children().map(function () { return $(this).outerWidth(); }); row.remove(); return this.iterator( 'column', function ( settings, column ) { var visIdx = _fnColumnIndexToVisible( settings, column ); return visIdx !== null ? widths[visIdx] : 0; }, 1); } ); _api_registerPlural( 'columns().indexes()', 'column().index()', function ( type ) { return this.iterator( 'column', function ( settings, column ) { return type === 'visible' ? _fnColumnIndexToVisible( settings, column ) : column; }, 1 ); } ); _api_register( 'columns.adjust()', function () { return this.iterator( 'table', function ( settings ) { _fnAdjustColumnSizing( settings ); }, 1 ); } ); _api_register( 'column.index()', function ( type, idx ) { if ( this.context.length !== 0 ) { var ctx = this.context[0]; if ( type === 'fromVisible' || type === 'toData' ) { return _fnVisibleToColumnIndex( ctx, idx ); } else if ( type === 'fromData' || type === 'toVisible' ) { return _fnColumnIndexToVisible( ctx, idx ); } } } ); _api_register( 'column()', function ( selector, opts ) { return _selector_first( this.columns( selector, opts ) ); } ); var __cell_selector = function ( settings, selector, opts ) { var data = settings.aoData; var rows = _selector_row_indexes( settings, opts ); var cells = _removeEmpty( _pluck_order( data, rows, 'anCells' ) ); var allCells = $(_flatten( [], cells )); var row; var columns = settings.aoColumns.length; var a, i, ien, j, o, host; var run = function ( s ) { var fnSelector = typeof s === 'function'; if ( s === null || s === undefined || fnSelector ) { // All cells and function selectors a = []; for ( i=0, ien=rows.length ; i<ien ; i++ ) { row = rows[i]; for ( j=0 ; j<columns ; j++ ) { o = { row: row, column: j }; if ( fnSelector ) { // Selector - function host = data[ row ]; if ( s( o, _fnGetCellData(settings, row, j), host.anCells ? host.anCells[j] : null ) ) { a.push( o ); } } else { // Selector - all a.push( o ); } } } return a; } // Selector - index if ( $.isPlainObject( s ) ) { // Valid cell index and its in the array of selectable rows return s.column !== undefined && s.row !== undefined && rows.indexOf(s.row) !== -1 ? [s] : []; } // Selector - jQuery filtered cells var jqResult = allCells .filter( s ) .map( function (i, el) { return { // use a new object, in case someone changes the values row: el._DT_CellIndex.row, column: el._DT_CellIndex.column }; } ) .toArray(); if ( jqResult.length || ! s.nodeName ) { return jqResult; } // Otherwise the selector is a node, and there is one last option - the // element might be a child of an element which has dt-row and dt-column // data attributes host = $(s).closest('*[data-dt-row]'); return host.length ? [ { row: host.data('dt-row'), column: host.data('dt-column') } ] : []; }; return _selector_run( 'cell', selector, run, settings, opts ); }; _api_register( 'cells()', function ( rowSelector, columnSelector, opts ) { // Argument shifting if ( $.isPlainObject( rowSelector ) ) { // Indexes if ( rowSelector.row === undefined ) { // Selector options in first parameter opts = rowSelector; rowSelector = null; } else { // Cell index objects in first parameter opts = columnSelector; columnSelector = null; } } if ( $.isPlainObject( columnSelector ) ) { opts = columnSelector; columnSelector = null; } // Cell selector if ( columnSelector === null || columnSelector === undefined ) { return this.iterator( 'table', function ( settings ) { return __cell_selector( settings, rowSelector, _selector_opts( opts ) ); } ); } // The default built in options need to apply to row and columns var internalOpts = opts ? { page: opts.page, order: opts.order, search: opts.search } : {}; // Row + column selector var columns = this.columns( columnSelector, internalOpts ); var rows = this.rows( rowSelector, internalOpts ); var i, ien, j, jen; var cellsNoOpts = this.iterator( 'table', function ( settings, idx ) { var a = []; for ( i=0, ien=rows[idx].length ; i<ien ; i++ ) { for ( j=0, jen=columns[idx].length ; j<jen ; j++ ) { a.push( { row: rows[idx][i], column: columns[idx][j] } ); } } return a; }, 1 ); // There is currently only one extension which uses a cell selector extension // It is a _major_ performance drag to run this if it isn't needed, so this is // an extension specific check at the moment var cells = opts && opts.selected ? this.cells( cellsNoOpts, opts ) : cellsNoOpts; $.extend( cells.selector, { cols: columnSelector, rows: rowSelector, opts: opts } ); return cells; } ); _api_registerPlural( 'cells().nodes()', 'cell().node()', function () { return this.iterator( 'cell', function ( settings, row, column ) { var data = settings.aoData[ row ]; return data && data.anCells ? data.anCells[ column ] : undefined; }, 1 ); } ); _api_register( 'cells().data()', function () { return this.iterator( 'cell', function ( settings, row, column ) { return _fnGetCellData( settings, row, column ); }, 1 ); } ); _api_registerPlural( 'cells().cache()', 'cell().cache()', function ( type ) { type = type === 'search' ? '_aFilterData' : '_aSortData'; return this.iterator( 'cell', function ( settings, row, column ) { return settings.aoData[ row ][ type ][ column ]; }, 1 ); } ); _api_registerPlural( 'cells().render()', 'cell().render()', function ( type ) { return this.iterator( 'cell', function ( settings, row, column ) { return _fnGetCellData( settings, row, column, type ); }, 1 ); } ); _api_registerPlural( 'cells().indexes()', 'cell().index()', function () { return this.iterator( 'cell', function ( settings, row, column ) { return { row: row, column: column, columnVisible: _fnColumnIndexToVisible( settings, column ) }; }, 1 ); } ); _api_registerPlural( 'cells().invalidate()', 'cell().invalidate()', function ( src ) { return this.iterator( 'cell', function ( settings, row, column ) { _fnInvalidate( settings, row, src, column ); } ); } ); _api_register( 'cell()', function ( rowSelector, columnSelector, opts ) { return _selector_first( this.cells( rowSelector, columnSelector, opts ) ); } ); _api_register( 'cell().data()', function ( data ) { var ctx = this.context; var cell = this[0]; if ( data === undefined ) { // Get return ctx.length && cell.length ? _fnGetCellData( ctx[0], cell[0].row, cell[0].column ) : undefined; } // Set _fnSetCellData( ctx[0], cell[0].row, cell[0].column, data ); _fnInvalidate( ctx[0], cell[0].row, 'data', cell[0].column ); return this; } ); /** * Get current ordering (sorting) that has been applied to the table. * * @returns {array} 2D array containing the sorting information for the first * table in the current context. Each element in the parent array represents * a column being sorted upon (i.e. multi-sorting with two columns would have * 2 inner arrays). The inner arrays may have 2 or 3 elements. The first is * the column index that the sorting condition applies to, the second is the * direction of the sort (`desc` or `asc`) and, optionally, the third is the * index of the sorting order from the `column.sorting` initialisation array. *//** * Set the ordering for the table. * * @param {integer} order Column index to sort upon. * @param {string} direction Direction of the sort to be applied (`asc` or `desc`) * @returns {DataTables.Api} this *//** * Set the ordering for the table. * * @param {array} order 1D array of sorting information to be applied. * @param {array} [...] Optional additional sorting conditions * @returns {DataTables.Api} this *//** * Set the ordering for the table. * * @param {array} order 2D array of sorting information to be applied. * @returns {DataTables.Api} this */ _api_register( 'order()', function ( order, dir ) { var ctx = this.context; var args = Array.prototype.slice.call( arguments ); if ( order === undefined ) { // get return ctx.length !== 0 ? ctx[0].aaSorting : undefined; } // set if ( typeof order === 'number' ) { // Simple column / direction passed in order = [ [ order, dir ] ]; } else if ( args.length > 1 ) { // Arguments passed in (list of 1D arrays) order = args; } // otherwise a 2D array was passed in return this.iterator( 'table', function ( settings ) { settings.aaSorting = Array.isArray(order) ? order.slice() : order; } ); } ); /** * Attach a sort listener to an element for a given column * * @param {node|jQuery|string} node Identifier for the element(s) to attach the * listener to. This can take the form of a single DOM node, a jQuery * collection of nodes or a jQuery selector which will identify the node(s). * @param {integer} column the column that a click on this node will sort on * @param {function} [callback] callback function when sort is run * @returns {DataTables.Api} this */ _api_register( 'order.listener()', function ( node, column, callback ) { return this.iterator( 'table', function ( settings ) { _fnSortAttachListener(settings, node, {}, column, callback); } ); } ); _api_register( 'order.fixed()', function ( set ) { if ( ! set ) { var ctx = this.context; var fixed = ctx.length ? ctx[0].aaSortingFixed : undefined; return Array.isArray( fixed ) ? { pre: fixed } : fixed; } return this.iterator( 'table', function ( settings ) { settings.aaSortingFixed = $.extend( true, {}, set ); } ); } ); // Order by the selected column(s) _api_register( [ 'columns().order()', 'column().order()' ], function ( dir ) { var that = this; if ( ! dir ) { return this.iterator( 'column', function ( settings, idx ) { var sort = _fnSortFlatten( settings ); for ( var i=0, ien=sort.length ; i<ien ; i++ ) { if ( sort[i].col === idx ) { return sort[i].dir; } } return null; }, 1 ); } else { return this.iterator( 'table', function ( settings, i ) { settings.aaSorting = that[i].map( function (col) { return [ col, dir ]; } ); } ); } } ); _api_registerPlural('columns().orderable()', 'column().orderable()', function ( directions ) { return this.iterator( 'column', function ( settings, idx ) { var col = settings.aoColumns[idx]; return directions ? col.asSorting : col.bSortable; }, 1 ); } ); _api_register( 'processing()', function ( show ) { return this.iterator( 'table', function ( ctx ) { _fnProcessingDisplay( ctx, show ); } ); } ); _api_register( 'search()', function ( input, regex, smart, caseInsen ) { var ctx = this.context; if ( input === undefined ) { // get return ctx.length !== 0 ? ctx[0].oPreviousSearch.search : undefined; } // set return this.iterator( 'table', function ( settings ) { if ( ! settings.oFeatures.bFilter ) { return; } if (typeof regex === 'object') { // New style options to pass to the search builder _fnFilterComplete( settings, $.extend( settings.oPreviousSearch, regex, { search: input } ) ); } else { // Compat for the old options _fnFilterComplete( settings, $.extend( settings.oPreviousSearch, { search: input, regex: regex === null ? false : regex, smart: smart === null ? true : smart, caseInsensitive: caseInsen === null ? true : caseInsen } ) ); } } ); } ); _api_register( 'search.fixed()', function ( name, search ) { var ret = this.iterator( true, 'table', function ( settings ) { var fixed = settings.searchFixed; if (! name) { return Object.keys(fixed) } else if (search === undefined) { return fixed[name]; } else if (search === null) { delete fixed[name]; } else { fixed[name] = search; } return this; } ); return name !== undefined && search === undefined ? ret[0] : ret; } ); _api_registerPlural( 'columns().search()', 'column().search()', function ( input, regex, smart, caseInsen ) { return this.iterator( 'column', function ( settings, column ) { var preSearch = settings.aoPreSearchCols; if ( input === undefined ) { // get return preSearch[ column ].search; } // set if ( ! settings.oFeatures.bFilter ) { return; } if (typeof regex === 'object') { // New style options to pass to the search builder $.extend( preSearch[ column ], regex, { search: input } ); } else { // Old style (with not all options available) $.extend( preSearch[ column ], { search: input, regex: regex === null ? false : regex, smart: smart === null ? true : smart, caseInsensitive: caseInsen === null ? true : caseInsen } ); } _fnFilterComplete( settings, settings.oPreviousSearch ); } ); } ); _api_register([ 'columns().search.fixed()', 'column().search.fixed()' ], function ( name, search ) { var ret = this.iterator( true, 'column', function ( settings, colIdx ) { var fixed = settings.aoColumns[colIdx].searchFixed; if (! name) { return Object.keys(fixed) } else if (search === undefined) { return fixed[name]; } else if (search === null) { delete fixed[name]; } else { fixed[name] = search; } return this; } ); return name !== undefined && search === undefined ? ret[0] : ret; } ); /* * State API methods */ _api_register( 'state()', function ( set, ignoreTime ) { // getter if ( ! set ) { return this.context.length ? this.context[0].oSavedState : null; } var setMutate = $.extend( true, {}, set ); // setter return this.iterator( 'table', function ( settings ) { if ( ignoreTime !== false ) { setMutate.time = +new Date() + 100; } _fnImplementState( settings, setMutate, function(){} ); } ); } ); _api_register( 'state.clear()', function () { return this.iterator( 'table', function ( settings ) { // Save an empty object settings.fnStateSaveCallback.call( settings.oInstance, settings, {} ); } ); } ); _api_register( 'state.loaded()', function () { return this.context.length ? this.context[0].oLoadedState : null; } ); _api_register( 'state.save()', function () { return this.iterator( 'table', function ( settings ) { _fnSaveState( settings ); } ); } ); /** * Set the jQuery or window object to be used by DataTables * * @param {*} module Library / container object * @param {string} [type] Library or container type `lib`, `win` or `datetime`. * If not provided, automatic detection is attempted. */ DataTable.use = function (module, type) { if (type === 'lib' || module.fn) { $ = module; } else if (type == 'win' || module.document) { window = module; document = module.document; } else if (type === 'datetime' || module.type === 'DateTime') { DataTable.DateTime = module; } } /** * CommonJS factory function pass through. This will check if the arguments * given are a window object or a jQuery object. If so they are set * accordingly. * @param {*} root Window * @param {*} jq jQUery * @returns {boolean} Indicator */ DataTable.factory = function (root, jq) { var is = false; // Test if the first parameter is a window object if (root && root.document) { window = root; document = root.document; } // Test if the second parameter is a jQuery object if (jq && jq.fn && jq.fn.jquery) { $ = jq; is = true; } return is; } /** * Provide a common method for plug-ins to check the version of DataTables being * used, in order to ensure compatibility. * * @param {string} version Version string to check for, in the format "X.Y.Z". * Note that the formats "X" and "X.Y" are also acceptable. * @param {string} [version2=current DataTables version] As above, but optional. * If not given the current DataTables version will be used. * @returns {boolean} true if this version of DataTables is greater or equal to * the required version, or false if this version of DataTales is not * suitable * @static * @dtopt API-Static * * @example * alert( $.fn.dataTable.versionCheck( '1.9.0' ) ); */ DataTable.versionCheck = function( version, version2 ) { var aThis = version2 ? version2.split('.') : DataTable.version.split('.'); var aThat = version.split('.'); var iThis, iThat; for ( var i=0, iLen=aThat.length ; i<iLen ; i++ ) { iThis = parseInt( aThis[i], 10 ) || 0; iThat = parseInt( aThat[i], 10 ) || 0; // Parts are the same, keep comparing if (iThis === iThat) { continue; } // Parts are different, return immediately return iThis > iThat; } return true; }; /** * Check if a `<table>` node is a DataTable table already or not. * * @param {node|jquery|string} table Table node, jQuery object or jQuery * selector for the table to test. Note that if more than more than one * table is passed on, only the first will be checked * @returns {boolean} true the table given is a DataTable, or false otherwise * @static * @dtopt API-Static * * @example * if ( ! $.fn.DataTable.isDataTable( '#example' ) ) { * $('#example').dataTable(); * } */ DataTable.isDataTable = function ( table ) { var t = $(table).get(0); var is = false; if ( table instanceof DataTable.Api ) { return true; } $.each( DataTable.settings, function (i, o) { var head = o.nScrollHead ? $('table', o.nScrollHead)[0] : null; var foot = o.nScrollFoot ? $('table', o.nScrollFoot)[0] : null; if ( o.nTable === t || head === t || foot === t ) { is = true; } } ); return is; }; /** * Get all DataTable tables that have been initialised - optionally you can * select to get only currently visible tables. * * @param {boolean} [visible=false] Flag to indicate if you want all (default) * or visible tables only. * @returns {array} Array of `table` nodes (not DataTable instances) which are * DataTables * @static * @dtopt API-Static * * @example * $.each( $.fn.dataTable.tables(true), function () { * $(table).DataTable().columns.adjust(); * } ); */ DataTable.tables = function ( visible ) { var api = false; if ( $.isPlainObject( visible ) ) { api = visible.api; visible = visible.visible; } var a = DataTable.settings .filter( function (o) { return !visible || (visible && $(o.nTable).is(':visible')) ? true : false; } ) .map( function (o) { return o.nTable; }); return api ? new _Api( a ) : a; }; /** * Convert from camel case parameters to Hungarian notation. This is made public * for the extensions to provide the same ability as DataTables core to accept * either the 1.9 style Hungarian notation, or the 1.10+ style camelCase * parameters. * * @param {object} src The model object which holds all parameters that can be * mapped. * @param {object} user The object to convert from camel case to Hungarian. * @param {boolean} force When set to `true`, properties which already have a * Hungarian value in the `user` object will be overwritten. Otherwise they * won't be. */ DataTable.camelToHungarian = _fnCamelToHungarian; /** * */ _api_register( '$()', function ( selector, opts ) { var rows = this.rows( opts ).nodes(), // Get all rows jqRows = $(rows); return $( [].concat( jqRows.filter( selector ).toArray(), jqRows.find( selector ).toArray() ) ); } ); // jQuery functions to operate on the tables $.each( [ 'on', 'one', 'off' ], function (i, key) { _api_register( key+'()', function ( /* event, handler */ ) { var args = Array.prototype.slice.call(arguments); // Add the `dt` namespace automatically if it isn't already present args[0] = args[0].split( /\s/ ).map( function ( e ) { return ! e.match(/\.dt\b/) ? e+'.dt' : e; } ).join( ' ' ); var inst = $( this.tables().nodes() ); inst[key].apply( inst, args ); return this; } ); } ); _api_register( 'clear()', function () { return this.iterator( 'table', function ( settings ) { _fnClearTable( settings ); } ); } ); _api_register( 'error()', function (msg) { return this.iterator( 'table', function ( settings ) { _fnLog( settings, 0, msg ); } ); } ); _api_register( 'settings()', function () { return new _Api( this.context, this.context ); } ); _api_register( 'init()', function () { var ctx = this.context; return ctx.length ? ctx[0].oInit : null; } ); _api_register( 'data()', function () { return this.iterator( 'table', function ( settings ) { return _pluck( settings.aoData, '_aData' ); } ).flatten(); } ); _api_register( 'trigger()', function ( name, args, bubbles ) { return this.iterator( 'table', function ( settings ) { return _fnCallbackFire( settings, null, name, args, bubbles ); } ).flatten(); } ); _api_register( 'ready()', function ( fn ) { var ctx = this.context; // Get status of first table if (! fn) { return ctx.length ? (ctx[0]._bInitComplete || false) : null; } // Function to run either once the table becomes ready or // immediately if it is already ready. return this.tables().every(function () { if (this.context[0]._bInitComplete) { fn.call(this); } else { this.on('init', function () { fn.call(this); }); } } ); } ); _api_register( 'destroy()', function ( remove ) { remove = remove || false; return this.iterator( 'table', function ( settings ) { var classes = settings.oClasses; var table = settings.nTable; var tbody = settings.nTBody; var thead = settings.nTHead; var tfoot = settings.nTFoot; var jqTable = $(table); var jqTbody = $(tbody); var jqWrapper = $(settings.nTableWrapper); var rows = settings.aoData.map( function (r) { return r ? r.nTr : null; } ); var orderClasses = classes.order; // Flag to note that the table is currently being destroyed - no action // should be taken settings.bDestroying = true; // Fire off the destroy callbacks for plug-ins etc _fnCallbackFire( settings, "aoDestroyCallback", "destroy", [settings], true ); // If not being removed from the document, make all columns visible if ( ! remove ) { new _Api( settings ).columns().visible( true ); } // Blitz all `DT` namespaced events (these are internal events, the // lowercase, `dt` events are user subscribed and they are responsible // for removing them jqWrapper.off('.DT').find(':not(tbody *)').off('.DT'); $(window).off('.DT-'+settings.sInstance); // When scrolling we had to break the table up - restore it if ( table != thead.parentNode ) { jqTable.children('thead').detach(); jqTable.append( thead ); } if ( tfoot && table != tfoot.parentNode ) { jqTable.children('tfoot').detach(); jqTable.append( tfoot ); } settings.colgroup.remove(); settings.aaSorting = []; settings.aaSortingFixed = []; _fnSortingClasses( settings ); $('th, td', thead) .removeClass( orderClasses.canAsc + ' ' + orderClasses.canDesc + ' ' + orderClasses.isAsc + ' ' + orderClasses.isDesc ) .css('width', ''); // Add the TR elements back into the table in their original order jqTbody.children().detach(); jqTbody.append( rows ); var orig = settings.nTableWrapper.parentNode; var insertBefore = settings.nTableWrapper.nextSibling; // Remove the DataTables generated nodes, events and classes var removedMethod = remove ? 'remove' : 'detach'; jqTable[ removedMethod ](); jqWrapper[ removedMethod ](); // If we need to reattach the table to the document if ( ! remove && orig ) { // insertBefore acts like appendChild if !arg[1] orig.insertBefore( table, insertBefore ); // Restore the width of the original table - was read from the style property, // so we can restore directly to that jqTable .css( 'width', settings.sDestroyWidth ) .removeClass( classes.table ); } /* Remove the settings object from the settings array */ var idx = DataTable.settings.indexOf(settings); if ( idx !== -1 ) { DataTable.settings.splice( idx, 1 ); } } ); } ); // Add the `every()` method for rows, columns and cells in a compact form $.each( [ 'column', 'row', 'cell' ], function ( i, type ) { _api_register( type+'s().every()', function ( fn ) { var opts = this.selector.opts; var api = this; var inst; var counter = 0; return this.iterator( 'every', function ( settings, selectedIdx, tableIdx ) { inst = api[ type ](selectedIdx, opts); if (type === 'cell') { fn.call(inst, inst[0][0].row, inst[0][0].column, tableIdx, counter); } else { fn.call(inst, selectedIdx, tableIdx, counter); } counter++; } ); } ); } ); // i18n method for extensions to be able to use the language object from the // DataTable _api_register( 'i18n()', function ( token, def, plural ) { var ctx = this.context[0]; var resolved = _fnGetObjectDataFn( token )( ctx.oLanguage ); if ( resolved === undefined ) { resolved = def; } if ( $.isPlainObject( resolved ) ) { resolved = plural !== undefined && resolved[ plural ] !== undefined ? resolved[ plural ] : resolved._; } return typeof resolved === 'string' ? resolved.replace( '%d', plural ) // nb: plural might be undefined, : resolved; } ); /** * Version string for plug-ins to check compatibility. Allowed format is * `a.b.c-d` where: a:int, b:int, c:int, d:string(dev|beta|alpha). `d` is used * only for non-release builds. See https://semver.org/ for more information. * @member * @type string * @default Version number */ DataTable.version = "2.0.8"; /** * Private data store, containing all of the settings objects that are * created for the tables on a given page. * * Note that the `DataTable.settings` object is aliased to * `jQuery.fn.dataTableExt` through which it may be accessed and * manipulated, or `jQuery.fn.dataTable.settings`. * @member * @type array * @default [] * @private */ DataTable.settings = []; /** * Object models container, for the various models that DataTables has * available to it. These models define the objects that are used to hold * the active state and configuration of the table. * @namespace */ DataTable.models = {}; /** * Template object for the way in which DataTables holds information about * search information for the global filter and individual column filters. * @namespace */ DataTable.models.oSearch = { /** * Flag to indicate if the filtering should be case insensitive or not */ "caseInsensitive": true, /** * Applied search term */ "search": "", /** * Flag to indicate if the search term should be interpreted as a * regular expression (true) or not (false) and therefore and special * regex characters escaped. */ "regex": false, /** * Flag to indicate if DataTables is to use its smart filtering or not. */ "smart": true, /** * Flag to indicate if DataTables should only trigger a search when * the return key is pressed. */ "return": false }; /** * Template object for the way in which DataTables holds information about * each individual row. This is the object format used for the settings * aoData array. * @namespace */ DataTable.models.oRow = { /** * TR element for the row */ "nTr": null, /** * Array of TD elements for each row. This is null until the row has been * created. */ "anCells": null, /** * Data object from the original data source for the row. This is either * an array if using the traditional form of DataTables, or an object if * using mData options. The exact type will depend on the passed in * data from the data source, or will be an array if using DOM a data * source. */ "_aData": [], /** * Sorting data cache - this array is ostensibly the same length as the * number of columns (although each index is generated only as it is * needed), and holds the data that is used for sorting each column in the * row. We do this cache generation at the start of the sort in order that * the formatting of the sort data need be done only once for each cell * per sort. This array should not be read from or written to by anything * other than the master sorting methods. */ "_aSortData": null, /** * Per cell filtering data cache. As per the sort data cache, used to * increase the performance of the filtering in DataTables */ "_aFilterData": null, /** * Filtering data cache. This is the same as the cell filtering cache, but * in this case a string rather than an array. This is easily computed with * a join on `_aFilterData`, but is provided as a cache so the join isn't * needed on every search (memory traded for performance) */ "_sFilterRow": null, /** * Denote if the original data source was from the DOM, or the data source * object. This is used for invalidating data, so DataTables can * automatically read data from the original source, unless uninstructed * otherwise. */ "src": null, /** * Index in the aoData array. This saves an indexOf lookup when we have the * object, but want to know the index */ "idx": -1, /** * Cached display value */ displayData: null }; /** * Template object for the column information object in DataTables. This object * is held in the settings aoColumns array and contains all the information that * DataTables needs about each individual column. * * Note that this object is related to {@link DataTable.defaults.column} * but this one is the internal data store for DataTables's cache of columns. * It should NOT be manipulated outside of DataTables. Any configuration should * be done through the initialisation options. * @namespace */ DataTable.models.oColumn = { /** * Column index. */ "idx": null, /** * A list of the columns that sorting should occur on when this column * is sorted. That this property is an array allows multi-column sorting * to be defined for a column (for example first name / last name columns * would benefit from this). The values are integers pointing to the * columns to be sorted on (typically it will be a single integer pointing * at itself, but that doesn't need to be the case). */ "aDataSort": null, /** * Define the sorting directions that are applied to the column, in sequence * as the column is repeatedly sorted upon - i.e. the first value is used * as the sorting direction when the column if first sorted (clicked on). * Sort it again (click again) and it will move on to the next index. * Repeat until loop. */ "asSorting": null, /** * Flag to indicate if the column is searchable, and thus should be included * in the filtering or not. */ "bSearchable": null, /** * Flag to indicate if the column is sortable or not. */ "bSortable": null, /** * Flag to indicate if the column is currently visible in the table or not */ "bVisible": null, /** * Store for manual type assignment using the `column.type` option. This * is held in store so we can manipulate the column's `sType` property. */ "_sManualType": null, /** * Flag to indicate if HTML5 data attributes should be used as the data * source for filtering or sorting. True is either are. */ "_bAttrSrc": false, /** * Developer definable function that is called whenever a cell is created (Ajax source, * etc) or processed for input (DOM source). This can be used as a compliment to mRender * allowing you to modify the DOM element (add background colour for example) when the * element is available. */ "fnCreatedCell": null, /** * Function to get data from a cell in a column. You should <b>never</b> * access data directly through _aData internally in DataTables - always use * the method attached to this property. It allows mData to function as * required. This function is automatically assigned by the column * initialisation method */ "fnGetData": null, /** * Function to set data for a cell in the column. You should <b>never</b> * set the data directly to _aData internally in DataTables - always use * this method. It allows mData to function as required. This function * is automatically assigned by the column initialisation method */ "fnSetData": null, /** * Property to read the value for the cells in the column from the data * source array / object. If null, then the default content is used, if a * function is given then the return from the function is used. */ "mData": null, /** * Partner property to mData which is used (only when defined) to get * the data - i.e. it is basically the same as mData, but without the * 'set' option, and also the data fed to it is the result from mData. * This is the rendering method to match the data method of mData. */ "mRender": null, /** * The class to apply to all TD elements in the table's TBODY for the column */ "sClass": null, /** * When DataTables calculates the column widths to assign to each column, * it finds the longest string in each column and then constructs a * temporary table and reads the widths from that. The problem with this * is that "mmm" is much wider then "iiii", but the latter is a longer * string - thus the calculation can go wrong (doing it properly and putting * it into an DOM object and measuring that is horribly(!) slow). Thus as * a "work around" we provide this option. It will append its value to the * text that is found to be the longest string for the column - i.e. padding. */ "sContentPadding": null, /** * Allows a default value to be given for a column's data, and will be used * whenever a null data source is encountered (this can be because mData * is set to null, or because the data source itself is null). */ "sDefaultContent": null, /** * Name for the column, allowing reference to the column by name as well as * by index (needs a lookup to work by name). */ "sName": null, /** * Custom sorting data type - defines which of the available plug-ins in * afnSortData the custom sorting will use - if any is defined. */ "sSortDataType": 'std', /** * Class to be applied to the header element when sorting on this column */ "sSortingClass": null, /** * Title of the column - what is seen in the TH element (nTh). */ "sTitle": null, /** * Column sorting and filtering type */ "sType": null, /** * Width of the column */ "sWidth": null, /** * Width of the column when it was first "encountered" */ "sWidthOrig": null, /** Cached string which is the longest in the column */ maxLenString: null, /** * Store for named searches */ searchFixed: null }; /* * Developer note: The properties of the object below are given in Hungarian * notation, that was used as the interface for DataTables prior to v1.10, however * from v1.10 onwards the primary interface is camel case. In order to avoid * breaking backwards compatibility utterly with this change, the Hungarian * version is still, internally the primary interface, but is is not documented * - hence the @name tags in each doc comment. This allows a Javascript function * to create a map from Hungarian notation to camel case (going the other direction * would require each property to be listed, which would add around 3K to the size * of DataTables, while this method is about a 0.5K hit). * * Ultimately this does pave the way for Hungarian notation to be dropped * completely, but that is a massive amount of work and will break current * installs (therefore is on-hold until v2). */ /** * Initialisation options that can be given to DataTables at initialisation * time. * @namespace */ DataTable.defaults = { /** * An array of data to use for the table, passed in at initialisation which * will be used in preference to any data which is already in the DOM. This is * particularly useful for constructing tables purely in Javascript, for * example with a custom Ajax call. */ "aaData": null, /** * If ordering is enabled, then DataTables will perform a first pass sort on * initialisation. You can define which column(s) the sort is performed * upon, and the sorting direction, with this variable. The `sorting` array * should contain an array for each column to be sorted initially containing * the column's index and a direction string ('asc' or 'desc'). */ "aaSorting": [[0,'asc']], /** * This parameter is basically identical to the `sorting` parameter, but * cannot be overridden by user interaction with the table. What this means * is that you could have a column (visible or hidden) which the sorting * will always be forced on first - any sorting after that (from the user) * will then be performed as required. This can be useful for grouping rows * together. */ "aaSortingFixed": [], /** * DataTables can be instructed to load data to display in the table from a * Ajax source. This option defines how that Ajax call is made and where to. * * The `ajax` property has three different modes of operation, depending on * how it is defined. These are: * * * `string` - Set the URL from where the data should be loaded from. * * `object` - Define properties for `jQuery.ajax`. * * `function` - Custom data get function * * `string` * -------- * * As a string, the `ajax` property simply defines the URL from which * DataTables will load data. * * `object` * -------- * * As an object, the parameters in the object are passed to * [jQuery.ajax](https://api.jquery.com/jQuery.ajax/) allowing fine control * of the Ajax request. DataTables has a number of default parameters which * you can override using this option. Please refer to the jQuery * documentation for a full description of the options available, although * the following parameters provide additional options in DataTables or * require special consideration: * * * `data` - As with jQuery, `data` can be provided as an object, but it * can also be used as a function to manipulate the data DataTables sends * to the server. The function takes a single parameter, an object of * parameters with the values that DataTables has readied for sending. An * object may be returned which will be merged into the DataTables * defaults, or you can add the items to the object that was passed in and * not return anything from the function. This supersedes `fnServerParams` * from DataTables 1.9-. * * * `dataSrc` - By default DataTables will look for the property `data` (or * `aaData` for compatibility with DataTables 1.9-) when obtaining data * from an Ajax source or for server-side processing - this parameter * allows that property to be changed. You can use Javascript dotted * object notation to get a data source for multiple levels of nesting, or * it my be used as a function. As a function it takes a single parameter, * the JSON returned from the server, which can be manipulated as * required, with the returned value being that used by DataTables as the * data source for the table. * * * `success` - Should not be overridden it is used internally in * DataTables. To manipulate / transform the data returned by the server * use `ajax.dataSrc`, or use `ajax` as a function (see below). * * `function` * ---------- * * As a function, making the Ajax call is left up to yourself allowing * complete control of the Ajax request. Indeed, if desired, a method other * than Ajax could be used to obtain the required data, such as Web storage * or an AIR database. * * The function is given four parameters and no return is required. The * parameters are: * * 1. _object_ - Data to send to the server * 2. _function_ - Callback function that must be executed when the required * data has been obtained. That data should be passed into the callback * as the only parameter * 3. _object_ - DataTables settings object for the table */ "ajax": null, /** * This parameter allows you to readily specify the entries in the length drop * down menu that DataTables shows when pagination is enabled. It can be * either a 1D array of options which will be used for both the displayed * option and the value, or a 2D array which will use the array in the first * position as the value, and the array in the second position as the * displayed options (useful for language strings such as 'All'). * * Note that the `pageLength` property will be automatically set to the * first value given in this array, unless `pageLength` is also provided. */ "aLengthMenu": [ 10, 25, 50, 100 ], /** * The `columns` option in the initialisation parameter allows you to define * details about the way individual columns behave. For a full list of * column options that can be set, please see * {@link DataTable.defaults.column}. Note that if you use `columns` to * define your columns, you must have an entry in the array for every single * column that you have in your table (these can be null if you don't which * to specify any options). */ "aoColumns": null, /** * Very similar to `columns`, `columnDefs` allows you to target a specific * column, multiple columns, or all columns, using the `targets` property of * each object in the array. This allows great flexibility when creating * tables, as the `columnDefs` arrays can be of any length, targeting the * columns you specifically want. `columnDefs` may use any of the column * options available: {@link DataTable.defaults.column}, but it _must_ * have `targets` defined in each object in the array. Values in the `targets` * array may be: * <ul> * <li>a string - class name will be matched on the TH for the column</li> * <li>0 or a positive integer - column index counting from the left</li> * <li>a negative integer - column index counting from the right</li> * <li>the string "_all" - all columns (i.e. assign a default)</li> * </ul> */ "aoColumnDefs": null, /** * Basically the same as `search`, this parameter defines the individual column * filtering state at initialisation time. The array must be of the same size * as the number of columns, and each element be an object with the parameters * `search` and `escapeRegex` (the latter is optional). 'null' is also * accepted and the default will be used. */ "aoSearchCols": [], /** * Enable or disable automatic column width calculation. This can be disabled * as an optimisation (it takes some time to calculate the widths) if the * tables widths are passed in using `columns`. */ "bAutoWidth": true, /** * Deferred rendering can provide DataTables with a huge speed boost when you * are using an Ajax or JS data source for the table. This option, when set to * true, will cause DataTables to defer the creation of the table elements for * each row until they are needed for a draw - saving a significant amount of * time. */ "bDeferRender": true, /** * Replace a DataTable which matches the given selector and replace it with * one which has the properties of the new initialisation object passed. If no * table matches the selector, then the new DataTable will be constructed as * per normal. */ "bDestroy": false, /** * Enable or disable filtering of data. Filtering in DataTables is "smart" in * that it allows the end user to input multiple words (space separated) and * will match a row containing those words, even if not in the order that was * specified (this allow matching across multiple columns). Note that if you * wish to use filtering in DataTables this must remain 'true' - to remove the * default filtering input box and retain filtering abilities, please use * {@link DataTable.defaults.dom}. */ "bFilter": true, /** * Used only for compatiblity with DT1 * @deprecated */ "bInfo": true, /** * Used only for compatiblity with DT1 * @deprecated */ "bLengthChange": true, /** * Enable or disable pagination. */ "bPaginate": true, /** * Enable or disable the display of a 'processing' indicator when the table is * being processed (e.g. a sort). This is particularly useful for tables with * large amounts of data where it can take a noticeable amount of time to sort * the entries. */ "bProcessing": false, /** * Retrieve the DataTables object for the given selector. Note that if the * table has already been initialised, this parameter will cause DataTables * to simply return the object that has already been set up - it will not take * account of any changes you might have made to the initialisation object * passed to DataTables (setting this parameter to true is an acknowledgement * that you understand this). `destroy` can be used to reinitialise a table if * you need. */ "bRetrieve": false, /** * When vertical (y) scrolling is enabled, DataTables will force the height of * the table's viewport to the given height at all times (useful for layout). * However, this can look odd when filtering data down to a small data set, * and the footer is left "floating" further down. This parameter (when * enabled) will cause DataTables to collapse the table's viewport down when * the result set will fit within the given Y height. */ "bScrollCollapse": false, /** * Configure DataTables to use server-side processing. Note that the * `ajax` parameter must also be given in order to give DataTables a * source to obtain the required data for each draw. */ "bServerSide": false, /** * Enable or disable sorting of columns. Sorting of individual columns can be * disabled by the `sortable` option for each column. */ "bSort": true, /** * Enable or display DataTables' ability to sort multiple columns at the * same time (activated by shift-click by the user). */ "bSortMulti": true, /** * Allows control over whether DataTables should use the top (true) unique * cell that is found for a single column, or the bottom (false - default). * This is useful when using complex headers. */ "bSortCellsTop": null, /** * Enable or disable the addition of the classes `sorting\_1`, `sorting\_2` and * `sorting\_3` to the columns which are currently being sorted on. This is * presented as a feature switch as it can increase processing time (while * classes are removed and added) so for large data sets you might want to * turn this off. */ "bSortClasses": true, /** * Enable or disable state saving. When enabled HTML5 `localStorage` will be * used to save table display information such as pagination information, * display length, filtering and sorting. As such when the end user reloads * the page the display display will match what thy had previously set up. */ "bStateSave": false, /** * This function is called when a TR element is created (and all TD child * elements have been inserted), or registered if using a DOM source, allowing * manipulation of the TR element (adding classes etc). */ "fnCreatedRow": null, /** * This function is called on every 'draw' event, and allows you to * dynamically modify any aspect you want about the created DOM. */ "fnDrawCallback": null, /** * Identical to fnHeaderCallback() but for the table footer this function * allows you to modify the table footer on every 'draw' event. */ "fnFooterCallback": null, /** * When rendering large numbers in the information element for the table * (i.e. "Showing 1 to 10 of 57 entries") DataTables will render large numbers * to have a comma separator for the 'thousands' units (e.g. 1 million is * rendered as "1,000,000") to help readability for the end user. This * function will override the default method DataTables uses. */ "fnFormatNumber": function ( toFormat ) { return toFormat.toString().replace( /\B(?=(\d{3})+(?!\d))/g, this.oLanguage.sThousands ); }, /** * This function is called on every 'draw' event, and allows you to * dynamically modify the header row. This can be used to calculate and * display useful information about the table. */ "fnHeaderCallback": null, /** * The information element can be used to convey information about the current * state of the table. Although the internationalisation options presented by * DataTables are quite capable of dealing with most customisations, there may * be times where you wish to customise the string further. This callback * allows you to do exactly that. */ "fnInfoCallback": null, /** * Called when the table has been initialised. Normally DataTables will * initialise sequentially and there will be no need for this function, * however, this does not hold true when using external language information * since that is obtained using an async XHR call. */ "fnInitComplete": null, /** * Called at the very start of each table draw and can be used to cancel the * draw by returning false, any other return (including undefined) results in * the full draw occurring). */ "fnPreDrawCallback": null, /** * This function allows you to 'post process' each row after it have been * generated for each table draw, but before it is rendered on screen. This * function might be used for setting the row class name etc. */ "fnRowCallback": null, /** * Load the table state. With this function you can define from where, and how, the * state of a table is loaded. By default DataTables will load from `localStorage` * but you might wish to use a server-side database or cookies. */ "fnStateLoadCallback": function ( settings ) { try { return JSON.parse( (settings.iStateDuration === -1 ? sessionStorage : localStorage).getItem( 'DataTables_'+settings.sInstance+'_'+location.pathname ) ); } catch (e) { return {}; } }, /** * Callback which allows modification of the saved state prior to loading that state. * This callback is called when the table is loading state from the stored data, but * prior to the settings object being modified by the saved state. Note that for * plug-in authors, you should use the `stateLoadParams` event to load parameters for * a plug-in. */ "fnStateLoadParams": null, /** * Callback that is called when the state has been loaded from the state saving method * and the DataTables settings object has been modified as a result of the loaded state. */ "fnStateLoaded": null, /** * Save the table state. This function allows you to define where and how the state * information for the table is stored By default DataTables will use `localStorage` * but you might wish to use a server-side database or cookies. */ "fnStateSaveCallback": function ( settings, data ) { try { (settings.iStateDuration === -1 ? sessionStorage : localStorage).setItem( 'DataTables_'+settings.sInstance+'_'+location.pathname, JSON.stringify( data ) ); } catch (e) { // noop } }, /** * Callback which allows modification of the state to be saved. Called when the table * has changed state a new state save is required. This method allows modification of * the state saving object prior to actually doing the save, including addition or * other state properties or modification. Note that for plug-in authors, you should * use the `stateSaveParams` event to save parameters for a plug-in. */ "fnStateSaveParams": null, /** * Duration for which the saved state information is considered valid. After this period * has elapsed the state will be returned to the default. * Value is given in seconds. */ "iStateDuration": 7200, /** * Number of rows to display on a single page when using pagination. If * feature enabled (`lengthChange`) then the end user will be able to override * this to a custom setting using a pop-up menu. */ "iDisplayLength": 10, /** * Define the starting point for data display when using DataTables with * pagination. Note that this parameter is the number of records, rather than * the page number, so if you have 10 records per page and want to start on * the third page, it should be "20". */ "iDisplayStart": 0, /** * By default DataTables allows keyboard navigation of the table (sorting, paging, * and filtering) by adding a `tabindex` attribute to the required elements. This * allows you to tab through the controls and press the enter key to activate them. * The tabindex is default 0, meaning that the tab follows the flow of the document. * You can overrule this using this parameter if you wish. Use a value of -1 to * disable built-in keyboard navigation. */ "iTabIndex": 0, /** * Classes that DataTables assigns to the various components and features * that it adds to the HTML table. This allows classes to be configured * during initialisation in addition to through the static * {@link DataTable.ext.oStdClasses} object). */ "oClasses": {}, /** * All strings that DataTables uses in the user interface that it creates * are defined in this object, allowing you to modified them individually or * completely replace them all as required. */ "oLanguage": { /** * Strings that are used for WAI-ARIA labels and controls only (these are not * actually visible on the page, but will be read by screenreaders, and thus * must be internationalised as well). */ "oAria": { /** * ARIA label that is added to the table headers when the column may be sorted */ "orderable": ": Activate to sort", /** * ARIA label that is added to the table headers when the column is currently being sorted */ "orderableReverse": ": Activate to invert sorting", /** * ARIA label that is added to the table headers when the column is currently being * sorted and next step is to remove sorting */ "orderableRemove": ": Activate to remove sorting", paginate: { first: 'First', last: 'Last', next: 'Next', previous: 'Previous' } }, /** * Pagination string used by DataTables for the built-in pagination * control types. */ "oPaginate": { /** * Label and character for first page button («) */ "sFirst": "\u00AB", /** * Last page button (») */ "sLast": "\u00BB", /** * Next page button (›) */ "sNext": "\u203A", /** * Previous page button (‹) */ "sPrevious": "\u2039", }, /** * Plural object for the data type the table is showing */ entries: { _: "entries", 1: "entry" }, /** * This string is shown in preference to `zeroRecords` when the table is * empty of data (regardless of filtering). Note that this is an optional * parameter - if it is not given, the value of `zeroRecords` will be used * instead (either the default or given value). */ "sEmptyTable": "No data available in table", /** * This string gives information to the end user about the information * that is current on display on the page. The following tokens can be * used in the string and will be dynamically replaced as the table * display updates. This tokens can be placed anywhere in the string, or * removed as needed by the language requires: * * * `\_START\_` - Display index of the first record on the current page * * `\_END\_` - Display index of the last record on the current page * * `\_TOTAL\_` - Number of records in the table after filtering * * `\_MAX\_` - Number of records in the table without filtering * * `\_PAGE\_` - Current page number * * `\_PAGES\_` - Total number of pages of data in the table */ "sInfo": "Showing _START_ to _END_ of _TOTAL_ _ENTRIES-TOTAL_", /** * Display information string for when the table is empty. Typically the * format of this string should match `info`. */ "sInfoEmpty": "Showing 0 to 0 of 0 _ENTRIES-TOTAL_", /** * When a user filters the information in a table, this string is appended * to the information (`info`) to give an idea of how strong the filtering * is. The variable _MAX_ is dynamically updated. */ "sInfoFiltered": "(filtered from _MAX_ total _ENTRIES-MAX_)", /** * If can be useful to append extra information to the info string at times, * and this variable does exactly that. This information will be appended to * the `info` (`infoEmpty` and `infoFiltered` in whatever combination they are * being used) at all times. */ "sInfoPostFix": "", /** * This decimal place operator is a little different from the other * language options since DataTables doesn't output floating point * numbers, so it won't ever use this for display of a number. Rather, * what this parameter does is modify the sort methods of the table so * that numbers which are in a format which has a character other than * a period (`.`) as a decimal place will be sorted numerically. * * Note that numbers with different decimal places cannot be shown in * the same table and still be sortable, the table must be consistent. * However, multiple different tables on the page can use different * decimal place characters. */ "sDecimal": "", /** * DataTables has a build in number formatter (`formatNumber`) which is * used to format large numbers that are used in the table information. * By default a comma is used, but this can be trivially changed to any * character you wish with this parameter. */ "sThousands": ",", /** * Detail the action that will be taken when the drop down menu for the * pagination length option is changed. The '_MENU_' variable is replaced * with a default select list of 10, 25, 50 and 100, and can be replaced * with a custom select box if required. */ "sLengthMenu": "_MENU_ _ENTRIES_ per page", /** * When using Ajax sourced data and during the first draw when DataTables is * gathering the data, this message is shown in an empty row in the table to * indicate to the end user the the data is being loaded. Note that this * parameter is not used when loading data by server-side processing, just * Ajax sourced data with client-side processing. */ "sLoadingRecords": "Loading...", /** * Text which is displayed when the table is processing a user action * (usually a sort command or similar). */ "sProcessing": "", /** * Details the actions that will be taken when the user types into the * filtering input text box. The variable "_INPUT_", if used in the string, * is replaced with the HTML text box for the filtering input allowing * control over where it appears in the string. If "_INPUT_" is not given * then the input box is appended to the string automatically. */ "sSearch": "Search:", /** * Assign a `placeholder` attribute to the search `input` element * @type string * @default * * @dtopt Language * @name DataTable.defaults.language.searchPlaceholder */ "sSearchPlaceholder": "", /** * All of the language information can be stored in a file on the * server-side, which DataTables will look up if this parameter is passed. * It must store the URL of the language file, which is in a JSON format, * and the object has the same properties as the oLanguage object in the * initialiser object (i.e. the above parameters). Please refer to one of * the example language files to see how this works in action. */ "sUrl": "", /** * Text shown inside the table records when the is no information to be * displayed after filtering. `emptyTable` is shown when there is simply no * information in the table at all (regardless of filtering). */ "sZeroRecords": "No matching records found" }, /** * This parameter allows you to have define the global filtering state at * initialisation time. As an object the `search` parameter must be * defined, but all other parameters are optional. When `regex` is true, * the search string will be treated as a regular expression, when false * (default) it will be treated as a straight string. When `smart` * DataTables will use it's smart filtering methods (to word match at * any point in the data), when false this will not be done. */ "oSearch": $.extend( {}, DataTable.models.oSearch ), /** * Table and control layout. This replaces the legacy `dom` option. */ layout: { topStart: 'pageLength', topEnd: 'search', bottomStart: 'info', bottomEnd: 'paging' }, /** * Legacy DOM layout option */ "sDom": null, /** * Search delay option. This will throttle full table searches that use the * DataTables provided search input element (it does not effect calls to * `dt-api search()`, providing a delay before the search is made. */ "searchDelay": null, /** * DataTables features six different built-in options for the buttons to * display for pagination control: * * * `numbers` - Page number buttons only * * `simple` - 'Previous' and 'Next' buttons only * * 'simple_numbers` - 'Previous' and 'Next' buttons, plus page numbers * * `full` - 'First', 'Previous', 'Next' and 'Last' buttons * * `full_numbers` - 'First', 'Previous', 'Next' and 'Last' buttons, plus page numbers * * `first_last_numbers` - 'First' and 'Last' buttons, plus page numbers */ "sPaginationType": "full_numbers", /** * Enable horizontal scrolling. When a table is too wide to fit into a * certain layout, or you have a large number of columns in the table, you * can enable x-scrolling to show the table in a viewport, which can be * scrolled. This property can be `true` which will allow the table to * scroll horizontally when needed, or any CSS unit, or a number (in which * case it will be treated as a pixel measurement). Setting as simply `true` * is recommended. */ "sScrollX": "", /** * This property can be used to force a DataTable to use more width than it * might otherwise do when x-scrolling is enabled. For example if you have a * table which requires to be well spaced, this parameter is useful for * "over-sizing" the table, and thus forcing scrolling. This property can by * any CSS unit, or a number (in which case it will be treated as a pixel * measurement). */ "sScrollXInner": "", /** * Enable vertical scrolling. Vertical scrolling will constrain the DataTable * to the given height, and enable scrolling for any data which overflows the * current viewport. This can be used as an alternative to paging to display * a lot of data in a small area (although paging and scrolling can both be * enabled at the same time). This property can be any CSS unit, or a number * (in which case it will be treated as a pixel measurement). */ "sScrollY": "", /** * __Deprecated__ The functionality provided by this parameter has now been * superseded by that provided through `ajax`, which should be used instead. * * Set the HTTP method that is used to make the Ajax call for server-side * processing or Ajax sourced data. */ "sServerMethod": "GET", /** * DataTables makes use of renderers when displaying HTML elements for * a table. These renderers can be added or modified by plug-ins to * generate suitable mark-up for a site. For example the Bootstrap * integration plug-in for DataTables uses a paging button renderer to * display pagination buttons in the mark-up required by Bootstrap. * * For further information about the renderers available see * DataTable.ext.renderer */ "renderer": null, /** * Set the data property name that DataTables should use to get a row's id * to set as the `id` property in the node. */ "rowId": "DT_RowId", /** * Caption value */ "caption": null }; _fnHungarianMap( DataTable.defaults ); /* * Developer note - See note in model.defaults.js about the use of Hungarian * notation and camel case. */ /** * Column options that can be given to DataTables at initialisation time. * @namespace */ DataTable.defaults.column = { /** * Define which column(s) an order will occur on for this column. This * allows a column's ordering to take multiple columns into account when * doing a sort or use the data from a different column. For example first * name / last name columns make sense to do a multi-column sort over the * two columns. */ "aDataSort": null, "iDataSort": -1, ariaTitle: '', /** * You can control the default ordering direction, and even alter the * behaviour of the sort handler (i.e. only allow ascending ordering etc) * using this parameter. */ "asSorting": [ 'asc', 'desc', '' ], /** * Enable or disable filtering on the data in this column. */ "bSearchable": true, /** * Enable or disable ordering on this column. */ "bSortable": true, /** * Enable or disable the display of this column. */ "bVisible": true, /** * Developer definable function that is called whenever a cell is created (Ajax source, * etc) or processed for input (DOM source). This can be used as a compliment to mRender * allowing you to modify the DOM element (add background colour for example) when the * element is available. */ "fnCreatedCell": null, /** * This property can be used to read data from any data source property, * including deeply nested objects / properties. `data` can be given in a * number of different ways which effect its behaviour: * * * `integer` - treated as an array index for the data source. This is the * default that DataTables uses (incrementally increased for each column). * * `string` - read an object property from the data source. There are * three 'special' options that can be used in the string to alter how * DataTables reads the data from the source object: * * `.` - Dotted Javascript notation. Just as you use a `.` in * Javascript to read from nested objects, so to can the options * specified in `data`. For example: `browser.version` or * `browser.name`. If your object parameter name contains a period, use * `\\` to escape it - i.e. `first\\.name`. * * `[]` - Array notation. DataTables can automatically combine data * from and array source, joining the data with the characters provided * between the two brackets. For example: `name[, ]` would provide a * comma-space separated list from the source array. If no characters * are provided between the brackets, the original array source is * returned. * * `()` - Function notation. Adding `()` to the end of a parameter will * execute a function of the name given. For example: `browser()` for a * simple function on the data source, `browser.version()` for a * function in a nested property or even `browser().version` to get an * object property if the function called returns an object. Note that * function notation is recommended for use in `render` rather than * `data` as it is much simpler to use as a renderer. * * `null` - use the original data source for the row rather than plucking * data directly from it. This action has effects on two other * initialisation options: * * `defaultContent` - When null is given as the `data` option and * `defaultContent` is specified for the column, the value defined by * `defaultContent` will be used for the cell. * * `render` - When null is used for the `data` option and the `render` * option is specified for the column, the whole data source for the * row is used for the renderer. * * `function` - the function given will be executed whenever DataTables * needs to set or get the data for a cell in the column. The function * takes three parameters: * * Parameters: * * `{array|object}` The data source for the row * * `{string}` The type call data requested - this will be 'set' when * setting data or 'filter', 'display', 'type', 'sort' or undefined * when gathering data. Note that when `undefined` is given for the * type DataTables expects to get the raw data for the object back< * * `{*}` Data to set when the second parameter is 'set'. * * Return: * * The return value from the function is not required when 'set' is * the type of call, but otherwise the return is what will be used * for the data requested. * * Note that `data` is a getter and setter option. If you just require * formatting of data for output, you will likely want to use `render` which * is simply a getter and thus simpler to use. * * Note that prior to DataTables 1.9.2 `data` was called `mDataProp`. The * name change reflects the flexibility of this property and is consistent * with the naming of mRender. If 'mDataProp' is given, then it will still * be used by DataTables, as it automatically maps the old name to the new * if required. */ "mData": null, /** * This property is the rendering partner to `data` and it is suggested that * when you want to manipulate data for display (including filtering, * sorting etc) without altering the underlying data for the table, use this * property. `render` can be considered to be the the read only companion to * `data` which is read / write (then as such more complex). Like `data` * this option can be given in a number of different ways to effect its * behaviour: * * * `integer` - treated as an array index for the data source. This is the * default that DataTables uses (incrementally increased for each column). * * `string` - read an object property from the data source. There are * three 'special' options that can be used in the string to alter how * DataTables reads the data from the source object: * * `.` - Dotted Javascript notation. Just as you use a `.` in * Javascript to read from nested objects, so to can the options * specified in `data`. For example: `browser.version` or * `browser.name`. If your object parameter name contains a period, use * `\\` to escape it - i.e. `first\\.name`. * * `[]` - Array notation. DataTables can automatically combine data * from and array source, joining the data with the characters provided * between the two brackets. For example: `name[, ]` would provide a * comma-space separated list from the source array. If no characters * are provided between the brackets, the original array source is * returned. * * `()` - Function notation. Adding `()` to the end of a parameter will * execute a function of the name given. For example: `browser()` for a * simple function on the data source, `browser.version()` for a * function in a nested property or even `browser().version` to get an * object property if the function called returns an object. * * `object` - use different data for the different data types requested by * DataTables ('filter', 'display', 'type' or 'sort'). The property names * of the object is the data type the property refers to and the value can * defined using an integer, string or function using the same rules as * `render` normally does. Note that an `_` option _must_ be specified. * This is the default value to use if you haven't specified a value for * the data type requested by DataTables. * * `function` - the function given will be executed whenever DataTables * needs to set or get the data for a cell in the column. The function * takes three parameters: * * Parameters: * * {array|object} The data source for the row (based on `data`) * * {string} The type call data requested - this will be 'filter', * 'display', 'type' or 'sort'. * * {array|object} The full data source for the row (not based on * `data`) * * Return: * * The return value from the function is what will be used for the * data requested. */ "mRender": null, /** * Change the cell type created for the column - either TD cells or TH cells. This * can be useful as TH cells have semantic meaning in the table body, allowing them * to act as a header for a row (you may wish to add scope='row' to the TH elements). */ "sCellType": "td", /** * Class to give to each cell in this column. */ "sClass": "", /** * When DataTables calculates the column widths to assign to each column, * it finds the longest string in each column and then constructs a * temporary table and reads the widths from that. The problem with this * is that "mmm" is much wider then "iiii", but the latter is a longer * string - thus the calculation can go wrong (doing it properly and putting * it into an DOM object and measuring that is horribly(!) slow). Thus as * a "work around" we provide this option. It will append its value to the * text that is found to be the longest string for the column - i.e. padding. * Generally you shouldn't need this! */ "sContentPadding": "", /** * Allows a default value to be given for a column's data, and will be used * whenever a null data source is encountered (this can be because `data` * is set to null, or because the data source itself is null). */ "sDefaultContent": null, /** * This parameter is only used in DataTables' server-side processing. It can * be exceptionally useful to know what columns are being displayed on the * client side, and to map these to database fields. When defined, the names * also allow DataTables to reorder information from the server if it comes * back in an unexpected order (i.e. if you switch your columns around on the * client-side, your server-side code does not also need updating). */ "sName": "", /** * Defines a data source type for the ordering which can be used to read * real-time information from the table (updating the internally cached * version) prior to ordering. This allows ordering to occur on user * editable elements such as form inputs. */ "sSortDataType": "std", /** * The title of this column. */ "sTitle": null, /** * The type allows you to specify how the data for this column will be * ordered. Four types (string, numeric, date and html (which will strip * HTML tags before ordering)) are currently available. Note that only date * formats understood by Javascript's Date() object will be accepted as type * date. For example: "Mar 26, 2008 5:03 PM". May take the values: 'string', * 'numeric', 'date' or 'html' (by default). Further types can be adding * through plug-ins. */ "sType": null, /** * Defining the width of the column, this parameter may take any CSS value * (3em, 20px etc). DataTables applies 'smart' widths to columns which have not * been given a specific width through this interface ensuring that the table * remains readable. */ "sWidth": null }; _fnHungarianMap( DataTable.defaults.column ); /** * DataTables settings object - this holds all the information needed for a * given table, including configuration, data and current application of the * table options. DataTables does not have a single instance for each DataTable * with the settings attached to that instance, but rather instances of the * DataTable "class" are created on-the-fly as needed (typically by a * $().dataTable() call) and the settings object is then applied to that * instance. * * Note that this object is related to {@link DataTable.defaults} but this * one is the internal data store for DataTables's cache of columns. It should * NOT be manipulated outside of DataTables. Any configuration should be done * through the initialisation options. */ DataTable.models.oSettings = { /** * Primary features of DataTables and their enablement state. */ "oFeatures": { /** * Flag to say if DataTables should automatically try to calculate the * optimum table and columns widths (true) or not (false). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bAutoWidth": null, /** * Delay the creation of TR and TD elements until they are actually * needed by a driven page draw. This can give a significant speed * increase for Ajax source and Javascript source data, but makes no * difference at all for DOM and server-side processing tables. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bDeferRender": null, /** * Enable filtering on the table or not. Note that if this is disabled * then there is no filtering at all on the table, including fnFilter. * To just remove the filtering input use sDom and remove the 'f' option. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bFilter": null, /** * Used only for compatiblity with DT1 * @deprecated */ "bInfo": true, /** * Used only for compatiblity with DT1 * @deprecated */ "bLengthChange": true, /** * Pagination enabled or not. Note that if this is disabled then length * changing must also be disabled. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bPaginate": null, /** * Processing indicator enable flag whenever DataTables is enacting a * user request - typically an Ajax request for server-side processing. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bProcessing": null, /** * Server-side processing enabled flag - when enabled DataTables will * get all data from the server for every draw - there is no filtering, * sorting or paging done on the client-side. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bServerSide": null, /** * Sorting enablement flag. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSort": null, /** * Multi-column sorting * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSortMulti": null, /** * Apply a class to the columns which are being sorted to provide a * visual highlight or not. This can slow things down when enabled since * there is a lot of DOM interaction. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSortClasses": null, /** * State saving enablement flag. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bStateSave": null }, /** * Scrolling settings for a table. */ "oScroll": { /** * When the table is shorter in height than sScrollY, collapse the * table container down to the height of the table (when true). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bCollapse": null, /** * Width of the scrollbar for the web-browser's platform. Calculated * during table initialisation. */ "iBarWidth": 0, /** * Viewport width for horizontal scrolling. Horizontal scrolling is * disabled if an empty string. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sX": null, /** * Width to expand the table to when using x-scrolling. Typically you * should not need to use this. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. * @deprecated */ "sXInner": null, /** * Viewport height for vertical scrolling. Vertical scrolling is disabled * if an empty string. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sY": null }, /** * Language information for the table. */ "oLanguage": { /** * Information callback function. See * {@link DataTable.defaults.fnInfoCallback} */ "fnInfoCallback": null }, /** * Browser support parameters */ "oBrowser": { /** * Determine if the vertical scrollbar is on the right or left of the * scrolling container - needed for rtl language layout, although not * all browsers move the scrollbar (Safari). */ "bScrollbarLeft": false, /** * Browser scrollbar width */ "barWidth": 0 }, "ajax": null, /** * Array referencing the nodes which are used for the features. The * parameters of this object match what is allowed by sDom - i.e. * <ul> * <li>'l' - Length changing</li> * <li>'f' - Filtering input</li> * <li>'t' - The table!</li> * <li>'i' - Information</li> * <li>'p' - Pagination</li> * <li>'r' - pRocessing</li> * </ul> */ "aanFeatures": [], /** * Store data information - see {@link DataTable.models.oRow} for detailed * information. */ "aoData": [], /** * Array of indexes which are in the current display (after filtering etc) */ "aiDisplay": [], /** * Array of indexes for display - no filtering */ "aiDisplayMaster": [], /** * Map of row ids to data indexes */ "aIds": {}, /** * Store information about each column that is in use */ "aoColumns": [], /** * Store information about the table's header */ "aoHeader": [], /** * Store information about the table's footer */ "aoFooter": [], /** * Store the applied global search information in case we want to force a * research or compare the old search to a new one. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "oPreviousSearch": {}, /** * Store for named searches */ searchFixed: {}, /** * Store the applied search for each column - see * {@link DataTable.models.oSearch} for the format that is used for the * filtering information for each column. */ "aoPreSearchCols": [], /** * Sorting that is applied to the table. Note that the inner arrays are * used in the following manner: * <ul> * <li>Index 0 - column number</li> * <li>Index 1 - current sorting direction</li> * </ul> * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "aaSorting": null, /** * Sorting that is always applied to the table (i.e. prefixed in front of * aaSorting). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "aaSortingFixed": [], /** * If restoring a table - we should restore its width */ "sDestroyWidth": 0, /** * Callback functions array for every time a row is inserted (i.e. on a draw). */ "aoRowCallback": [], /** * Callback functions for the header on each draw. */ "aoHeaderCallback": [], /** * Callback function for the footer on each draw. */ "aoFooterCallback": [], /** * Array of callback functions for draw callback functions */ "aoDrawCallback": [], /** * Array of callback functions for row created function */ "aoRowCreatedCallback": [], /** * Callback functions for just before the table is redrawn. A return of * false will be used to cancel the draw. */ "aoPreDrawCallback": [], /** * Callback functions for when the table has been initialised. */ "aoInitComplete": [], /** * Callbacks for modifying the settings to be stored for state saving, prior to * saving state. */ "aoStateSaveParams": [], /** * Callbacks for modifying the settings that have been stored for state saving * prior to using the stored values to restore the state. */ "aoStateLoadParams": [], /** * Callbacks for operating on the settings object once the saved state has been * loaded */ "aoStateLoaded": [], /** * Cache the table ID for quick access */ "sTableId": "", /** * The TABLE node for the main table */ "nTable": null, /** * Permanent ref to the thead element */ "nTHead": null, /** * Permanent ref to the tfoot element - if it exists */ "nTFoot": null, /** * Permanent ref to the tbody element */ "nTBody": null, /** * Cache the wrapper node (contains all DataTables controlled elements) */ "nTableWrapper": null, /** * Indicate if all required information has been read in */ "bInitialised": false, /** * Information about open rows. Each object in the array has the parameters * 'nTr' and 'nParent' */ "aoOpenRows": [], /** * Dictate the positioning of DataTables' control elements - see * {@link DataTable.model.oInit.sDom}. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sDom": null, /** * Search delay (in mS) */ "searchDelay": null, /** * Which type of pagination should be used. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sPaginationType": "two_button", /** * Number of paging controls on the page. Only used for backwards compatibility */ pagingControls: 0, /** * The state duration (for `stateSave`) in seconds. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "iStateDuration": 0, /** * Array of callback functions for state saving. Each array element is an * object with the following parameters: * <ul> * <li>function:fn - function to call. Takes two parameters, oSettings * and the JSON string to save that has been thus far created. Returns * a JSON string to be inserted into a json object * (i.e. '"param": [ 0, 1, 2]')</li> * <li>string:sName - name of callback</li> * </ul> */ "aoStateSave": [], /** * Array of callback functions for state loading. Each array element is an * object with the following parameters: * <ul> * <li>function:fn - function to call. Takes two parameters, oSettings * and the object stored. May return false to cancel state loading</li> * <li>string:sName - name of callback</li> * </ul> */ "aoStateLoad": [], /** * State that was saved. Useful for back reference */ "oSavedState": null, /** * State that was loaded. Useful for back reference */ "oLoadedState": null, /** * Note if draw should be blocked while getting data */ "bAjaxDataGet": true, /** * The last jQuery XHR object that was used for server-side data gathering. * This can be used for working with the XHR information in one of the * callbacks */ "jqXHR": null, /** * JSON returned from the server in the last Ajax request */ "json": undefined, /** * Data submitted as part of the last Ajax request */ "oAjaxData": undefined, /** * Send the XHR HTTP method - GET or POST (could be PUT or DELETE if * required). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sServerMethod": null, /** * Format numbers for display. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "fnFormatNumber": null, /** * List of options that can be used for the user selectable length menu. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "aLengthMenu": null, /** * Counter for the draws that the table does. Also used as a tracker for * server-side processing */ "iDraw": 0, /** * Indicate if a redraw is being done - useful for Ajax */ "bDrawing": false, /** * Draw index (iDraw) of the last error when parsing the returned data */ "iDrawError": -1, /** * Paging display length */ "_iDisplayLength": 10, /** * Paging start point - aiDisplay index */ "_iDisplayStart": 0, /** * Server-side processing - number of records in the result set * (i.e. before filtering), Use fnRecordsTotal rather than * this property to get the value of the number of records, regardless of * the server-side processing setting. */ "_iRecordsTotal": 0, /** * Server-side processing - number of records in the current display set * (i.e. after filtering). Use fnRecordsDisplay rather than * this property to get the value of the number of records, regardless of * the server-side processing setting. */ "_iRecordsDisplay": 0, /** * The classes to use for the table */ "oClasses": {}, /** * Flag attached to the settings object so you can check in the draw * callback if filtering has been done in the draw. Deprecated in favour of * events. * @deprecated */ "bFiltered": false, /** * Flag attached to the settings object so you can check in the draw * callback if sorting has been done in the draw. Deprecated in favour of * events. * @deprecated */ "bSorted": false, /** * Indicate that if multiple rows are in the header and there is more than * one unique cell per column, if the top one (true) or bottom one (false) * should be used for sorting / title by DataTables. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSortCellsTop": null, /** * Initialisation object that is used for the table */ "oInit": null, /** * Destroy callback functions - for plug-ins to attach themselves to the * destroy so they can clean up markup and events. */ "aoDestroyCallback": [], /** * Get the number of records in the current record set, before filtering */ "fnRecordsTotal": function () { return _fnDataSource( this ) == 'ssp' ? this._iRecordsTotal * 1 : this.aiDisplayMaster.length; }, /** * Get the number of records in the current record set, after filtering */ "fnRecordsDisplay": function () { return _fnDataSource( this ) == 'ssp' ? this._iRecordsDisplay * 1 : this.aiDisplay.length; }, /** * Get the display end point - aiDisplay index */ "fnDisplayEnd": function () { var len = this._iDisplayLength, start = this._iDisplayStart, calc = start + len, records = this.aiDisplay.length, features = this.oFeatures, paginate = features.bPaginate; if ( features.bServerSide ) { return paginate === false || len === -1 ? start + records : Math.min( start+len, this._iRecordsDisplay ); } else { return ! paginate || calc>records || len===-1 ? records : calc; } }, /** * The DataTables object for this table */ "oInstance": null, /** * Unique identifier for each instance of the DataTables object. If there * is an ID on the table node, then it takes that value, otherwise an * incrementing internal counter is used. */ "sInstance": null, /** * tabindex attribute value that is added to DataTables control elements, allowing * keyboard navigation of the table and its controls. */ "iTabIndex": 0, /** * DIV container for the footer scrolling table if scrolling */ "nScrollHead": null, /** * DIV container for the footer scrolling table if scrolling */ "nScrollFoot": null, /** * Last applied sort */ "aLastSort": [], /** * Stored plug-in instances */ "oPlugins": {}, /** * Function used to get a row's id from the row's data */ "rowIdFn": null, /** * Data location where to store a row's id */ "rowId": null, caption: '', captionNode: null, colgroup: null }; /** * Extension object for DataTables that is used to provide all extension * options. * * Note that the `DataTable.ext` object is available through * `jQuery.fn.dataTable.ext` where it may be accessed and manipulated. It is * also aliased to `jQuery.fn.dataTableExt` for historic reasons. * @namespace * @extends DataTable.models.ext */ var extPagination = DataTable.ext.pager; // Paging buttons configuration $.extend( extPagination, { simple: function () { return [ 'previous', 'next' ]; }, full: function () { return [ 'first', 'previous', 'next', 'last' ]; }, numbers: function () { return [ 'numbers' ]; }, simple_numbers: function () { return [ 'previous', 'numbers', 'next' ]; }, full_numbers: function () { return [ 'first', 'previous', 'numbers', 'next', 'last' ]; }, first_last: function () { return ['first', 'last']; }, first_last_numbers: function () { return ['first', 'numbers', 'last']; }, // For testing and plug-ins to use _numbers: _pagingNumbers, // Number of number buttons - legacy, use `numbers` option for paging feature numbers_length: 7 } ); $.extend( true, DataTable.ext.renderer, { pagingButton: { _: function (settings, buttonType, content, active, disabled) { var classes = settings.oClasses.paging; var btnClasses = [classes.button]; var btn; if (active) { btnClasses.push(classes.active); } if (disabled) { btnClasses.push(classes.disabled) } if (buttonType === 'ellipsis') { btn = $('<span class="ellipsis"></span>').html(content)[0]; } else { btn = $('<button>', { class: btnClasses.join(' '), role: 'link', type: 'button' }).html(content); } return { display: btn, clicker: btn } } }, pagingContainer: { _: function (settings, buttons) { // No wrapping element - just append directly to the host return buttons; } } } ); // Common function to remove new lines, strip HTML and diacritic control var _filterString = function (stripHtml, normalize) { return function (str) { if (_empty(str) || typeof str !== 'string') { return str; } str = str.replace( _re_new_lines, " " ); if (stripHtml) { str = _stripHtml(str); } if (normalize) { str = _normalize(str, false); } return str; }; } /* * Public helper functions. These aren't used internally by DataTables, or * called by any of the options passed into DataTables, but they can be used * externally by developers working with DataTables. They are helper functions * to make working with DataTables a little bit easier. */ function __mldFnName(name) { return name.replace(/[\W]/g, '_') } // Common logic for moment, luxon or a date action function __mld( dt, momentFn, luxonFn, dateFn, arg1 ) { if (window.moment) { return dt[momentFn]( arg1 ); } else if (window.luxon) { return dt[luxonFn]( arg1 ); } return dateFn ? dt[dateFn]( arg1 ) : dt; } var __mlWarning = false; function __mldObj (d, format, locale) { var dt; if (window.moment) { dt = window.moment.utc( d, format, locale, true ); if (! dt.isValid()) { return null; } } else if (window.luxon) { dt = format && typeof d === 'string' ? window.luxon.DateTime.fromFormat( d, format ) : window.luxon.DateTime.fromISO( d ); if (! dt.isValid) { return null; } dt.setLocale(locale); } else if (! format) { // No format given, must be ISO dt = new Date(d); } else { if (! __mlWarning) { alert('DataTables warning: Formatted date without Moment.js or Luxon - https://datatables.net/tn/17'); } __mlWarning = true; } return dt; } // Wrapper for date, datetime and time which all operate the same way with the exception of // the output string for auto locale support function __mlHelper (localeString) { return function ( from, to, locale, def ) { // Luxon and Moment support // Argument shifting if ( arguments.length === 0 ) { locale = 'en'; to = null; // means toLocaleString from = null; // means iso8601 } else if ( arguments.length === 1 ) { locale = 'en'; to = from; from = null; } else if ( arguments.length === 2 ) { locale = to; to = from; from = null; } var typeName = 'datetime' + (to ? '-' + __mldFnName(to) : ''); // Add type detection and sorting specific to this date format - we need to be able to identify // date type columns as such, rather than as numbers in extensions. Hence the need for this. if (! DataTable.ext.type.order[typeName]) { DataTable.type(typeName, { detect: function (d) { // The renderer will give the value to type detect as the type! return d === typeName ? typeName : false; }, order: { pre: function (d) { // The renderer gives us Moment, Luxon or Date obects for the sorting, all of which have a // `valueOf` which gives milliseconds epoch return d.valueOf(); } }, className: 'dt-right' }); } return function ( d, type ) { // Allow for a default value if (d === null || d === undefined) { if (def === '--now') { // We treat everything as UTC further down, so no changes are // made, as such need to get the local date / time as if it were // UTC var local = new Date(); d = new Date( Date.UTC( local.getFullYear(), local.getMonth(), local.getDate(), local.getHours(), local.getMinutes(), local.getSeconds() ) ); } else { d = ''; } } if (type === 'type') { // Typing uses the type name for fast matching return typeName; } if (d === '') { return type !== 'sort' ? '' : __mldObj('0000-01-01 00:00:00', null, locale); } // Shortcut. If `from` and `to` are the same, we are using the renderer to // format for ordering, not display - its already in the display format. if ( to !== null && from === to && type !== 'sort' && type !== 'type' && ! (d instanceof Date) ) { return d; } var dt = __mldObj(d, from, locale); if (dt === null) { return d; } if (type === 'sort') { return dt; } var formatted = to === null ? __mld(dt, 'toDate', 'toJSDate', '')[localeString]() : __mld(dt, 'format', 'toFormat', 'toISOString', to); // XSS protection return type === 'display' ? _escapeHtml( formatted ) : formatted; }; } } // Based on locale, determine standard number formatting // Fallback for legacy browsers is US English var __thousands = ','; var __decimal = '.'; if (window.Intl !== undefined) { try { var num = new Intl.NumberFormat().formatToParts(100000.1); for (var i=0 ; i<num.length ; i++) { if (num[i].type === 'group') { __thousands = num[i].value; } else if (num[i].type === 'decimal') { __decimal = num[i].value; } } } catch (e) { // noop } } // Formatted date time detection - use by declaring the formats you are going to use DataTable.datetime = function ( format, locale ) { var typeName = 'datetime-detect-' + __mldFnName(format); if (! locale) { locale = 'en'; } if (! DataTable.ext.type.order[typeName]) { DataTable.type(typeName, { detect: function (d) { var dt = __mldObj(d, format, locale); return d === '' || dt ? typeName : false; }, order: { pre: function (d) { return __mldObj(d, format, locale) || 0; } }, className: 'dt-right' }); } } /** * Helpers for `columns.render`. * * The options defined here can be used with the `columns.render` initialisation * option to provide a display renderer. The following functions are defined: * * * `moment` - Uses the MomentJS library to convert from a given format into another. * This renderer has three overloads: * * 1 parameter: * * `string` - Format to convert to (assumes input is ISO8601 and locale is `en`) * * 2 parameters: * * `string` - Format to convert from * * `string` - Format to convert to. Assumes `en` locale * * 3 parameters: * * `string` - Format to convert from * * `string` - Format to convert to * * `string` - Locale * * `number` - Will format numeric data (defined by `columns.data`) for * display, retaining the original unformatted data for sorting and filtering. * It takes 5 parameters: * * `string` - Thousands grouping separator * * `string` - Decimal point indicator * * `integer` - Number of decimal points to show * * `string` (optional) - Prefix. * * `string` (optional) - Postfix (/suffix). * * `text` - Escape HTML to help prevent XSS attacks. It has no optional * parameters. * * @example * // Column definition using the number renderer * { * data: "salary", * render: $.fn.dataTable.render.number( '\'', '.', 0, '$' ) * } * * @namespace */ DataTable.render = { date: __mlHelper('toLocaleDateString'), datetime: __mlHelper('toLocaleString'), time: __mlHelper('toLocaleTimeString'), number: function ( thousands, decimal, precision, prefix, postfix ) { // Auto locale detection if (thousands === null || thousands === undefined) { thousands = __thousands; } if (decimal === null || decimal === undefined) { decimal = __decimal; } return { display: function ( d ) { if ( typeof d !== 'number' && typeof d !== 'string' ) { return d; } if (d === '' || d === null) { return d; } var negative = d < 0 ? '-' : ''; var flo = parseFloat( d ); var abs = Math.abs(flo); // Scientific notation for large and small numbers if (abs >= 100000000000 || (abs < 0.0001 && abs !== 0) ) { var exp = flo.toExponential(precision).split(/e\+?/); return exp[0] + ' x 10<sup>' + exp[1] + '</sup>'; } // If NaN then there isn't much formatting that we can do - just // return immediately, escaping any HTML (this was supposed to // be a number after all) if ( isNaN( flo ) ) { return _escapeHtml( d ); } flo = flo.toFixed( precision ); d = Math.abs( flo ); var intPart = parseInt( d, 10 ); var floatPart = precision ? decimal+(d - intPart).toFixed( precision ).substring( 2 ): ''; // If zero, then can't have a negative prefix if (intPart === 0 && parseFloat(floatPart) === 0) { negative = ''; } return negative + (prefix||'') + intPart.toString().replace( /\B(?=(\d{3})+(?!\d))/g, thousands ) + floatPart + (postfix||''); } }; }, text: function () { return { display: _escapeHtml, filter: _escapeHtml }; } }; var _extTypes = DataTable.ext.type; // Get / set type DataTable.type = function (name, prop, val) { if (! prop) { return { className: _extTypes.className[name], detect: _extTypes.detect.find(function (fn) { return fn.name === name; }), order: { pre: _extTypes.order[name + '-pre'], asc: _extTypes.order[name + '-asc'], desc: _extTypes.order[name + '-desc'] }, render: _extTypes.render[name], search: _extTypes.search[name] }; } var setProp = function(prop, propVal) { _extTypes[prop][name] = propVal; }; var setDetect = function (fn) { // Wrap to allow the function to return `true` rather than // specifying the type name. var cb = function (d, s) { var ret = fn(d, s); return ret === true ? name : ret; }; Object.defineProperty(cb, "name", {value: name}); var idx = _extTypes.detect.findIndex(function (fn) { return fn.name === name; }); if (idx === -1) { _extTypes.detect.unshift(cb); } else { _extTypes.detect.splice(idx, 1, cb); } }; var setOrder = function (obj) { _extTypes.order[name + '-pre'] = obj.pre; // can be undefined _extTypes.order[name + '-asc'] = obj.asc; // can be undefined _extTypes.order[name + '-desc'] = obj.desc; // can be undefined }; // prop is optional if (val === undefined) { val = prop; prop = null; } if (prop === 'className') { setProp('className', val); } else if (prop === 'detect') { setDetect(val); } else if (prop === 'order') { setOrder(val); } else if (prop === 'render') { setProp('render', val); } else if (prop === 'search') { setProp('search', val); } else if (! prop) { if (val.className) { setProp('className', val.className); } if (val.detect !== undefined) { setDetect(val.detect); } if (val.order) { setOrder(val.order); } if (val.render !== undefined) { setProp('render', val.render); } if (val.search !== undefined) { setProp('search', val.search); } } } // Get a list of types DataTable.types = function () { return _extTypes.detect.map(function (fn) { return fn.name; }); }; // // Built in data types // DataTable.type('string', { detect: function () { return 'string'; }, order: { pre: function ( a ) { // This is a little complex, but faster than always calling toString, // http://jsperf.com/tostring-v-check return _empty(a) ? '' : typeof a === 'string' ? a.toLowerCase() : ! a.toString ? '' : a.toString(); } }, search: _filterString(false, true) }); DataTable.type('html', { detect: function ( d ) { return _empty( d ) || (typeof d === 'string' && d.indexOf('<') !== -1) ? 'html' : null; }, order: { pre: function ( a ) { return _empty(a) ? '' : a.replace ? _stripHtml(a).trim().toLowerCase() : a+''; } }, search: _filterString(true, true) }); DataTable.type('date', { className: 'dt-type-date', detect: function ( d ) { // V8 tries _very_ hard to make a string passed into `Date.parse()` // valid, so we need to use a regex to restrict date formats. Use a // plug-in for anything other than ISO8601 style strings if ( d && !(d instanceof Date) && ! _re_date.test(d) ) { return null; } var parsed = Date.parse(d); return (parsed !== null && !isNaN(parsed)) || _empty(d) ? 'date' : null; }, order: { pre: function ( d ) { var ts = Date.parse( d ); return isNaN(ts) ? -Infinity : ts; } } }); DataTable.type('html-num-fmt', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _htmlNumeric( d, decimal, true ) ? 'html-num-fmt' : null; }, order: { pre: function ( d, s ) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp, _re_html, _re_formatted_numeric ); } }, search: _filterString(true, true) }); DataTable.type('html-num', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _htmlNumeric( d, decimal ) ? 'html-num' : null; }, order: { pre: function ( d, s ) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp, _re_html ); } }, search: _filterString(true, true) }); DataTable.type('num-fmt', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _isNumber( d, decimal, true ) ? 'num-fmt' : null; }, order: { pre: function ( d, s ) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp, _re_formatted_numeric ); } } }); DataTable.type('num', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _isNumber( d, decimal ) ? 'num' : null; }, order: { pre: function (d, s) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp ); } } }); var __numericReplace = function ( d, decimalPlace, re1, re2 ) { if ( d !== 0 && (!d || d === '-') ) { return -Infinity; } var type = typeof d; if (type === 'number' || type === 'bigint') { return d; } // If a decimal place other than `.` is used, it needs to be given to the // function so we can detect it and replace with a `.` which is the only // decimal place Javascript recognises - it is not locale aware. if ( decimalPlace ) { d = _numToDecimal( d, decimalPlace ); } if ( d.replace ) { if ( re1 ) { d = d.replace( re1, '' ); } if ( re2 ) { d = d.replace( re2, '' ); } } return d * 1; }; $.extend( true, DataTable.ext.renderer, { footer: { _: function ( settings, cell, classes ) { cell.addClass(classes.tfoot.cell); } }, header: { _: function ( settings, cell, classes ) { cell.addClass(classes.thead.cell); if (! settings.oFeatures.bSort) { cell.addClass(classes.order.none); } var legacyTop = settings.bSortCellsTop; var headerRows = cell.closest('thead').find('tr'); var rowIdx = cell.parent().index(); // Conditions to not apply the ordering icons if ( // Cells and rows which have the attribute to disable the icons cell.attr('data-dt-order') === 'disable' || cell.parent().attr('data-dt-order') === 'disable' || // Legacy support for `orderCellsTop`. If it is set, then cells // which are not in the top or bottom row of the header (depending // on the value) do not get the sorting classes applied to them (legacyTop === true && rowIdx !== 0) || (legacyTop === false && rowIdx !== headerRows.length - 1) ) { return; } // No additional mark-up required // Attach a sort listener to update on sort - note that using the // `DT` namespace will allow the event to be removed automatically // on destroy, while the `dt` namespaced event is the one we are // listening for $(settings.nTable).on( 'order.dt.DT', function ( e, ctx, sorting ) { if ( settings !== ctx ) { // need to check this this is the host return; // table, not a nested one } var orderClasses = classes.order; var columns = ctx.api.columns( cell ); var col = settings.aoColumns[columns.flatten()[0]]; var orderable = columns.orderable().includes(true); var ariaType = ''; var indexes = columns.indexes(); var sortDirs = columns.orderable(true).flatten(); var orderedColumns = ',' + sorting.map( function (val) { return val.col; } ).join(',') + ','; cell .removeClass( orderClasses.isAsc +' '+ orderClasses.isDesc ) .toggleClass( orderClasses.none, ! orderable ) .toggleClass( orderClasses.canAsc, orderable && sortDirs.includes('asc') ) .toggleClass( orderClasses.canDesc, orderable && sortDirs.includes('desc') ); var sortIdx = orderedColumns.indexOf( ',' + indexes.toArray().join(',') + ',' ); if ( sortIdx !== -1 ) { // Get the ordering direction for the columns under this cell // Note that it is possible for a cell to be asc and desc sorting // (column spanning cells) var orderDirs = columns.order(); cell.addClass( orderDirs.includes('asc') ? orderClasses.isAsc : '' + orderDirs.includes('desc') ? orderClasses.isDesc : '' ); } // The ARIA spec says that only one column should be marked with aria-sort if ( sortIdx === 0 ) { var firstSort = sorting[0]; var sortOrder = col.asSorting; cell.attr('aria-sort', firstSort.dir === 'asc' ? 'ascending' : 'descending'); // Determine if the next click will remove sorting or change the sort ariaType = ! sortOrder[firstSort.index + 1] ? 'Remove' : 'Reverse'; } else { cell.removeAttr('aria-sort'); } cell.attr('aria-label', orderable ? col.ariaTitle + ctx.api.i18n('oAria.orderable' + ariaType) : col.ariaTitle ); if (orderable) { cell.find('.dt-column-title').attr('role', 'button'); cell.attr('tabindex', 0) } } ); } }, layout: { _: function ( settings, container, items ) { var row = $('<div/>') .addClass('dt-layout-row') .appendTo( container ); $.each( items, function (key, val) { var klass = ! val.table ? 'dt-'+key+' ' : ''; if (val.table) { row.addClass('dt-layout-table'); } $('<div/>') .attr({ id: val.id || null, "class": 'dt-layout-cell '+klass+(val.className || '') }) .append( val.contents ) .appendTo( row ); } ); } } } ); DataTable.feature = {}; // Third parameter is internal only! DataTable.feature.register = function ( name, cb, legacy ) { DataTable.ext.features[ name ] = cb; if (legacy) { _ext.feature.push({ cFeature: legacy, fnInit: cb }); } }; DataTable.feature.register( 'info', function ( settings, opts ) { // For compatibility with the legacy `info` top level option if (! settings.oFeatures.bInfo) { return null; } var lang = settings.oLanguage, tid = settings.sTableId, n = $('<div/>', { 'class': settings.oClasses.info.container, } ); opts = $.extend({ callback: lang.fnInfoCallback, empty: lang.sInfoEmpty, postfix: lang.sInfoPostFix, search: lang.sInfoFiltered, text: lang.sInfo, }, opts); // Update display on each draw settings.aoDrawCallback.push(function (s) { _fnUpdateInfo(s, opts, n); }); // For the first info display in the table, we add a callback and aria information. if (! settings._infoEl) { n.attr({ 'aria-live': 'polite', id: tid+'_info', role: 'status' }); // Table is described by our info div $(settings.nTable).attr( 'aria-describedby', tid+'_info' ); settings._infoEl = n; } return n; }, 'i' ); /** * Update the information elements in the display * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnUpdateInfo ( settings, opts, node ) { var start = settings._iDisplayStart+1, end = settings.fnDisplayEnd(), max = settings.fnRecordsTotal(), total = settings.fnRecordsDisplay(), out = total ? opts.text : opts.empty; if ( total !== max ) { // Record set after filtering out += ' ' + opts.search; } // Convert the macros out += opts.postfix; out = _fnMacros( settings, out ); if ( opts.callback ) { out = opts.callback.call( settings.oInstance, settings, start, end, max, total, out ); } node.html( out ); _fnCallbackFire(settings, null, 'info', [settings, node[0], out]); } var __searchCounter = 0; // opts // - text // - placeholder DataTable.feature.register( 'search', function ( settings, opts ) { // Don't show the input if filtering isn't available on the table if (! settings.oFeatures.bFilter) { return null; } var classes = settings.oClasses.search; var tableId = settings.sTableId; var language = settings.oLanguage; var previousSearch = settings.oPreviousSearch; var input = '<input type="search" class="'+classes.input+'"/>'; opts = $.extend({ placeholder: language.sSearchPlaceholder, text: language.sSearch }, opts); // The _INPUT_ is optional - is appended if not present if (opts.text.indexOf('_INPUT_') === -1) { opts.text += '_INPUT_'; } opts.text = _fnMacros(settings, opts.text); // We can put the <input> outside of the label if it is at the start or end // which helps improve accessability (not all screen readers like implicit // for elements). var end = opts.text.match(/_INPUT_$/); var start = opts.text.match(/^_INPUT_/); var removed = opts.text.replace(/_INPUT_/, ''); var str = '<label>' + opts.text + '</label>'; if (start) { str = '_INPUT_<label>' + removed + '</label>'; } else if (end) { str = '<label>' + removed + '</label>_INPUT_'; } var filter = $('<div>') .addClass(classes.container) .append(str.replace(/_INPUT_/, input)); // add for and id to label and input filter.find('label').attr('for', 'dt-search-' + __searchCounter); filter.find('input').attr('id', 'dt-search-' + __searchCounter); __searchCounter++; var searchFn = function(event) { var val = this.value; if(previousSearch.return && event.key !== "Enter") { return; } /* Now do the filter */ if ( val != previousSearch.search ) { previousSearch.search = val; _fnFilterComplete( settings, previousSearch ); // Need to redraw, without resorting settings._iDisplayStart = 0; _fnDraw( settings ); } }; var searchDelay = settings.searchDelay !== null ? settings.searchDelay : 0; var jqFilter = $('input', filter) .val( previousSearch.search ) .attr( 'placeholder', opts.placeholder ) .on( 'keyup.DT search.DT input.DT paste.DT cut.DT', searchDelay ? DataTable.util.debounce( searchFn, searchDelay ) : searchFn ) .on( 'mouseup.DT', function(e) { // Edge fix! Edge 17 does not trigger anything other than mouse events when clicking // on the clear icon (Edge bug 17584515). This is safe in other browsers as `searchFn` // checks the value to see if it has changed. In other browsers it won't have. setTimeout( function () { searchFn.call(jqFilter[0], e); }, 10); } ) .on( 'keypress.DT', function(e) { /* Prevent form submission */ if ( e.keyCode == 13 ) { return false; } } ) .attr('aria-controls', tableId); // Update the input elements whenever the table is filtered $(settings.nTable).on( 'search.dt.DT', function ( ev, s ) { if ( settings === s && jqFilter[0] !== document.activeElement ) { jqFilter.val( typeof previousSearch.search !== 'function' ? previousSearch.search : '' ); } } ); return filter; }, 'f' ); // opts // - type - button configuration // - buttons - number of buttons to show - must be odd DataTable.feature.register( 'paging', function ( settings, opts ) { // Don't show the paging input if the table doesn't have paging enabled if (! settings.oFeatures.bPaginate) { return null; } opts = $.extend({ buttons: DataTable.ext.pager.numbers_length, type: settings.sPaginationType, boundaryNumbers: true }, opts); // To be removed in 2.1 if (opts.numbers) { opts.buttons = opts.numbers; } var host = $('<div/>').addClass( settings.oClasses.paging.container + ' paging_' + opts.type ); var draw = function () { _pagingDraw(settings, host, opts); }; settings.aoDrawCallback.push(draw); // Responsive redraw of paging control $(settings.nTable).on('column-sizing.dt.DT', draw); return host; }, 'p' ); function _pagingDraw(settings, host, opts) { if (! settings._bInitComplete) { return; } var plugin = DataTable.ext.pager[ opts.type ], aria = settings.oLanguage.oAria.paginate || {}, start = settings._iDisplayStart, len = settings._iDisplayLength, visRecords = settings.fnRecordsDisplay(), all = len === -1, page = all ? 0 : Math.ceil( start / len ), pages = all ? 1 : Math.ceil( visRecords / len ), buttons = plugin() .map(function (val) { return val === 'numbers' ? _pagingNumbers(page, pages, opts.buttons, opts.boundaryNumbers) : val; }) .flat(); var buttonEls = []; for (var i=0 ; i<buttons.length ; i++) { var button = buttons[i]; var btnInfo = _pagingButtonInfo(settings, button, page, pages); var btn = _fnRenderer( settings, 'pagingButton' )( settings, button, btnInfo.display, btnInfo.active, btnInfo.disabled ); // Common attributes $(btn.clicker).attr({ 'aria-controls': settings.sTableId, 'aria-disabled': btnInfo.disabled ? 'true' : null, 'aria-current': btnInfo.active ? 'page' : null, 'aria-label': aria[ button ], 'data-dt-idx': button, 'tabIndex': btnInfo.disabled ? -1 : settings.iTabIndex, }); if (typeof button !== 'number') { $(btn.clicker).addClass(button); } _fnBindAction( btn.clicker, {action: button}, function(e) { e.preventDefault(); _fnPageChange( settings, e.data.action, true ); } ); buttonEls.push(btn.display); } var wrapped = _fnRenderer(settings, 'pagingContainer')( settings, buttonEls ); var activeEl = host.find(document.activeElement).data('dt-idx'); host.empty().append(wrapped); if ( activeEl !== undefined ) { host.find( '[data-dt-idx='+activeEl+']' ).trigger('focus'); } // Responsive - check if the buttons are over two lines based on the // height of the buttons and the container. if ( buttonEls.length && // any buttons opts.numbers > 1 && // prevent infinite $(host).height() >= ($(buttonEls[0]).outerHeight() * 2) - 10 ) { _pagingDraw(settings, host, $.extend({}, opts, { numbers: opts.numbers - 2 })); } } /** * Get properties for a button based on the current paging state of the table * * @param {*} settings DT settings object * @param {*} button The button type in question * @param {*} page Table's current page * @param {*} pages Number of pages * @returns Info object */ function _pagingButtonInfo(settings, button, page, pages) { var lang = settings.oLanguage.oPaginate; var o = { display: '', active: false, disabled: false }; switch ( button ) { case 'ellipsis': o.display = '…'; o.disabled = true; break; case 'first': o.display = lang.sFirst; if (page === 0) { o.disabled = true; } break; case 'previous': o.display = lang.sPrevious; if ( page === 0 ) { o.disabled = true; } break; case 'next': o.display = lang.sNext; if ( pages === 0 || page === pages-1 ) { o.disabled = true; } break; case 'last': o.display = lang.sLast; if ( pages === 0 || page === pages-1 ) { o.disabled = true; } break; default: if ( typeof button === 'number' ) { o.display = settings.fnFormatNumber( button + 1 ); if (page === button) { o.active = true; } } break; } return o; } /** * Compute what number buttons to show in the paging control * * @param {*} page Current page * @param {*} pages Total number of pages * @param {*} buttons Target number of number buttons * @param {boolean} addFirstLast Indicate if page 1 and end should be included * @returns Buttons to show */ function _pagingNumbers ( page, pages, buttons, addFirstLast ) { var numbers = [], half = Math.floor(buttons / 2), before = addFirstLast ? 2 : 1, after = addFirstLast ? 1 : 0; if ( pages <= buttons ) { numbers = _range(0, pages); } else if (buttons === 1) { // Single button - current page only numbers = [page]; } else if (buttons === 3) { // Special logic for just three buttons if (page <= 1) { numbers = [0, 1, 'ellipsis']; } else if (page >= pages - 2) { numbers = _range(pages-2, pages); numbers.unshift('ellipsis'); } else { numbers = ['ellipsis', page, 'ellipsis']; } } else if ( page <= half ) { numbers = _range(0, buttons-before); numbers.push('ellipsis'); if (addFirstLast) { numbers.push(pages-1); } } else if ( page >= pages - 1 - half ) { numbers = _range(pages-(buttons-before), pages); numbers.unshift('ellipsis'); if (addFirstLast) { numbers.unshift(0); } } else { numbers = _range(page-half+before, page+half-after); numbers.push('ellipsis'); numbers.unshift('ellipsis'); if (addFirstLast) { numbers.push(pages-1); numbers.unshift(0); } } return numbers; } var __lengthCounter = 0; // opts // - menu // - text DataTable.feature.register( 'pageLength', function ( settings, opts ) { var features = settings.oFeatures; // For compatibility with the legacy `pageLength` top level option if (! features.bPaginate || ! features.bLengthChange) { return null; } opts = $.extend({ menu: settings.aLengthMenu, text: settings.oLanguage.sLengthMenu }, opts); var classes = settings.oClasses.length, tableId = settings.sTableId, menu = opts.menu, lengths = [], language = [], i; // Options can be given in a number of ways if (Array.isArray( menu[0] )) { // Old 1.x style - 2D array lengths = menu[0]; language = menu[1]; } else { for ( i=0 ; i<menu.length ; i++ ) { // An object with different label and value if ($.isPlainObject(menu[i])) { lengths.push(menu[i].value); language.push(menu[i].label); } else { // Or just a number to display and use lengths.push(menu[i]); language.push(menu[i]); } } } // We can put the <select> outside of the label if it is at the start or // end which helps improve accessability (not all screen readers like // implicit for elements). var end = opts.text.match(/_MENU_$/); var start = opts.text.match(/^_MENU_/); var removed = opts.text.replace(/_MENU_/, ''); var str = '<label>' + opts.text + '</label>'; if (start) { str = '_MENU_<label>' + removed + '</label>'; } else if (end) { str = '<label>' + removed + '</label>_MENU_'; } // Wrapper element - use a span as a holder for where the select will go var div = $('<div/>') .addClass( classes.container ) .append( str.replace( '_MENU_', '<span></span>' ) ); // Save text node content for macro updating var textNodes = []; div.find('label')[0].childNodes.forEach(function (el) { if (el.nodeType === Node.TEXT_NODE) { textNodes.push({ el: el, text: el.textContent }); } }) // Update the label text in case it has an entries value var updateEntries = function (len) { textNodes.forEach(function (node) { node.el.textContent = _fnMacros(settings, node.text, len); }); } // Next, the select itself, along with the options var select = $('<select/>', { 'name': tableId+'_length', 'aria-controls': tableId, 'class': classes.select } ); for ( i=0 ; i<lengths.length ; i++ ) { select[0][ i ] = new Option( typeof language[i] === 'number' ? settings.fnFormatNumber( language[i] ) : language[i], lengths[i] ); } // add for and id to label and input div.find('label').attr('for', 'dt-length-' + __lengthCounter); select.attr('id', 'dt-length-' + __lengthCounter); __lengthCounter++; // Swap in the select list div.find('span').replaceWith(select); // Can't use `select` variable as user might provide their own and the // reference is broken by the use of outerHTML $('select', div) .val( settings._iDisplayLength ) .on( 'change.DT', function() { _fnLengthChange( settings, $(this).val() ); _fnDraw( settings ); } ); // Update node value whenever anything changes the table's length $(settings.nTable).on( 'length.dt.DT', function (e, s, len) { if ( settings === s ) { $('select', div).val( len ); // Resolve plurals in the text for the new length updateEntries(len); } } ); updateEntries(settings._iDisplayLength); return div; }, 'l' ); // jQuery access $.fn.dataTable = DataTable; // Provide access to the host jQuery object (circular reference) DataTable.$ = $; // Legacy aliases $.fn.dataTableSettings = DataTable.settings; $.fn.dataTableExt = DataTable.ext; // With a capital `D` we return a DataTables API instance rather than a // jQuery object $.fn.DataTable = function ( opts ) { return $(this).dataTable( opts ).api(); }; // All properties that are available to $.fn.dataTable should also be // available on $.fn.DataTable $.each( DataTable, function ( prop, val ) { $.fn.DataTable[ prop ] = val; } ); export default DataTable; node_modules/datatables.net/js/dataTables.js000064400001315501151676725350015157 0ustar00/*! DataTables 2.0.8 * © SpryMedia Ltd - datatables.net/license */ /** * @summary DataTables * @description Paginate, search and order HTML tables * @version 2.0.8 * @author SpryMedia Ltd * @contact www.datatables.net * @copyright SpryMedia Ltd. * * This source file is free software, available under the following license: * MIT license - https://datatables.net/license * * This source file is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the license files for details. * * For details please refer to: https://www.datatables.net */ (function( factory ) { "use strict"; if ( typeof define === 'function' && define.amd ) { // AMD define( ['jquery'], function ( $ ) { return factory( $, window, document ); } ); } else if ( typeof exports === 'object' ) { // CommonJS // jQuery's factory checks for a global window - if it isn't present then it // returns a factory function that expects the window object var jq = require('jquery'); if (typeof window === 'undefined') { module.exports = function (root, $) { if ( ! root ) { // CommonJS environments without a window global must pass a // root. This will give an error otherwise root = window; } if ( ! $ ) { $ = jq( root ); } return factory( $, root, root.document ); }; } else { module.exports = factory( jq, window, window.document ); } } else { // Browser window.DataTable = factory( jQuery, window, document ); } }(function( $, window, document ) { "use strict"; var DataTable = function ( selector, options ) { // Check if called with a window or jQuery object for DOM less applications // This is for backwards compatibility if (DataTable.factory(selector, options)) { return DataTable; } // When creating with `new`, create a new DataTable, returning the API instance if (this instanceof DataTable) { return $(selector).DataTable(options); } else { // Argument switching options = selector; } var _that = this; var emptyInit = options === undefined; var len = this.length; if ( emptyInit ) { options = {}; } // Method to get DT API instance from jQuery object this.api = function () { return new _Api( this ); }; this.each(function() { // For each initialisation we want to give it a clean initialisation // object that can be bashed around var o = {}; var oInit = len > 1 ? // optimisation for single table case _fnExtend( o, options, true ) : options; var i=0, iLen; var sId = this.getAttribute( 'id' ); var bInitHandedOff = false; var defaults = DataTable.defaults; var $this = $(this); /* Sanity check */ if ( this.nodeName.toLowerCase() != 'table' ) { _fnLog( null, 0, 'Non-table node initialisation ('+this.nodeName+')', 2 ); return; } $(this).trigger( 'options.dt', oInit ); /* Backwards compatibility for the defaults */ _fnCompatOpts( defaults ); _fnCompatCols( defaults.column ); /* Convert the camel-case defaults to Hungarian */ _fnCamelToHungarian( defaults, defaults, true ); _fnCamelToHungarian( defaults.column, defaults.column, true ); /* Setting up the initialisation object */ _fnCamelToHungarian( defaults, $.extend( oInit, $this.data() ), true ); /* Check to see if we are re-initialising a table */ var allSettings = DataTable.settings; for ( i=0, iLen=allSettings.length ; i<iLen ; i++ ) { var s = allSettings[i]; /* Base check on table node */ if ( s.nTable == this || (s.nTHead && s.nTHead.parentNode == this) || (s.nTFoot && s.nTFoot.parentNode == this) ) { var bRetrieve = oInit.bRetrieve !== undefined ? oInit.bRetrieve : defaults.bRetrieve; var bDestroy = oInit.bDestroy !== undefined ? oInit.bDestroy : defaults.bDestroy; if ( emptyInit || bRetrieve ) { return s.oInstance; } else if ( bDestroy ) { new DataTable.Api(s).destroy(); break; } else { _fnLog( s, 0, 'Cannot reinitialise DataTable', 3 ); return; } } /* If the element we are initialising has the same ID as a table which was previously * initialised, but the table nodes don't match (from before) then we destroy the old * instance by simply deleting it. This is under the assumption that the table has been * destroyed by other methods. Anyone using non-id selectors will need to do this manually */ if ( s.sTableId == this.id ) { allSettings.splice( i, 1 ); break; } } /* Ensure the table has an ID - required for accessibility */ if ( sId === null || sId === "" ) { sId = "DataTables_Table_"+(DataTable.ext._unique++); this.id = sId; } /* Create the settings object for this table and set some of the default parameters */ var oSettings = $.extend( true, {}, DataTable.models.oSettings, { "sDestroyWidth": $this[0].style.width, "sInstance": sId, "sTableId": sId, colgroup: $('<colgroup>').prependTo(this), fastData: function (row, column, type) { return _fnGetCellData(oSettings, row, column, type); } } ); oSettings.nTable = this; oSettings.oInit = oInit; allSettings.push( oSettings ); // Make a single API instance available for internal handling oSettings.api = new _Api( oSettings ); // Need to add the instance after the instance after the settings object has been added // to the settings array, so we can self reference the table instance if more than one oSettings.oInstance = (_that.length===1) ? _that : $this.dataTable(); // Backwards compatibility, before we apply all the defaults _fnCompatOpts( oInit ); // If the length menu is given, but the init display length is not, use the length menu if ( oInit.aLengthMenu && ! oInit.iDisplayLength ) { oInit.iDisplayLength = Array.isArray(oInit.aLengthMenu[0]) ? oInit.aLengthMenu[0][0] : $.isPlainObject( oInit.aLengthMenu[0] ) ? oInit.aLengthMenu[0].value : oInit.aLengthMenu[0]; } // Apply the defaults and init options to make a single init object will all // options defined from defaults and instance options. oInit = _fnExtend( $.extend( true, {}, defaults ), oInit ); // Map the initialisation options onto the settings object _fnMap( oSettings.oFeatures, oInit, [ "bPaginate", "bLengthChange", "bFilter", "bSort", "bSortMulti", "bInfo", "bProcessing", "bAutoWidth", "bSortClasses", "bServerSide", "bDeferRender" ] ); _fnMap( oSettings, oInit, [ "ajax", "fnFormatNumber", "sServerMethod", "aaSorting", "aaSortingFixed", "aLengthMenu", "sPaginationType", "iStateDuration", "bSortCellsTop", "iTabIndex", "sDom", "fnStateLoadCallback", "fnStateSaveCallback", "renderer", "searchDelay", "rowId", "caption", "layout", [ "iCookieDuration", "iStateDuration" ], // backwards compat [ "oSearch", "oPreviousSearch" ], [ "aoSearchCols", "aoPreSearchCols" ], [ "iDisplayLength", "_iDisplayLength" ] ] ); _fnMap( oSettings.oScroll, oInit, [ [ "sScrollX", "sX" ], [ "sScrollXInner", "sXInner" ], [ "sScrollY", "sY" ], [ "bScrollCollapse", "bCollapse" ] ] ); _fnMap( oSettings.oLanguage, oInit, "fnInfoCallback" ); /* Callback functions which are array driven */ _fnCallbackReg( oSettings, 'aoDrawCallback', oInit.fnDrawCallback ); _fnCallbackReg( oSettings, 'aoStateSaveParams', oInit.fnStateSaveParams ); _fnCallbackReg( oSettings, 'aoStateLoadParams', oInit.fnStateLoadParams ); _fnCallbackReg( oSettings, 'aoStateLoaded', oInit.fnStateLoaded ); _fnCallbackReg( oSettings, 'aoRowCallback', oInit.fnRowCallback ); _fnCallbackReg( oSettings, 'aoRowCreatedCallback', oInit.fnCreatedRow ); _fnCallbackReg( oSettings, 'aoHeaderCallback', oInit.fnHeaderCallback ); _fnCallbackReg( oSettings, 'aoFooterCallback', oInit.fnFooterCallback ); _fnCallbackReg( oSettings, 'aoInitComplete', oInit.fnInitComplete ); _fnCallbackReg( oSettings, 'aoPreDrawCallback', oInit.fnPreDrawCallback ); oSettings.rowIdFn = _fnGetObjectDataFn( oInit.rowId ); /* Browser support detection */ _fnBrowserDetect( oSettings ); var oClasses = oSettings.oClasses; $.extend( oClasses, DataTable.ext.classes, oInit.oClasses ); $this.addClass( oClasses.table ); if (! oSettings.oFeatures.bPaginate) { oInit.iDisplayStart = 0; } if ( oSettings.iInitDisplayStart === undefined ) { /* Display start point, taking into account the save saving */ oSettings.iInitDisplayStart = oInit.iDisplayStart; oSettings._iDisplayStart = oInit.iDisplayStart; } /* Language definitions */ var oLanguage = oSettings.oLanguage; $.extend( true, oLanguage, oInit.oLanguage ); if ( oLanguage.sUrl ) { /* Get the language definitions from a file - because this Ajax call makes the language * get async to the remainder of this function we use bInitHandedOff to indicate that * _fnInitialise will be fired by the returned Ajax handler, rather than the constructor */ $.ajax( { dataType: 'json', url: oLanguage.sUrl, success: function ( json ) { _fnCamelToHungarian( defaults.oLanguage, json ); $.extend( true, oLanguage, json, oSettings.oInit.oLanguage ); _fnCallbackFire( oSettings, null, 'i18n', [oSettings], true); _fnInitialise( oSettings ); }, error: function () { // Error occurred loading language file _fnLog( oSettings, 0, 'i18n file loading error', 21 ); // continue on as best we can _fnInitialise( oSettings ); } } ); bInitHandedOff = true; } else { _fnCallbackFire( oSettings, null, 'i18n', [oSettings]); } /* * Columns * See if we should load columns automatically or use defined ones */ var columnsInit = []; var thead = this.getElementsByTagName('thead'); var initHeaderLayout = _fnDetectHeader( oSettings, thead[0] ); // If we don't have a columns array, then generate one with nulls if ( oInit.aoColumns ) { columnsInit = oInit.aoColumns; } else if ( initHeaderLayout.length ) { for ( i=0, iLen=initHeaderLayout[0].length ; i<iLen ; i++ ) { columnsInit.push( null ); } } // Add the columns for ( i=0, iLen=columnsInit.length ; i<iLen ; i++ ) { _fnAddColumn( oSettings ); } // Apply the column definitions _fnApplyColumnDefs( oSettings, oInit.aoColumnDefs, columnsInit, initHeaderLayout, function (iCol, oDef) { _fnColumnOptions( oSettings, iCol, oDef ); } ); /* HTML5 attribute detection - build an mData object automatically if the * attributes are found */ var rowOne = $this.children('tbody').find('tr').eq(0); if ( rowOne.length ) { var a = function ( cell, name ) { return cell.getAttribute( 'data-'+name ) !== null ? name : null; }; $( rowOne[0] ).children('th, td').each( function (i, cell) { var col = oSettings.aoColumns[i]; if (! col) { _fnLog( oSettings, 0, 'Incorrect column count', 18 ); } if ( col.mData === i ) { var sort = a( cell, 'sort' ) || a( cell, 'order' ); var filter = a( cell, 'filter' ) || a( cell, 'search' ); if ( sort !== null || filter !== null ) { col.mData = { _: i+'.display', sort: sort !== null ? i+'.@data-'+sort : undefined, type: sort !== null ? i+'.@data-'+sort : undefined, filter: filter !== null ? i+'.@data-'+filter : undefined }; col._isArrayHost = true; _fnColumnOptions( oSettings, i ); } } } ); } var features = oSettings.oFeatures; var loadedInit = function () { /* * Sorting * @todo For modularisation (1.11) this needs to do into a sort start up handler */ // If aaSorting is not defined, then we use the first indicator in asSorting // in case that has been altered, so the default sort reflects that option if ( oInit.aaSorting === undefined ) { var sorting = oSettings.aaSorting; for ( i=0, iLen=sorting.length ; i<iLen ; i++ ) { sorting[i][1] = oSettings.aoColumns[ i ].asSorting[0]; } } /* Do a first pass on the sorting classes (allows any size changes to be taken into * account, and also will apply sorting disabled classes if disabled */ _fnSortingClasses( oSettings ); _fnCallbackReg( oSettings, 'aoDrawCallback', function () { if ( oSettings.bSorted || _fnDataSource( oSettings ) === 'ssp' || features.bDeferRender ) { _fnSortingClasses( oSettings ); } } ); /* * Final init * Cache the header, body and footer as required, creating them if needed */ var caption = $this.children('caption'); if ( oSettings.caption ) { if ( caption.length === 0 ) { caption = $('<caption/>').appendTo( $this ); } caption.html( oSettings.caption ); } // Store the caption side, so we can remove the element from the document // when creating the element if (caption.length) { caption[0]._captionSide = caption.css('caption-side'); oSettings.captionNode = caption[0]; } if ( thead.length === 0 ) { thead = $('<thead/>').appendTo($this); } oSettings.nTHead = thead[0]; $('tr', thead).addClass(oClasses.thead.row); var tbody = $this.children('tbody'); if ( tbody.length === 0 ) { tbody = $('<tbody/>').insertAfter(thead); } oSettings.nTBody = tbody[0]; var tfoot = $this.children('tfoot'); if ( tfoot.length === 0 ) { // If we are a scrolling table, and no footer has been given, then we need to create // a tfoot element for the caption element to be appended to tfoot = $('<tfoot/>').appendTo($this); } oSettings.nTFoot = tfoot[0]; $('tr', tfoot).addClass(oClasses.tfoot.row); // Check if there is data passing into the constructor if ( oInit.aaData ) { for ( i=0 ; i<oInit.aaData.length ; i++ ) { _fnAddData( oSettings, oInit.aaData[ i ] ); } } else if ( _fnDataSource( oSettings ) == 'dom' ) { // Grab the data from the page _fnAddTr( oSettings, $(oSettings.nTBody).children('tr') ); } /* Copy the data index array */ oSettings.aiDisplay = oSettings.aiDisplayMaster.slice(); /* Initialisation complete - table can be drawn */ oSettings.bInitialised = true; /* Check if we need to initialise the table (it might not have been handed off to the * language processor) */ if ( bInitHandedOff === false ) { _fnInitialise( oSettings ); } }; /* Must be done after everything which can be overridden by the state saving! */ _fnCallbackReg( oSettings, 'aoDrawCallback', _fnSaveState ); if ( oInit.bStateSave ) { features.bStateSave = true; _fnLoadState( oSettings, oInit, loadedInit ); } else { loadedInit(); } } ); _that = null; return this; }; /** * DataTables extensions * * This namespace acts as a collection area for plug-ins that can be used to * extend DataTables capabilities. Indeed many of the build in methods * use this method to provide their own capabilities (sorting methods for * example). * * Note that this namespace is aliased to `jQuery.fn.dataTableExt` for legacy * reasons * * @namespace */ DataTable.ext = _ext = { /** * Buttons. For use with the Buttons extension for DataTables. This is * defined here so other extensions can define buttons regardless of load * order. It is _not_ used by DataTables core. * * @type object * @default {} */ buttons: {}, /** * Element class names * * @type object * @default {} */ classes: {}, /** * DataTables build type (expanded by the download builder) * * @type string */ builder: "-source-", /** * Error reporting. * * How should DataTables report an error. Can take the value 'alert', * 'throw', 'none' or a function. * * @type string|function * @default alert */ errMode: "alert", /** * Legacy so v1 plug-ins don't throw js errors on load */ feature: [], /** * Feature plug-ins. * * This is an object of callbacks which provide the features for DataTables * to be initialised via the `layout` option. */ features: {}, /** * Row searching. * * This method of searching is complimentary to the default type based * searching, and a lot more comprehensive as it allows you complete control * over the searching logic. Each element in this array is a function * (parameters described below) that is called for every row in the table, * and your logic decides if it should be included in the searching data set * or not. * * Searching functions have the following input parameters: * * 1. `{object}` DataTables settings object: see * {@link DataTable.models.oSettings} * 2. `{array|object}` Data for the row to be processed (same as the * original format that was passed in as the data source, or an array * from a DOM data source * 3. `{int}` Row index ({@link DataTable.models.oSettings.aoData}), which * can be useful to retrieve the `TR` element if you need DOM interaction. * * And the following return is expected: * * * {boolean} Include the row in the searched result set (true) or not * (false) * * Note that as with the main search ability in DataTables, technically this * is "filtering", since it is subtractive. However, for consistency in * naming we call it searching here. * * @type array * @default [] * * @example * // The following example shows custom search being applied to the * // fourth column (i.e. the data[3] index) based on two input values * // from the end-user, matching the data in a certain range. * $.fn.dataTable.ext.search.push( * function( settings, data, dataIndex ) { * var min = document.getElementById('min').value * 1; * var max = document.getElementById('max').value * 1; * var version = data[3] == "-" ? 0 : data[3]*1; * * if ( min == "" && max == "" ) { * return true; * } * else if ( min == "" && version < max ) { * return true; * } * else if ( min < version && "" == max ) { * return true; * } * else if ( min < version && version < max ) { * return true; * } * return false; * } * ); */ search: [], /** * Selector extensions * * The `selector` option can be used to extend the options available for the * selector modifier options (`selector-modifier` object data type) that * each of the three built in selector types offer (row, column and cell + * their plural counterparts). For example the Select extension uses this * mechanism to provide an option to select only rows, columns and cells * that have been marked as selected by the end user (`{selected: true}`), * which can be used in conjunction with the existing built in selector * options. * * Each property is an array to which functions can be pushed. The functions * take three attributes: * * * Settings object for the host table * * Options object (`selector-modifier` object type) * * Array of selected item indexes * * The return is an array of the resulting item indexes after the custom * selector has been applied. * * @type object */ selector: { cell: [], column: [], row: [] }, /** * Legacy configuration options. Enable and disable legacy options that * are available in DataTables. * * @type object */ legacy: { /** * Enable / disable DataTables 1.9 compatible server-side processing * requests * * @type boolean * @default null */ ajax: null }, /** * Pagination plug-in methods. * * Each entry in this object is a function and defines which buttons should * be shown by the pagination rendering method that is used for the table: * {@link DataTable.ext.renderer.pageButton}. The renderer addresses how the * buttons are displayed in the document, while the functions here tell it * what buttons to display. This is done by returning an array of button * descriptions (what each button will do). * * Pagination types (the four built in options and any additional plug-in * options defined here) can be used through the `paginationType` * initialisation parameter. * * The functions defined take two parameters: * * 1. `{int} page` The current page index * 2. `{int} pages` The number of pages in the table * * Each function is expected to return an array where each element of the * array can be one of: * * * `first` - Jump to first page when activated * * `last` - Jump to last page when activated * * `previous` - Show previous page when activated * * `next` - Show next page when activated * * `{int}` - Show page of the index given * * `{array}` - A nested array containing the above elements to add a * containing 'DIV' element (might be useful for styling). * * Note that DataTables v1.9- used this object slightly differently whereby * an object with two functions would be defined for each plug-in. That * ability is still supported by DataTables 1.10+ to provide backwards * compatibility, but this option of use is now decremented and no longer * documented in DataTables 1.10+. * * @type object * @default {} * * @example * // Show previous, next and current page buttons only * $.fn.dataTableExt.oPagination.current = function ( page, pages ) { * return [ 'previous', page, 'next' ]; * }; */ pager: {}, renderer: { pageButton: {}, header: {} }, /** * Ordering plug-ins - custom data source * * The extension options for ordering of data available here is complimentary * to the default type based ordering that DataTables typically uses. It * allows much greater control over the the data that is being used to * order a column, but is necessarily therefore more complex. * * This type of ordering is useful if you want to do ordering based on data * live from the DOM (for example the contents of an 'input' element) rather * than just the static string that DataTables knows of. * * The way these plug-ins work is that you create an array of the values you * wish to be ordering for the column in question and then return that * array. The data in the array much be in the index order of the rows in * the table (not the currently ordering order!). Which order data gathering * function is run here depends on the `dt-init columns.orderDataType` * parameter that is used for the column (if any). * * The functions defined take two parameters: * * 1. `{object}` DataTables settings object: see * {@link DataTable.models.oSettings} * 2. `{int}` Target column index * * Each function is expected to return an array: * * * `{array}` Data for the column to be ordering upon * * @type array * * @example * // Ordering using `input` node values * $.fn.dataTable.ext.order['dom-text'] = function ( settings, col ) * { * return this.api().column( col, {order:'index'} ).nodes().map( function ( td, i ) { * return $('input', td).val(); * } ); * } */ order: {}, /** * Type based plug-ins. * * Each column in DataTables has a type assigned to it, either by automatic * detection or by direct assignment using the `type` option for the column. * The type of a column will effect how it is ordering and search (plug-ins * can also make use of the column type if required). * * @namespace */ type: { /** * Automatic column class assignment */ className: {}, /** * Type detection functions. * * The functions defined in this object are used to automatically detect * a column's type, making initialisation of DataTables super easy, even * when complex data is in the table. * * The functions defined take two parameters: * * 1. `{*}` Data from the column cell to be analysed * 2. `{settings}` DataTables settings object. This can be used to * perform context specific type detection - for example detection * based on language settings such as using a comma for a decimal * place. Generally speaking the options from the settings will not * be required * * Each function is expected to return: * * * `{string|null}` Data type detected, or null if unknown (and thus * pass it on to the other type detection functions. * * @type array * * @example * // Currency type detection plug-in: * $.fn.dataTable.ext.type.detect.push( * function ( data, settings ) { * // Check the numeric part * if ( ! data.substring(1).match(/[0-9]/) ) { * return null; * } * * // Check prefixed by currency * if ( data.charAt(0) == '$' || data.charAt(0) == '£' ) { * return 'currency'; * } * return null; * } * ); */ detect: [], /** * Automatic renderer assignment */ render: {}, /** * Type based search formatting. * * The type based searching functions can be used to pre-format the * data to be search on. For example, it can be used to strip HTML * tags or to de-format telephone numbers for numeric only searching. * * Note that is a search is not defined for a column of a given type, * no search formatting will be performed. * * Pre-processing of searching data plug-ins - When you assign the sType * for a column (or have it automatically detected for you by DataTables * or a type detection plug-in), you will typically be using this for * custom sorting, but it can also be used to provide custom searching * by allowing you to pre-processing the data and returning the data in * the format that should be searched upon. This is done by adding * functions this object with a parameter name which matches the sType * for that target column. This is the corollary of <i>afnSortData</i> * for searching data. * * The functions defined take a single parameter: * * 1. `{*}` Data from the column cell to be prepared for searching * * Each function is expected to return: * * * `{string|null}` Formatted string that will be used for the searching. * * @type object * @default {} * * @example * $.fn.dataTable.ext.type.search['title-numeric'] = function ( d ) { * return d.replace(/\n/g," ").replace( /<.*?>/g, "" ); * } */ search: {}, /** * Type based ordering. * * The column type tells DataTables what ordering to apply to the table * when a column is sorted upon. The order for each type that is defined, * is defined by the functions available in this object. * * Each ordering option can be described by three properties added to * this object: * * * `{type}-pre` - Pre-formatting function * * `{type}-asc` - Ascending order function * * `{type}-desc` - Descending order function * * All three can be used together, only `{type}-pre` or only * `{type}-asc` and `{type}-desc` together. It is generally recommended * that only `{type}-pre` is used, as this provides the optimal * implementation in terms of speed, although the others are provided * for compatibility with existing Javascript sort functions. * * `{type}-pre`: Functions defined take a single parameter: * * 1. `{*}` Data from the column cell to be prepared for ordering * * And return: * * * `{*}` Data to be sorted upon * * `{type}-asc` and `{type}-desc`: Functions are typical Javascript sort * functions, taking two parameters: * * 1. `{*}` Data to compare to the second parameter * 2. `{*}` Data to compare to the first parameter * * And returning: * * * `{*}` Ordering match: <0 if first parameter should be sorted lower * than the second parameter, ===0 if the two parameters are equal and * >0 if the first parameter should be sorted height than the second * parameter. * * @type object * @default {} * * @example * // Numeric ordering of formatted numbers with a pre-formatter * $.extend( $.fn.dataTable.ext.type.order, { * "string-pre": function(x) { * a = (a === "-" || a === "") ? 0 : a.replace( /[^\d\-\.]/g, "" ); * return parseFloat( a ); * } * } ); * * @example * // Case-sensitive string ordering, with no pre-formatting method * $.extend( $.fn.dataTable.ext.order, { * "string-case-asc": function(x,y) { * return ((x < y) ? -1 : ((x > y) ? 1 : 0)); * }, * "string-case-desc": function(x,y) { * return ((x < y) ? 1 : ((x > y) ? -1 : 0)); * } * } ); */ order: {} }, /** * Unique DataTables instance counter * * @type int * @private */ _unique: 0, // // Depreciated // The following properties are retained for backwards compatibility only. // The should not be used in new projects and will be removed in a future // version // /** * Version check function. * @type function * @depreciated Since 1.10 */ fnVersionCheck: DataTable.fnVersionCheck, /** * Index for what 'this' index API functions should use * @type int * @deprecated Since v1.10 */ iApiIndex: 0, /** * Software version * @type string * @deprecated Since v1.10 */ sVersion: DataTable.version }; // // Backwards compatibility. Alias to pre 1.10 Hungarian notation counter parts // $.extend( _ext, { afnFiltering: _ext.search, aTypes: _ext.type.detect, ofnSearch: _ext.type.search, oSort: _ext.type.order, afnSortData: _ext.order, aoFeatures: _ext.feature, oStdClasses: _ext.classes, oPagination: _ext.pager } ); $.extend( DataTable.ext.classes, { container: 'dt-container', empty: { row: 'dt-empty' }, info: { container: 'dt-info' }, length: { container: 'dt-length', select: 'dt-input' }, order: { canAsc: 'dt-orderable-asc', canDesc: 'dt-orderable-desc', isAsc: 'dt-ordering-asc', isDesc: 'dt-ordering-desc', none: 'dt-orderable-none', position: 'sorting_' }, processing: { container: 'dt-processing' }, scrolling: { body: 'dt-scroll-body', container: 'dt-scroll', footer: { self: 'dt-scroll-foot', inner: 'dt-scroll-footInner' }, header: { self: 'dt-scroll-head', inner: 'dt-scroll-headInner' } }, search: { container: 'dt-search', input: 'dt-input' }, table: 'dataTable', tbody: { cell: '', row: '' }, thead: { cell: '', row: '' }, tfoot: { cell: '', row: '' }, paging: { active: 'current', button: 'dt-paging-button', container: 'dt-paging', disabled: 'disabled' } } ); /* * It is useful to have variables which are scoped locally so only the * DataTables functions can access them and they don't leak into global space. * At the same time these functions are often useful over multiple files in the * core and API, so we list, or at least document, all variables which are used * by DataTables as private variables here. This also ensures that there is no * clashing of variable names and that they can easily referenced for reuse. */ // Defined else where // _selector_run // _selector_opts // _selector_row_indexes var _ext; // DataTable.ext var _Api; // DataTable.Api var _api_register; // DataTable.Api.register var _api_registerPlural; // DataTable.Api.registerPlural var _re_dic = {}; var _re_new_lines = /[\r\n\u2028]/g; var _re_html = /<([^>]*>)/g; var _max_str_len = Math.pow(2, 28); // This is not strict ISO8601 - Date.parse() is quite lax, although // implementations differ between browsers. var _re_date = /^\d{2,4}[./-]\d{1,2}[./-]\d{1,2}([T ]{1}\d{1,2}[:.]\d{2}([.:]\d{2})?)?$/; // Escape regular expression special characters var _re_escape_regex = new RegExp( '(\\' + [ '/', '.', '*', '+', '?', '|', '(', ')', '[', ']', '{', '}', '\\', '$', '^', '-' ].join('|\\') + ')', 'g' ); // https://en.wikipedia.org/wiki/Foreign_exchange_market // - \u20BD - Russian ruble. // - \u20a9 - South Korean Won // - \u20BA - Turkish Lira // - \u20B9 - Indian Rupee // - R - Brazil (R$) and South Africa // - fr - Swiss Franc // - kr - Swedish krona, Norwegian krone and Danish krone // - \u2009 is thin space and \u202F is narrow no-break space, both used in many // - Ƀ - Bitcoin // - Ξ - Ethereum // standards as thousands separators. var _re_formatted_numeric = /['\u00A0,$£€¥%\u2009\u202F\u20BD\u20a9\u20BArfkɃΞ]/gi; var _empty = function ( d ) { return !d || d === true || d === '-' ? true : false; }; var _intVal = function ( s ) { var integer = parseInt( s, 10 ); return !isNaN(integer) && isFinite(s) ? integer : null; }; // Convert from a formatted number with characters other than `.` as the // decimal place, to a Javascript number var _numToDecimal = function ( num, decimalPoint ) { // Cache created regular expressions for speed as this function is called often if ( ! _re_dic[ decimalPoint ] ) { _re_dic[ decimalPoint ] = new RegExp( _fnEscapeRegex( decimalPoint ), 'g' ); } return typeof num === 'string' && decimalPoint !== '.' ? num.replace( /\./g, '' ).replace( _re_dic[ decimalPoint ], '.' ) : num; }; var _isNumber = function ( d, decimalPoint, formatted ) { var type = typeof d; var strType = type === 'string'; if ( type === 'number' || type === 'bigint') { return true; } // If empty return immediately so there must be a number if it is a // formatted string (this stops the string "k", or "kr", etc being detected // as a formatted number for currency if ( _empty( d ) ) { return true; } if ( decimalPoint && strType ) { d = _numToDecimal( d, decimalPoint ); } if ( formatted && strType ) { d = d.replace( _re_formatted_numeric, '' ); } return !isNaN( parseFloat(d) ) && isFinite( d ); }; // A string without HTML in it can be considered to be HTML still var _isHtml = function ( d ) { return _empty( d ) || typeof d === 'string'; }; // Is a string a number surrounded by HTML? var _htmlNumeric = function ( d, decimalPoint, formatted ) { if ( _empty( d ) ) { return true; } // input and select strings mean that this isn't just a number if (typeof d === 'string' && d.match(/<(input|select)/i)) { return null; } var html = _isHtml( d ); return ! html ? null : _isNumber( _stripHtml( d ), decimalPoint, formatted ) ? true : null; }; var _pluck = function ( a, prop, prop2 ) { var out = []; var i=0, ien=a.length; // Could have the test in the loop for slightly smaller code, but speed // is essential here if ( prop2 !== undefined ) { for ( ; i<ien ; i++ ) { if ( a[i] && a[i][ prop ] ) { out.push( a[i][ prop ][ prop2 ] ); } } } else { for ( ; i<ien ; i++ ) { if ( a[i] ) { out.push( a[i][ prop ] ); } } } return out; }; // Basically the same as _pluck, but rather than looping over `a` we use `order` // as the indexes to pick from `a` var _pluck_order = function ( a, order, prop, prop2 ) { var out = []; var i=0, ien=order.length; // Could have the test in the loop for slightly smaller code, but speed // is essential here if ( prop2 !== undefined ) { for ( ; i<ien ; i++ ) { if ( a[ order[i] ][ prop ] ) { out.push( a[ order[i] ][ prop ][ prop2 ] ); } } } else { for ( ; i<ien ; i++ ) { if ( a[ order[i] ] ) { out.push( a[ order[i] ][ prop ] ); } } } return out; }; var _range = function ( len, start ) { var out = []; var end; if ( start === undefined ) { start = 0; end = len; } else { end = start; start = len; } for ( var i=start ; i<end ; i++ ) { out.push( i ); } return out; }; var _removeEmpty = function ( a ) { var out = []; for ( var i=0, ien=a.length ; i<ien ; i++ ) { if ( a[i] ) { // careful - will remove all falsy values! out.push( a[i] ); } } return out; }; // Replaceable function in api.util var _stripHtml = function (input) { // Irrelevant check to workaround CodeQL's false positive on the regex if (input.length > _max_str_len) { throw new Error('Exceeded max str len'); } var previous; input = input.replace(_re_html, ''); // Complete tags // Safety for incomplete script tag - use do / while to ensure that // we get all instances do { previous = input; input = input.replace(/<script/i, ''); } while (input !== previous); return previous; }; // Replaceable function in api.util var _escapeHtml = function ( d ) { if (Array.isArray(d)) { d = d.join(','); } return typeof d === 'string' ? d .replace(/&/g, '&') .replace(/</g, '<') .replace(/>/g, '>') .replace(/"/g, '"') : d; }; // Remove diacritics from a string by decomposing it and then removing // non-ascii characters var _normalize = function (str, both) { if (typeof str !== 'string') { return str; } // It is faster to just run `normalize` than it is to check if // we need to with a regex! var res = str.normalize("NFD"); // Equally, here we check if a regex is needed or not return res.length !== str.length ? (both === true ? str + ' ' : '' ) + res.replace(/[\u0300-\u036f]/g, "") : res; } /** * Determine if all values in the array are unique. This means we can short * cut the _unique method at the cost of a single loop. A sorted array is used * to easily check the values. * * @param {array} src Source array * @return {boolean} true if all unique, false otherwise * @ignore */ var _areAllUnique = function ( src ) { if ( src.length < 2 ) { return true; } var sorted = src.slice().sort(); var last = sorted[0]; for ( var i=1, ien=sorted.length ; i<ien ; i++ ) { if ( sorted[i] === last ) { return false; } last = sorted[i]; } return true; }; /** * Find the unique elements in a source array. * * @param {array} src Source array * @return {array} Array of unique items * @ignore */ var _unique = function ( src ) { if (Array.from && Set) { return Array.from(new Set(src)); } if ( _areAllUnique( src ) ) { return src.slice(); } // A faster unique method is to use object keys to identify used values, // but this doesn't work with arrays or objects, which we must also // consider. See jsperf.app/compare-array-unique-versions/4 for more // information. var out = [], val, i, ien=src.length, j, k=0; again: for ( i=0 ; i<ien ; i++ ) { val = src[i]; for ( j=0 ; j<k ; j++ ) { if ( out[j] === val ) { continue again; } } out.push( val ); k++; } return out; }; // Surprisingly this is faster than [].concat.apply // https://jsperf.com/flatten-an-array-loop-vs-reduce/2 var _flatten = function (out, val) { if (Array.isArray(val)) { for (var i=0 ; i<val.length ; i++) { _flatten(out, val[i]); } } else { out.push(val); } return out; } // Similar to jQuery's addClass, but use classList.add function _addClass(el, name) { if (name) { name.split(' ').forEach(function (n) { if (n) { // `add` does deduplication, so no need to check `contains` el.classList.add(n); } }); } } /** * DataTables utility methods * * This namespace provides helper methods that DataTables uses internally to * create a DataTable, but which are not exclusively used only for DataTables. * These methods can be used by extension authors to save the duplication of * code. * * @namespace */ DataTable.util = { /** * Return a string with diacritic characters decomposed * @param {*} mixed Function or string to normalize * @param {*} both Return original string and the normalized string * @returns String or undefined */ diacritics: function (mixed, both) { var type = typeof mixed; if (type !== 'function') { return _normalize(mixed, both); } _normalize = mixed; }, /** * Debounce a function * * @param {function} fn Function to be called * @param {integer} freq Call frequency in mS * @return {function} Wrapped function */ debounce: function ( fn, timeout ) { var timer; return function () { var that = this; var args = arguments; clearTimeout(timer); timer = setTimeout( function () { fn.apply(that, args); }, timeout || 250 ); }; }, /** * Throttle the calls to a function. Arguments and context are maintained * for the throttled function. * * @param {function} fn Function to be called * @param {integer} freq Call frequency in mS * @return {function} Wrapped function */ throttle: function ( fn, freq ) { var frequency = freq !== undefined ? freq : 200, last, timer; return function () { var that = this, now = +new Date(), args = arguments; if ( last && now < last + frequency ) { clearTimeout( timer ); timer = setTimeout( function () { last = undefined; fn.apply( that, args ); }, frequency ); } else { last = now; fn.apply( that, args ); } }; }, /** * Escape a string such that it can be used in a regular expression * * @param {string} val string to escape * @returns {string} escaped string */ escapeRegex: function ( val ) { return val.replace( _re_escape_regex, '\\$1' ); }, /** * Create a function that will write to a nested object or array * @param {*} source JSON notation string * @returns Write function */ set: function ( source ) { if ( $.isPlainObject( source ) ) { /* Unlike get, only the underscore (global) option is used for for * setting data since we don't know the type here. This is why an object * option is not documented for `mData` (which is read/write), but it is * for `mRender` which is read only. */ return DataTable.util.set( source._ ); } else if ( source === null ) { // Nothing to do when the data source is null return function () {}; } else if ( typeof source === 'function' ) { return function (data, val, meta) { source( data, 'set', val, meta ); }; } else if ( typeof source === 'string' && (source.indexOf('.') !== -1 || source.indexOf('[') !== -1 || source.indexOf('(') !== -1) ) { // Like the get, we need to get data from a nested object var setData = function (data, val, src) { var a = _fnSplitObjNotation( src ), b; var aLast = a[a.length-1]; var arrayNotation, funcNotation, o, innerSrc; for ( var i=0, iLen=a.length-1 ; i<iLen ; i++ ) { // Protect against prototype pollution if (a[i] === '__proto__' || a[i] === 'constructor') { throw new Error('Cannot set prototype values'); } // Check if we are dealing with an array notation request arrayNotation = a[i].match(__reArray); funcNotation = a[i].match(__reFn); if ( arrayNotation ) { a[i] = a[i].replace(__reArray, ''); data[ a[i] ] = []; // Get the remainder of the nested object to set so we can recurse b = a.slice(); b.splice( 0, i+1 ); innerSrc = b.join('.'); // Traverse each entry in the array setting the properties requested if ( Array.isArray( val ) ) { for ( var j=0, jLen=val.length ; j<jLen ; j++ ) { o = {}; setData( o, val[j], innerSrc ); data[ a[i] ].push( o ); } } else { // We've been asked to save data to an array, but it // isn't array data to be saved. Best that can be done // is to just save the value. data[ a[i] ] = val; } // The inner call to setData has already traversed through the remainder // of the source and has set the data, thus we can exit here return; } else if ( funcNotation ) { // Function call a[i] = a[i].replace(__reFn, ''); data = data[ a[i] ]( val ); } // If the nested object doesn't currently exist - since we are // trying to set the value - create it if ( data[ a[i] ] === null || data[ a[i] ] === undefined ) { data[ a[i] ] = {}; } data = data[ a[i] ]; } // Last item in the input - i.e, the actual set if ( aLast.match(__reFn ) ) { // Function call data = data[ aLast.replace(__reFn, '') ]( val ); } else { // If array notation is used, we just want to strip it and use the property name // and assign the value. If it isn't used, then we get the result we want anyway data[ aLast.replace(__reArray, '') ] = val; } }; return function (data, val) { // meta is also passed in, but not used return setData( data, val, source ); }; } else { // Array or flat object mapping return function (data, val) { // meta is also passed in, but not used data[source] = val; }; } }, /** * Create a function that will read nested objects from arrays, based on JSON notation * @param {*} source JSON notation string * @returns Value read */ get: function ( source ) { if ( $.isPlainObject( source ) ) { // Build an object of get functions, and wrap them in a single call var o = {}; $.each( source, function (key, val) { if ( val ) { o[key] = DataTable.util.get( val ); } } ); return function (data, type, row, meta) { var t = o[type] || o._; return t !== undefined ? t(data, type, row, meta) : data; }; } else if ( source === null ) { // Give an empty string for rendering / sorting etc return function (data) { // type, row and meta also passed, but not used return data; }; } else if ( typeof source === 'function' ) { return function (data, type, row, meta) { return source( data, type, row, meta ); }; } else if ( typeof source === 'string' && (source.indexOf('.') !== -1 || source.indexOf('[') !== -1 || source.indexOf('(') !== -1) ) { /* If there is a . in the source string then the data source is in a * nested object so we loop over the data for each level to get the next * level down. On each loop we test for undefined, and if found immediately * return. This allows entire objects to be missing and sDefaultContent to * be used if defined, rather than throwing an error */ var fetchData = function (data, type, src) { var arrayNotation, funcNotation, out, innerSrc; if ( src !== "" ) { var a = _fnSplitObjNotation( src ); for ( var i=0, iLen=a.length ; i<iLen ; i++ ) { // Check if we are dealing with special notation arrayNotation = a[i].match(__reArray); funcNotation = a[i].match(__reFn); if ( arrayNotation ) { // Array notation a[i] = a[i].replace(__reArray, ''); // Condition allows simply [] to be passed in if ( a[i] !== "" ) { data = data[ a[i] ]; } out = []; // Get the remainder of the nested object to get a.splice( 0, i+1 ); innerSrc = a.join('.'); // Traverse each entry in the array getting the properties requested if ( Array.isArray( data ) ) { for ( var j=0, jLen=data.length ; j<jLen ; j++ ) { out.push( fetchData( data[j], type, innerSrc ) ); } } // If a string is given in between the array notation indicators, that // is used to join the strings together, otherwise an array is returned var join = arrayNotation[0].substring(1, arrayNotation[0].length-1); data = (join==="") ? out : out.join(join); // The inner call to fetchData has already traversed through the remainder // of the source requested, so we exit from the loop break; } else if ( funcNotation ) { // Function call a[i] = a[i].replace(__reFn, ''); data = data[ a[i] ](); continue; } if (data === null || data[ a[i] ] === null) { return null; } else if ( data === undefined || data[ a[i] ] === undefined ) { return undefined; } data = data[ a[i] ]; } } return data; }; return function (data, type) { // row and meta also passed, but not used return fetchData( data, type, source ); }; } else { // Array or flat object mapping return function (data) { // row and meta also passed, but not used return data[source]; }; } }, stripHtml: function (mixed) { var type = typeof mixed; if (type === 'function') { _stripHtml = mixed; return; } else if (type === 'string') { return _stripHtml(mixed); } return mixed; }, escapeHtml: function (mixed) { var type = typeof mixed; if (type === 'function') { _escapeHtml = mixed; return; } else if (type === 'string' || Array.isArray(mixed)) { return _escapeHtml(mixed); } return mixed; }, unique: _unique }; /** * Create a mapping object that allows camel case parameters to be looked up * for their Hungarian counterparts. The mapping is stored in a private * parameter called `_hungarianMap` which can be accessed on the source object. * @param {object} o * @memberof DataTable#oApi */ function _fnHungarianMap ( o ) { var hungarian = 'a aa ai ao as b fn i m o s ', match, newKey, map = {}; $.each( o, function (key) { match = key.match(/^([^A-Z]+?)([A-Z])/); if ( match && hungarian.indexOf(match[1]+' ') !== -1 ) { newKey = key.replace( match[0], match[2].toLowerCase() ); map[ newKey ] = key; if ( match[1] === 'o' ) { _fnHungarianMap( o[key] ); } } } ); o._hungarianMap = map; } /** * Convert from camel case parameters to Hungarian, based on a Hungarian map * created by _fnHungarianMap. * @param {object} src The model object which holds all parameters that can be * mapped. * @param {object} user The object to convert from camel case to Hungarian. * @param {boolean} force When set to `true`, properties which already have a * Hungarian value in the `user` object will be overwritten. Otherwise they * won't be. * @memberof DataTable#oApi */ function _fnCamelToHungarian ( src, user, force ) { if ( ! src._hungarianMap ) { _fnHungarianMap( src ); } var hungarianKey; $.each( user, function (key) { hungarianKey = src._hungarianMap[ key ]; if ( hungarianKey !== undefined && (force || user[hungarianKey] === undefined) ) { // For objects, we need to buzz down into the object to copy parameters if ( hungarianKey.charAt(0) === 'o' ) { // Copy the camelCase options over to the hungarian if ( ! user[ hungarianKey ] ) { user[ hungarianKey ] = {}; } $.extend( true, user[hungarianKey], user[key] ); _fnCamelToHungarian( src[hungarianKey], user[hungarianKey], force ); } else { user[hungarianKey] = user[ key ]; } } } ); } /** * Map one parameter onto another * @param {object} o Object to map * @param {*} knew The new parameter name * @param {*} old The old parameter name */ var _fnCompatMap = function ( o, knew, old ) { if ( o[ knew ] !== undefined ) { o[ old ] = o[ knew ]; } }; /** * Provide backwards compatibility for the main DT options. Note that the new * options are mapped onto the old parameters, so this is an external interface * change only. * @param {object} init Object to map */ function _fnCompatOpts ( init ) { _fnCompatMap( init, 'ordering', 'bSort' ); _fnCompatMap( init, 'orderMulti', 'bSortMulti' ); _fnCompatMap( init, 'orderClasses', 'bSortClasses' ); _fnCompatMap( init, 'orderCellsTop', 'bSortCellsTop' ); _fnCompatMap( init, 'order', 'aaSorting' ); _fnCompatMap( init, 'orderFixed', 'aaSortingFixed' ); _fnCompatMap( init, 'paging', 'bPaginate' ); _fnCompatMap( init, 'pagingType', 'sPaginationType' ); _fnCompatMap( init, 'pageLength', 'iDisplayLength' ); _fnCompatMap( init, 'searching', 'bFilter' ); // Boolean initialisation of x-scrolling if ( typeof init.sScrollX === 'boolean' ) { init.sScrollX = init.sScrollX ? '100%' : ''; } if ( typeof init.scrollX === 'boolean' ) { init.scrollX = init.scrollX ? '100%' : ''; } // Column search objects are in an array, so it needs to be converted // element by element var searchCols = init.aoSearchCols; if ( searchCols ) { for ( var i=0, ien=searchCols.length ; i<ien ; i++ ) { if ( searchCols[i] ) { _fnCamelToHungarian( DataTable.models.oSearch, searchCols[i] ); } } } // Enable search delay if server-side processing is enabled if (init.serverSide && ! init.searchDelay) { init.searchDelay = 400; } } /** * Provide backwards compatibility for column options. Note that the new options * are mapped onto the old parameters, so this is an external interface change * only. * @param {object} init Object to map */ function _fnCompatCols ( init ) { _fnCompatMap( init, 'orderable', 'bSortable' ); _fnCompatMap( init, 'orderData', 'aDataSort' ); _fnCompatMap( init, 'orderSequence', 'asSorting' ); _fnCompatMap( init, 'orderDataType', 'sortDataType' ); // orderData can be given as an integer var dataSort = init.aDataSort; if ( typeof dataSort === 'number' && ! Array.isArray( dataSort ) ) { init.aDataSort = [ dataSort ]; } } /** * Browser feature detection for capabilities, quirks * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnBrowserDetect( settings ) { // We don't need to do this every time DataTables is constructed, the values // calculated are specific to the browser and OS configuration which we // don't expect to change between initialisations if ( ! DataTable.__browser ) { var browser = {}; DataTable.__browser = browser; // Scrolling feature / quirks detection var n = $('<div/>') .css( { position: 'fixed', top: 0, left: -1 * window.pageXOffset, // allow for scrolling height: 1, width: 1, overflow: 'hidden' } ) .append( $('<div/>') .css( { position: 'absolute', top: 1, left: 1, width: 100, overflow: 'scroll' } ) .append( $('<div/>') .css( { width: '100%', height: 10 } ) ) ) .appendTo( 'body' ); var outer = n.children(); var inner = outer.children(); // Get scrollbar width browser.barWidth = outer[0].offsetWidth - outer[0].clientWidth; // In rtl text layout, some browsers (most, but not all) will place the // scrollbar on the left, rather than the right. browser.bScrollbarLeft = Math.round( inner.offset().left ) !== 1; n.remove(); } $.extend( settings.oBrowser, DataTable.__browser ); settings.oScroll.iBarWidth = DataTable.__browser.barWidth; } /** * Add a column to the list used for the table with default values * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnAddColumn( oSettings ) { // Add column to aoColumns array var oDefaults = DataTable.defaults.column; var iCol = oSettings.aoColumns.length; var oCol = $.extend( {}, DataTable.models.oColumn, oDefaults, { "aDataSort": oDefaults.aDataSort ? oDefaults.aDataSort : [iCol], "mData": oDefaults.mData ? oDefaults.mData : iCol, idx: iCol, searchFixed: {}, colEl: $('<col>').attr('data-dt-column', iCol) } ); oSettings.aoColumns.push( oCol ); // Add search object for column specific search. Note that the `searchCols[ iCol ]` // passed into extend can be undefined. This allows the user to give a default // with only some of the parameters defined, and also not give a default var searchCols = oSettings.aoPreSearchCols; searchCols[ iCol ] = $.extend( {}, DataTable.models.oSearch, searchCols[ iCol ] ); } /** * Apply options for a column * @param {object} oSettings dataTables settings object * @param {int} iCol column index to consider * @param {object} oOptions object with sType, bVisible and bSearchable etc * @memberof DataTable#oApi */ function _fnColumnOptions( oSettings, iCol, oOptions ) { var oCol = oSettings.aoColumns[ iCol ]; /* User specified column options */ if ( oOptions !== undefined && oOptions !== null ) { // Backwards compatibility _fnCompatCols( oOptions ); // Map camel case parameters to their Hungarian counterparts _fnCamelToHungarian( DataTable.defaults.column, oOptions, true ); /* Backwards compatibility for mDataProp */ if ( oOptions.mDataProp !== undefined && !oOptions.mData ) { oOptions.mData = oOptions.mDataProp; } if ( oOptions.sType ) { oCol._sManualType = oOptions.sType; } // `class` is a reserved word in Javascript, so we need to provide // the ability to use a valid name for the camel case input if ( oOptions.className && ! oOptions.sClass ) { oOptions.sClass = oOptions.className; } var origClass = oCol.sClass; $.extend( oCol, oOptions ); _fnMap( oCol, oOptions, "sWidth", "sWidthOrig" ); // Merge class from previously defined classes with this one, rather than just // overwriting it in the extend above if (origClass !== oCol.sClass) { oCol.sClass = origClass + ' ' + oCol.sClass; } /* iDataSort to be applied (backwards compatibility), but aDataSort will take * priority if defined */ if ( oOptions.iDataSort !== undefined ) { oCol.aDataSort = [ oOptions.iDataSort ]; } _fnMap( oCol, oOptions, "aDataSort" ); } /* Cache the data get and set functions for speed */ var mDataSrc = oCol.mData; var mData = _fnGetObjectDataFn( mDataSrc ); // The `render` option can be given as an array to access the helper rendering methods. // The first element is the rendering method to use, the rest are the parameters to pass if ( oCol.mRender && Array.isArray( oCol.mRender ) ) { var copy = oCol.mRender.slice(); var name = copy.shift(); oCol.mRender = DataTable.render[name].apply(window, copy); } oCol._render = oCol.mRender ? _fnGetObjectDataFn( oCol.mRender ) : null; var attrTest = function( src ) { return typeof src === 'string' && src.indexOf('@') !== -1; }; oCol._bAttrSrc = $.isPlainObject( mDataSrc ) && ( attrTest(mDataSrc.sort) || attrTest(mDataSrc.type) || attrTest(mDataSrc.filter) ); oCol._setter = null; oCol.fnGetData = function (rowData, type, meta) { var innerData = mData( rowData, type, undefined, meta ); return oCol._render && type ? oCol._render( innerData, type, rowData, meta ) : innerData; }; oCol.fnSetData = function ( rowData, val, meta ) { return _fnSetObjectDataFn( mDataSrc )( rowData, val, meta ); }; // Indicate if DataTables should read DOM data as an object or array // Used in _fnGetRowElements if ( typeof mDataSrc !== 'number' && ! oCol._isArrayHost ) { oSettings._rowReadObject = true; } /* Feature sorting overrides column specific when off */ if ( !oSettings.oFeatures.bSort ) { oCol.bSortable = false; } } /** * Adjust the table column widths for new data. Note: you would probably want to * do a redraw after calling this function! * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnAdjustColumnSizing ( settings ) { _fnCalculateColumnWidths( settings ); _fnColumnSizes( settings ); var scroll = settings.oScroll; if ( scroll.sY !== '' || scroll.sX !== '') { _fnScrollDraw( settings ); } _fnCallbackFire( settings, null, 'column-sizing', [settings] ); } /** * Apply column sizes * * @param {*} settings DataTables settings object */ function _fnColumnSizes ( settings ) { var cols = settings.aoColumns; for (var i=0 ; i<cols.length ; i++) { var width = _fnColumnsSumWidth(settings, [i], false, false); cols[i].colEl.css('width', width); } } /** * Convert the index of a visible column to the index in the data array (take account * of hidden columns) * @param {object} oSettings dataTables settings object * @param {int} iMatch Visible column index to lookup * @returns {int} i the data index * @memberof DataTable#oApi */ function _fnVisibleToColumnIndex( oSettings, iMatch ) { var aiVis = _fnGetColumns( oSettings, 'bVisible' ); return typeof aiVis[iMatch] === 'number' ? aiVis[iMatch] : null; } /** * Convert the index of an index in the data array and convert it to the visible * column index (take account of hidden columns) * @param {int} iMatch Column index to lookup * @param {object} oSettings dataTables settings object * @returns {int} i the data index * @memberof DataTable#oApi */ function _fnColumnIndexToVisible( oSettings, iMatch ) { var aiVis = _fnGetColumns( oSettings, 'bVisible' ); var iPos = aiVis.indexOf(iMatch); return iPos !== -1 ? iPos : null; } /** * Get the number of visible columns * @param {object} oSettings dataTables settings object * @returns {int} i the number of visible columns * @memberof DataTable#oApi */ function _fnVisbleColumns( settings ) { var layout = settings.aoHeader; var columns = settings.aoColumns; var vis = 0; if ( layout.length ) { for ( var i=0, ien=layout[0].length ; i<ien ; i++ ) { if ( columns[i].bVisible && $(layout[0][i].cell).css('display') !== 'none' ) { vis++; } } } return vis; } /** * Get an array of column indexes that match a given property * @param {object} oSettings dataTables settings object * @param {string} sParam Parameter in aoColumns to look for - typically * bVisible or bSearchable * @returns {array} Array of indexes with matched properties * @memberof DataTable#oApi */ function _fnGetColumns( oSettings, sParam ) { var a = []; oSettings.aoColumns.map( function(val, i) { if ( val[sParam] ) { a.push( i ); } } ); return a; } /** * Calculate the 'type' of a column * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnColumnTypes ( settings ) { var columns = settings.aoColumns; var data = settings.aoData; var types = DataTable.ext.type.detect; var i, ien, j, jen, k, ken; var col, detectedType, cache; // For each column, spin over the for ( i=0, ien=columns.length ; i<ien ; i++ ) { col = columns[i]; cache = []; if ( ! col.sType && col._sManualType ) { col.sType = col._sManualType; } else if ( ! col.sType ) { for ( j=0, jen=types.length ; j<jen ; j++ ) { for ( k=0, ken=data.length ; k<ken ; k++ ) { if (! data[k]) { continue; } // Use a cache array so we only need to get the type data // from the formatter once (when using multiple detectors) if ( cache[k] === undefined ) { cache[k] = _fnGetCellData( settings, k, i, 'type' ); } detectedType = types[j]( cache[k], settings ); // If null, then this type can't apply to this column, so // rather than testing all cells, break out. There is an // exception for the last type which is `html`. We need to // scan all rows since it is possible to mix string and HTML // types if ( ! detectedType && j !== types.length-2 ) { break; } // Only a single match is needed for html type since it is // bottom of the pile and very similar to string - but it // must not be empty if ( detectedType === 'html' && ! _empty(cache[k]) ) { break; } } // Type is valid for all data points in the column - use this // type if ( detectedType ) { col.sType = detectedType; break; } } // Fall back - if no type was detected, always use string if ( ! col.sType ) { col.sType = 'string'; } } // Set class names for header / footer for auto type classes var autoClass = _ext.type.className[col.sType]; if (autoClass) { _columnAutoClass(settings.aoHeader, i, autoClass); _columnAutoClass(settings.aoFooter, i, autoClass); } var renderer = _ext.type.render[col.sType]; // This can only happen once! There is no way to remover // a renderer. After the first time the renderer has // already been set so createTr will run the renderer itself. if (renderer && ! col._render) { col._render = DataTable.util.get(renderer); _columnAutoRender(settings, i); } } } /** * Apply an auto detected renderer to data which doesn't yet have * a renderer */ function _columnAutoRender(settings, colIdx) { var data = settings.aoData; for (var i=0 ; i<data.length ; i++) { if (data[i].nTr) { // We have to update the display here since there is no // invalidation check for the data var display = _fnGetCellData( settings, i, colIdx, 'display' ); data[i].displayData[colIdx] = display; _fnWriteCell(data[i].anCells[colIdx], display); // No need to update sort / filter data since it has // been invalidated and will be re-read with the // renderer now applied } } } /** * Apply a class name to a column's header cells */ function _columnAutoClass(container, colIdx, className) { container.forEach(function (row) { if (row[colIdx] && row[colIdx].unique) { _addClass(row[colIdx].cell, className); } }); } /** * Take the column definitions and static columns arrays and calculate how * they relate to column indexes. The callback function will then apply the * definition found for a column to a suitable configuration object. * @param {object} oSettings dataTables settings object * @param {array} aoColDefs The aoColumnDefs array that is to be applied * @param {array} aoCols The aoColumns array that defines columns individually * @param {array} headerLayout Layout for header as it was loaded * @param {function} fn Callback function - takes two parameters, the calculated * column index and the definition for that column. * @memberof DataTable#oApi */ function _fnApplyColumnDefs( oSettings, aoColDefs, aoCols, headerLayout, fn ) { var i, iLen, j, jLen, k, kLen, def; var columns = oSettings.aoColumns; if ( aoCols ) { for ( i=0, iLen=aoCols.length ; i<iLen ; i++ ) { if (aoCols[i] && aoCols[i].name) { columns[i].sName = aoCols[i].name; } } } // Column definitions with aTargets if ( aoColDefs ) { /* Loop over the definitions array - loop in reverse so first instance has priority */ for ( i=aoColDefs.length-1 ; i>=0 ; i-- ) { def = aoColDefs[i]; /* Each definition can target multiple columns, as it is an array */ var aTargets = def.target !== undefined ? def.target : def.targets !== undefined ? def.targets : def.aTargets; if ( ! Array.isArray( aTargets ) ) { aTargets = [ aTargets ]; } for ( j=0, jLen=aTargets.length ; j<jLen ; j++ ) { var target = aTargets[j]; if ( typeof target === 'number' && target >= 0 ) { /* Add columns that we don't yet know about */ while( columns.length <= target ) { _fnAddColumn( oSettings ); } /* Integer, basic index */ fn( target, def ); } else if ( typeof target === 'number' && target < 0 ) { /* Negative integer, right to left column counting */ fn( columns.length+target, def ); } else if ( typeof target === 'string' ) { for ( k=0, kLen=columns.length ; k<kLen ; k++ ) { if (target === '_all') { // Apply to all columns fn( k, def ); } else if (target.indexOf(':name') !== -1) { // Column selector if (columns[k].sName === target.replace(':name', '')) { fn( k, def ); } } else { // Cell selector headerLayout.forEach(function (row) { if (row[k]) { var cell = $(row[k].cell); // Legacy support. Note that it means that we don't support // an element name selector only, since they are treated as // class names for 1.x compat. if (target.match(/^[a-z][\w-]*$/i)) { target = '.' + target; } if (cell.is( target )) { fn( k, def ); } } }); } } } } } } // Statically defined columns array if ( aoCols ) { for ( i=0, iLen=aoCols.length ; i<iLen ; i++ ) { fn( i, aoCols[i] ); } } } /** * Get the width for a given set of columns * * @param {*} settings DataTables settings object * @param {*} targets Columns - comma separated string or array of numbers * @param {*} original Use the original width (true) or calculated (false) * @param {*} incVisible Include visible columns (true) or not (false) * @returns Combined CSS value */ function _fnColumnsSumWidth( settings, targets, original, incVisible ) { if ( ! Array.isArray( targets ) ) { targets = _fnColumnsFromHeader( targets ); } var sum = 0; var unit; var columns = settings.aoColumns; for ( var i=0, ien=targets.length ; i<ien ; i++ ) { var column = columns[ targets[i] ]; var definedWidth = original ? column.sWidthOrig : column.sWidth; if ( ! incVisible && column.bVisible === false ) { continue; } if ( definedWidth === null || definedWidth === undefined ) { return null; // can't determine a defined width - browser defined } else if ( typeof definedWidth === 'number' ) { unit = 'px'; sum += definedWidth; } else { var matched = definedWidth.match(/([\d\.]+)([^\d]*)/); if ( matched ) { sum += matched[1] * 1; unit = matched.length === 3 ? matched[2] : 'px'; } } } return sum + unit; } function _fnColumnsFromHeader( cell ) { var attr = $(cell).closest('[data-dt-column]').attr('data-dt-column'); if ( ! attr ) { return []; } return attr.split(',').map( function (val) { return val * 1; } ); } /** * Add a data array to the table, creating DOM node etc. This is the parallel to * _fnGatherData, but for adding rows from a Javascript source, rather than a * DOM source. * @param {object} settings dataTables settings object * @param {array} data data array to be added * @param {node} [tr] TR element to add to the table - optional. If not given, * DataTables will create a row automatically * @param {array} [tds] Array of TD|TH elements for the row - must be given * if nTr is. * @returns {int} >=0 if successful (index of new aoData entry), -1 if failed * @memberof DataTable#oApi */ function _fnAddData ( settings, dataIn, tr, tds ) { /* Create the object for storing information about this new row */ var rowIdx = settings.aoData.length; var rowModel = $.extend( true, {}, DataTable.models.oRow, { src: tr ? 'dom' : 'data', idx: rowIdx } ); rowModel._aData = dataIn; settings.aoData.push( rowModel ); var columns = settings.aoColumns; for ( var i=0, iLen=columns.length ; i<iLen ; i++ ) { // Invalidate the column types as the new data needs to be revalidated columns[i].sType = null; } /* Add to the display array */ settings.aiDisplayMaster.push( rowIdx ); var id = settings.rowIdFn( dataIn ); if ( id !== undefined ) { settings.aIds[ id ] = rowModel; } /* Create the DOM information, or register it if already present */ if ( tr || ! settings.oFeatures.bDeferRender ) { _fnCreateTr( settings, rowIdx, tr, tds ); } return rowIdx; } /** * Add one or more TR elements to the table. Generally we'd expect to * use this for reading data from a DOM sourced table, but it could be * used for an TR element. Note that if a TR is given, it is used (i.e. * it is not cloned). * @param {object} settings dataTables settings object * @param {array|node|jQuery} trs The TR element(s) to add to the table * @returns {array} Array of indexes for the added rows * @memberof DataTable#oApi */ function _fnAddTr( settings, trs ) { var row; // Allow an individual node to be passed in if ( ! (trs instanceof $) ) { trs = $(trs); } return trs.map( function (i, el) { row = _fnGetRowElements( settings, el ); return _fnAddData( settings, row.data, el, row.cells ); } ); } /** * Get the data for a given cell from the internal cache, taking into account data mapping * @param {object} settings dataTables settings object * @param {int} rowIdx aoData row id * @param {int} colIdx Column index * @param {string} type data get type ('display', 'type' 'filter|search' 'sort|order') * @returns {*} Cell data * @memberof DataTable#oApi */ function _fnGetCellData( settings, rowIdx, colIdx, type ) { if (type === 'search') { type = 'filter'; } else if (type === 'order') { type = 'sort'; } var row = settings.aoData[rowIdx]; if (! row) { return undefined; } var draw = settings.iDraw; var col = settings.aoColumns[colIdx]; var rowData = row._aData; var defaultContent = col.sDefaultContent; var cellData = col.fnGetData( rowData, type, { settings: settings, row: rowIdx, col: colIdx } ); // Allow for a node being returned for non-display types if (type !== 'display' && cellData && typeof cellData === 'object' && cellData.nodeName) { cellData = cellData.innerHTML; } if ( cellData === undefined ) { if ( settings.iDrawError != draw && defaultContent === null ) { _fnLog( settings, 0, "Requested unknown parameter "+ (typeof col.mData=='function' ? '{function}' : "'"+col.mData+"'")+ " for row "+rowIdx+", column "+colIdx, 4 ); settings.iDrawError = draw; } return defaultContent; } // When the data source is null and a specific data type is requested (i.e. // not the original data), we can use default column data if ( (cellData === rowData || cellData === null) && defaultContent !== null && type !== undefined ) { cellData = defaultContent; } else if ( typeof cellData === 'function' ) { // If the data source is a function, then we run it and use the return, // executing in the scope of the data object (for instances) return cellData.call( rowData ); } if ( cellData === null && type === 'display' ) { return ''; } if ( type === 'filter' ) { var fomatters = DataTable.ext.type.search; if ( fomatters[ col.sType ] ) { cellData = fomatters[ col.sType ]( cellData ); } } return cellData; } /** * Set the value for a specific cell, into the internal data cache * @param {object} settings dataTables settings object * @param {int} rowIdx aoData row id * @param {int} colIdx Column index * @param {*} val Value to set * @memberof DataTable#oApi */ function _fnSetCellData( settings, rowIdx, colIdx, val ) { var col = settings.aoColumns[colIdx]; var rowData = settings.aoData[rowIdx]._aData; col.fnSetData( rowData, val, { settings: settings, row: rowIdx, col: colIdx } ); } /** * Write a value to a cell * @param {*} td Cell * @param {*} val Value */ function _fnWriteCell(td, val) { if (val && typeof val === 'object' && val.nodeName) { $(td) .empty() .append(val); } else { td.innerHTML = val; } } // Private variable that is used to match action syntax in the data property object var __reArray = /\[.*?\]$/; var __reFn = /\(\)$/; /** * Split string on periods, taking into account escaped periods * @param {string} str String to split * @return {array} Split string */ function _fnSplitObjNotation( str ) { var parts = str.match(/(\\.|[^.])+/g) || ['']; return parts.map( function ( s ) { return s.replace(/\\\./g, '.'); } ); } /** * Return a function that can be used to get data from a source object, taking * into account the ability to use nested objects as a source * @param {string|int|function} mSource The data source for the object * @returns {function} Data get function * @memberof DataTable#oApi */ var _fnGetObjectDataFn = DataTable.util.get; /** * Return a function that can be used to set data from a source object, taking * into account the ability to use nested objects as a source * @param {string|int|function} mSource The data source for the object * @returns {function} Data set function * @memberof DataTable#oApi */ var _fnSetObjectDataFn = DataTable.util.set; /** * Return an array with the full table data * @param {object} oSettings dataTables settings object * @returns array {array} aData Master data array * @memberof DataTable#oApi */ function _fnGetDataMaster ( settings ) { return _pluck( settings.aoData, '_aData' ); } /** * Nuke the table * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnClearTable( settings ) { settings.aoData.length = 0; settings.aiDisplayMaster.length = 0; settings.aiDisplay.length = 0; settings.aIds = {}; } /** * Mark cached data as invalid such that a re-read of the data will occur when * the cached data is next requested. Also update from the data source object. * * @param {object} settings DataTables settings object * @param {int} rowIdx Row index to invalidate * @param {string} [src] Source to invalidate from: undefined, 'auto', 'dom' * or 'data' * @param {int} [colIdx] Column index to invalidate. If undefined the whole * row will be invalidated * @memberof DataTable#oApi * * @todo For the modularisation of v1.11 this will need to become a callback, so * the sort and filter methods can subscribe to it. That will required * initialisation options for sorting, which is why it is not already baked in */ function _fnInvalidate( settings, rowIdx, src, colIdx ) { var row = settings.aoData[ rowIdx ]; var i, ien; // Remove the cached data for the row row._aSortData = null; row._aFilterData = null; row.displayData = null; // Are we reading last data from DOM or the data object? if ( src === 'dom' || ((! src || src === 'auto') && row.src === 'dom') ) { // Read the data from the DOM row._aData = _fnGetRowElements( settings, row, colIdx, colIdx === undefined ? undefined : row._aData ) .data; } else { // Reading from data object, update the DOM var cells = row.anCells; var display = _fnGetRowDisplay(settings, rowIdx); if ( cells ) { if ( colIdx !== undefined ) { _fnWriteCell(cells[colIdx], display[colIdx]); } else { for ( i=0, ien=cells.length ; i<ien ; i++ ) { _fnWriteCell(cells[i], display[i]); } } } } // Column specific invalidation var cols = settings.aoColumns; if ( colIdx !== undefined ) { // Type - the data might have changed cols[ colIdx ].sType = null; // Max length string. Its a fairly cheep recalculation, so not worth // something more complicated cols[ colIdx ].maxLenString = null; } else { for ( i=0, ien=cols.length ; i<ien ; i++ ) { cols[i].sType = null; cols[i].maxLenString = null; } // Update DataTables special `DT_*` attributes for the row _fnRowAttributes( settings, row ); } } /** * Build a data source object from an HTML row, reading the contents of the * cells that are in the row. * * @param {object} settings DataTables settings object * @param {node|object} TR element from which to read data or existing row * object from which to re-read the data from the cells * @param {int} [colIdx] Optional column index * @param {array|object} [d] Data source object. If `colIdx` is given then this * parameter should also be given and will be used to write the data into. * Only the column in question will be written * @returns {object} Object with two parameters: `data` the data read, in * document order, and `cells` and array of nodes (they can be useful to the * caller, so rather than needing a second traversal to get them, just return * them from here). * @memberof DataTable#oApi */ function _fnGetRowElements( settings, row, colIdx, d ) { var tds = [], td = row.firstChild, name, col, i=0, contents, columns = settings.aoColumns, objectRead = settings._rowReadObject; // Allow the data object to be passed in, or construct d = d !== undefined ? d : objectRead ? {} : []; var attr = function ( str, td ) { if ( typeof str === 'string' ) { var idx = str.indexOf('@'); if ( idx !== -1 ) { var attr = str.substring( idx+1 ); var setter = _fnSetObjectDataFn( str ); setter( d, td.getAttribute( attr ) ); } } }; // Read data from a cell and store into the data object var cellProcess = function ( cell ) { if ( colIdx === undefined || colIdx === i ) { col = columns[i]; contents = (cell.innerHTML).trim(); if ( col && col._bAttrSrc ) { var setter = _fnSetObjectDataFn( col.mData._ ); setter( d, contents ); attr( col.mData.sort, cell ); attr( col.mData.type, cell ); attr( col.mData.filter, cell ); } else { // Depending on the `data` option for the columns the data can // be read to either an object or an array. if ( objectRead ) { if ( ! col._setter ) { // Cache the setter function col._setter = _fnSetObjectDataFn( col.mData ); } col._setter( d, contents ); } else { d[i] = contents; } } } i++; }; if ( td ) { // `tr` element was passed in while ( td ) { name = td.nodeName.toUpperCase(); if ( name == "TD" || name == "TH" ) { cellProcess( td ); tds.push( td ); } td = td.nextSibling; } } else { // Existing row object passed in tds = row.anCells; for ( var j=0, jen=tds.length ; j<jen ; j++ ) { cellProcess( tds[j] ); } } // Read the ID from the DOM if present var rowNode = row.firstChild ? row : row.nTr; if ( rowNode ) { var id = rowNode.getAttribute( 'id' ); if ( id ) { _fnSetObjectDataFn( settings.rowId )( d, id ); } } return { data: d, cells: tds }; } /** * Render and cache a row's display data for the columns, if required * @returns */ function _fnGetRowDisplay (settings, rowIdx) { let rowModal = settings.aoData[rowIdx]; let columns = settings.aoColumns; if (! rowModal.displayData) { // Need to render and cache rowModal.displayData = []; for ( var colIdx=0, len=columns.length ; colIdx<len ; colIdx++ ) { rowModal.displayData.push( _fnGetCellData( settings, rowIdx, colIdx, 'display' ) ); } } return rowModal.displayData; } /** * Create a new TR element (and it's TD children) for a row * @param {object} oSettings dataTables settings object * @param {int} iRow Row to consider * @param {node} [nTrIn] TR element to add to the table - optional. If not given, * DataTables will create a row automatically * @param {array} [anTds] Array of TD|TH elements for the row - must be given * if nTr is. * @memberof DataTable#oApi */ function _fnCreateTr ( oSettings, iRow, nTrIn, anTds ) { var row = oSettings.aoData[iRow], rowData = row._aData, cells = [], nTr, nTd, oCol, i, iLen, create, trClass = oSettings.oClasses.tbody.row; if ( row.nTr === null ) { nTr = nTrIn || document.createElement('tr'); row.nTr = nTr; row.anCells = cells; _addClass(nTr, trClass); /* Use a private property on the node to allow reserve mapping from the node * to the aoData array for fast look up */ nTr._DT_RowIndex = iRow; /* Special parameters can be given by the data source to be used on the row */ _fnRowAttributes( oSettings, row ); /* Process each column */ for ( i=0, iLen=oSettings.aoColumns.length ; i<iLen ; i++ ) { oCol = oSettings.aoColumns[i]; create = nTrIn && anTds[i] ? false : true; nTd = create ? document.createElement( oCol.sCellType ) : anTds[i]; if (! nTd) { _fnLog( oSettings, 0, 'Incorrect column count', 18 ); } nTd._DT_CellIndex = { row: iRow, column: i }; cells.push( nTd ); var display = _fnGetRowDisplay(oSettings, iRow); // Need to create the HTML if new, or if a rendering function is defined if ( create || ( (oCol.mRender || oCol.mData !== i) && (!$.isPlainObject(oCol.mData) || oCol.mData._ !== i+'.display') ) ) { _fnWriteCell(nTd, display[i]); } // Visibility - add or remove as required if ( oCol.bVisible && create ) { nTr.appendChild( nTd ); } else if ( ! oCol.bVisible && ! create ) { nTd.parentNode.removeChild( nTd ); } if ( oCol.fnCreatedCell ) { oCol.fnCreatedCell.call( oSettings.oInstance, nTd, _fnGetCellData( oSettings, iRow, i ), rowData, iRow, i ); } } _fnCallbackFire( oSettings, 'aoRowCreatedCallback', 'row-created', [nTr, rowData, iRow, cells] ); } else { _addClass(row.nTr, trClass); } } /** * Add attributes to a row based on the special `DT_*` parameters in a data * source object. * @param {object} settings DataTables settings object * @param {object} DataTables row object for the row to be modified * @memberof DataTable#oApi */ function _fnRowAttributes( settings, row ) { var tr = row.nTr; var data = row._aData; if ( tr ) { var id = settings.rowIdFn( data ); if ( id ) { tr.id = id; } if ( data.DT_RowClass ) { // Remove any classes added by DT_RowClass before var a = data.DT_RowClass.split(' '); row.__rowc = row.__rowc ? _unique( row.__rowc.concat( a ) ) : a; $(tr) .removeClass( row.__rowc.join(' ') ) .addClass( data.DT_RowClass ); } if ( data.DT_RowAttr ) { $(tr).attr( data.DT_RowAttr ); } if ( data.DT_RowData ) { $(tr).data( data.DT_RowData ); } } } /** * Create the HTML header for the table * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnBuildHead( settings, side ) { var classes = settings.oClasses; var columns = settings.aoColumns; var i, ien, row; var target = side === 'header' ? settings.nTHead : settings.nTFoot; var titleProp = side === 'header' ? 'sTitle' : side; // Footer might be defined if (! target) { return; } // If no cells yet and we have content for them, then create if (side === 'header' || _pluck(settings.aoColumns, titleProp).join('')) { row = $('tr', target); // Add a row if needed if (! row.length) { row = $('<tr/>').appendTo(target) } // Add the number of cells needed to make up to the number of columns if (row.length === 1) { var cells = $('td, th', row); for ( i=cells.length, ien=columns.length ; i<ien ; i++ ) { $('<th/>') .html( columns[i][titleProp] || '' ) .appendTo( row ); } } } var detected = _fnDetectHeader( settings, target, true ); if (side === 'header') { settings.aoHeader = detected; } else { settings.aoFooter = detected; } // ARIA role for the rows $(target).children('tr').attr('role', 'row'); // Every cell needs to be passed through the renderer $(target).children('tr').children('th, td') .each( function () { _fnRenderer( settings, side )( settings, $(this), classes ); } ); } /** * Build a layout structure for a header or footer * * @param {*} settings DataTables settings * @param {*} source Source layout array * @param {*} incColumns What columns should be included * @returns Layout array */ function _fnHeaderLayout( settings, source, incColumns ) { var row, column, cell; var local = []; var structure = []; var columns = settings.aoColumns; var columnCount = columns.length; var rowspan, colspan; if ( ! source ) { return; } // Default is to work on only visible columns if ( ! incColumns ) { incColumns = _range(columnCount) .filter(function (idx) { return columns[idx].bVisible; }); } // Make a copy of the master layout array, but with only the columns we want for ( row=0 ; row<source.length ; row++ ) { // Remove any columns we haven't selected local[row] = source[row].slice().filter(function (cell, i) { return incColumns.includes(i); }); // Prep the structure array - it needs an element for each row structure.push( [] ); } for ( row=0 ; row<local.length ; row++ ) { for ( column=0 ; column<local[row].length ; column++ ) { rowspan = 1; colspan = 1; // Check to see if there is already a cell (row/colspan) covering our target // insert point. If there is, then there is nothing to do. if ( structure[row][column] === undefined ) { cell = local[row][column].cell; // Expand for rowspan while ( local[row+rowspan] !== undefined && local[row][column].cell == local[row+rowspan][column].cell ) { structure[row+rowspan][column] = null; rowspan++; } // And for colspan while ( local[row][column+colspan] !== undefined && local[row][column].cell == local[row][column+colspan].cell ) { // Which also needs to go over rows for ( var k=0 ; k<rowspan ; k++ ) { structure[row+k][column+colspan] = null; } colspan++; } var titleSpan = $('span.dt-column-title', cell); structure[row][column] = { cell: cell, colspan: colspan, rowspan: rowspan, title: titleSpan.length ? titleSpan.html() : $(cell).html() }; } } } return structure; } /** * Draw the header (or footer) element based on the column visibility states. * * @param object oSettings dataTables settings object * @param array aoSource Layout array from _fnDetectHeader * @memberof DataTable#oApi */ function _fnDrawHead( settings, source ) { var layout = _fnHeaderLayout(settings, source); var tr, n; for ( var row=0 ; row<source.length ; row++ ) { tr = source[row].row; // All cells are going to be replaced, so empty out the row // Can't use $().empty() as that kills event handlers if (tr) { while( (n = tr.firstChild) ) { tr.removeChild( n ); } } for ( var column=0 ; column<layout[row].length ; column++ ) { var point = layout[row][column]; if (point) { $(point.cell) .appendTo(tr) .attr('rowspan', point.rowspan) .attr('colspan', point.colspan); } } } } /** * Insert the required TR nodes into the table for display * @param {object} oSettings dataTables settings object * @param ajaxComplete true after ajax call to complete rendering * @memberof DataTable#oApi */ function _fnDraw( oSettings, ajaxComplete ) { // Allow for state saving and a custom start position _fnStart( oSettings ); /* Provide a pre-callback function which can be used to cancel the draw is false is returned */ var aPreDraw = _fnCallbackFire( oSettings, 'aoPreDrawCallback', 'preDraw', [oSettings] ); if ( aPreDraw.indexOf(false) !== -1 ) { _fnProcessingDisplay( oSettings, false ); return; } var anRows = []; var iRowCount = 0; var bServerSide = _fnDataSource( oSettings ) == 'ssp'; var aiDisplay = oSettings.aiDisplay; var iDisplayStart = oSettings._iDisplayStart; var iDisplayEnd = oSettings.fnDisplayEnd(); var columns = oSettings.aoColumns; var body = $(oSettings.nTBody); oSettings.bDrawing = true; /* Server-side processing draw intercept */ if ( !bServerSide ) { oSettings.iDraw++; } else if ( !oSettings.bDestroying && !ajaxComplete) { // Show loading message for server-side processing if (oSettings.iDraw === 0) { body.empty().append(_emptyRow(oSettings)); } _fnAjaxUpdate( oSettings ); return; } if ( aiDisplay.length !== 0 ) { var iStart = bServerSide ? 0 : iDisplayStart; var iEnd = bServerSide ? oSettings.aoData.length : iDisplayEnd; for ( var j=iStart ; j<iEnd ; j++ ) { var iDataIndex = aiDisplay[j]; var aoData = oSettings.aoData[ iDataIndex ]; if ( aoData.nTr === null ) { _fnCreateTr( oSettings, iDataIndex ); } var nRow = aoData.nTr; // Add various classes as needed for (var i=0 ; i<columns.length ; i++) { var col = columns[i]; var td = aoData.anCells[i]; _addClass(td, _ext.type.className[col.sType]); // auto class _addClass(td, col.sClass); // column class _addClass(td, oSettings.oClasses.tbody.cell); // all cells } // Row callback functions - might want to manipulate the row // iRowCount and j are not currently documented. Are they at all // useful? _fnCallbackFire( oSettings, 'aoRowCallback', null, [nRow, aoData._aData, iRowCount, j, iDataIndex] ); anRows.push( nRow ); iRowCount++; } } else { anRows[ 0 ] = _emptyRow(oSettings); } /* Header and footer callbacks */ _fnCallbackFire( oSettings, 'aoHeaderCallback', 'header', [ $(oSettings.nTHead).children('tr')[0], _fnGetDataMaster( oSettings ), iDisplayStart, iDisplayEnd, aiDisplay ] ); _fnCallbackFire( oSettings, 'aoFooterCallback', 'footer', [ $(oSettings.nTFoot).children('tr')[0], _fnGetDataMaster( oSettings ), iDisplayStart, iDisplayEnd, aiDisplay ] ); // replaceChildren is faster, but only became widespread in 2020, // so a fall back in jQuery is provided for older browsers. if (body[0].replaceChildren) { body[0].replaceChildren.apply(body[0], anRows); } else { body.children().detach(); body.append( $(anRows) ); } // Empty table needs a specific class $(oSettings.nTableWrapper).toggleClass('dt-empty-footer', $('tr', oSettings.nTFoot).length === 0); /* Call all required callback functions for the end of a draw */ _fnCallbackFire( oSettings, 'aoDrawCallback', 'draw', [oSettings], true ); /* Draw is complete, sorting and filtering must be as well */ oSettings.bSorted = false; oSettings.bFiltered = false; oSettings.bDrawing = false; } /** * Redraw the table - taking account of the various features which are enabled * @param {object} oSettings dataTables settings object * @param {boolean} [holdPosition] Keep the current paging position. By default * the paging is reset to the first page * @memberof DataTable#oApi */ function _fnReDraw( settings, holdPosition, recompute ) { var features = settings.oFeatures, sort = features.bSort, filter = features.bFilter; if (recompute === undefined || recompute === true) { if ( sort ) { _fnSort( settings ); } if ( filter ) { _fnFilterComplete( settings, settings.oPreviousSearch ); } else { // No filtering, so we want to just use the display master settings.aiDisplay = settings.aiDisplayMaster.slice(); } } if ( holdPosition !== true ) { settings._iDisplayStart = 0; } // Let any modules know about the draw hold position state (used by // scrolling internally) settings._drawHold = holdPosition; _fnDraw( settings ); settings._drawHold = false; } /* * Table is empty - create a row with an empty message in it */ function _emptyRow ( settings ) { var oLang = settings.oLanguage; var zero = oLang.sZeroRecords; var dataSrc = _fnDataSource( settings ); if ( (settings.iDraw < 1 && dataSrc === 'ssp') || (settings.iDraw <= 1 && dataSrc === 'ajax') ) { zero = oLang.sLoadingRecords; } else if ( oLang.sEmptyTable && settings.fnRecordsTotal() === 0 ) { zero = oLang.sEmptyTable; } return $( '<tr/>' ) .append( $('<td />', { 'colSpan': _fnVisbleColumns( settings ), 'class': settings.oClasses.empty.row } ).html( zero ) )[0]; } /** * Convert a `layout` object given by a user to the object structure needed * for the renderer. This is done twice, once for above and once for below * the table. Ordering must also be considered. * * @param {*} settings DataTables settings object * @param {*} layout Layout object to convert * @param {string} side `top` or `bottom` * @returns Converted array structure - one item for each row. */ function _layoutArray ( settings, layout, side ) { var groups = {}; // Combine into like groups (e.g. `top`, `top2`, etc) $.each( layout, function ( pos, val ) { if (val === null) { return; } var splitPos = pos.replace(/([A-Z])/g, ' $1').split(' '); if ( ! groups[ splitPos[0] ] ) { groups[ splitPos[0] ] = {}; } var align = splitPos.length === 1 ? 'full' : splitPos[1].toLowerCase(); var group = groups[ splitPos[0] ]; var groupRun = function (contents, innerVal) { // If it is an object, then there can be multiple features contained in it if ( $.isPlainObject( innerVal ) ) { Object.keys(innerVal).map(function (key) { contents.push( { feature: key, opts: innerVal[key] }); }); } else { contents.push(innerVal); } } // Transform to an object with a contents property if (! group[align] || ! group[align].contents) { group[align] = { contents: [] }; } // Allow for an array or just a single object if ( Array.isArray(val)) { for (var i=0 ; i<val.length ; i++) { groupRun(group[align].contents, val[i]); } } else { groupRun(group[ align ].contents, val); } // And make contents an array if ( ! Array.isArray( group[ align ].contents ) ) { group[ align ].contents = [ group[ align ].contents ]; } } ); var filtered = Object.keys(groups) .map( function ( pos ) { // Filter to only the side we need if ( pos.indexOf(side) !== 0 ) { return null; } return { name: pos, val: groups[pos] }; } ) .filter( function (item) { return item !== null; }); // Order by item identifier filtered.sort( function ( a, b ) { var order1 = a.name.replace(/[^0-9]/g, '') * 1; var order2 = b.name.replace(/[^0-9]/g, '') * 1; return order2 - order1; } ); if ( side === 'bottom' ) { filtered.reverse(); } // Split into rows var rows = []; for ( var i=0, ien=filtered.length ; i<ien ; i++ ) { if ( filtered[i].val.full ) { rows.push( { full: filtered[i].val.full } ); _layoutResolve( settings, rows[ rows.length - 1 ] ); delete filtered[i].val.full; } if ( Object.keys(filtered[i].val).length ) { rows.push( filtered[i].val ); _layoutResolve( settings, rows[ rows.length - 1 ] ); } } return rows; } /** * Convert the contents of a row's layout object to nodes that can be inserted * into the document by a renderer. Execute functions, look up plug-ins, etc. * * @param {*} settings DataTables settings object * @param {*} row Layout object for this row */ function _layoutResolve( settings, row ) { var getFeature = function (feature, opts) { if ( ! _ext.features[ feature ] ) { _fnLog( settings, 0, 'Unknown feature: '+ feature ); } return _ext.features[ feature ].apply( this, [settings, opts] ); }; var resolve = function ( item ) { var line = row[ item ].contents; for ( var i=0, ien=line.length ; i<ien ; i++ ) { if ( ! line[i] ) { continue; } else if ( typeof line[i] === 'string' ) { line[i] = getFeature( line[i], null ); } else if ( $.isPlainObject(line[i]) ) { // If it's an object, it just has feature and opts properties from // the transform in _layoutArray line[i] = getFeature(line[i].feature, line[i].opts); } else if ( typeof line[i].node === 'function' ) { line[i] = line[i].node( settings ); } else if ( typeof line[i] === 'function' ) { var inst = line[i]( settings ); line[i] = typeof inst.node === 'function' ? inst.node() : inst; } } }; $.each( row, function ( key ) { resolve( key ); } ); } /** * Add the options to the page HTML for the table * @param {object} settings DataTables settings object * @memberof DataTable#oApi */ function _fnAddOptionsHtml ( settings ) { var classes = settings.oClasses; var table = $(settings.nTable); // Wrapper div around everything DataTables controls var insert = $('<div/>') .attr({ id: settings.sTableId+'_wrapper', 'class': classes.container }) .insertBefore(table); settings.nTableWrapper = insert[0]; if (settings.sDom) { // Legacy _fnLayoutDom(settings, settings.sDom, insert); } else { var top = _layoutArray( settings, settings.layout, 'top' ); var bottom = _layoutArray( settings, settings.layout, 'bottom' ); var renderer = _fnRenderer( settings, 'layout' ); // Everything above - the renderer will actually insert the contents into the document top.forEach(function (item) { renderer( settings, insert, item ); }); // The table - always the center of attention renderer( settings, insert, { full: { table: true, contents: [ _fnFeatureHtmlTable(settings) ] } } ); // Everything below bottom.forEach(function (item) { renderer( settings, insert, item ); }); } // Processing floats on top, so it isn't an inserted feature _processingHtml( settings ); } /** * Draw the table with the legacy DOM property * @param {*} settings DT settings object * @param {*} dom DOM string * @param {*} insert Insert point */ function _fnLayoutDom( settings, dom, insert ) { var parts = dom.match(/(".*?")|('.*?')|./g); var featureNode, option, newNode, next, attr; for ( var i=0 ; i<parts.length ; i++ ) { featureNode = null; option = parts[i]; if ( option == '<' ) { // New container div newNode = $('<div/>'); // Check to see if we should append an id and/or a class name to the container next = parts[i+1]; if ( next[0] == "'" || next[0] == '"' ) { attr = next.replace(/['"]/g, ''); var id = '', className; /* The attribute can be in the format of "#id.class", "#id" or "class" This logic * breaks the string into parts and applies them as needed */ if ( attr.indexOf('.') != -1 ) { var split = attr.split('.'); id = split[0]; className = split[1]; } else if ( attr[0] == "#" ) { id = attr; } else { className = attr; } newNode .attr('id', id.substring(1)) .addClass(className); i++; // Move along the position array } insert.append( newNode ); insert = newNode; } else if ( option == '>' ) { // End container div insert = insert.parent(); } else if ( option == 't' ) { // Table featureNode = _fnFeatureHtmlTable( settings ); } else { DataTable.ext.feature.forEach(function(feature) { if ( option == feature.cFeature ) { featureNode = feature.fnInit( settings ); } }); } // Add to the display if ( featureNode ) { insert.append( featureNode ); } } } /** * Use the DOM source to create up an array of header cells. The idea here is to * create a layout grid (array) of rows x columns, which contains a reference * to the cell that that point in the grid (regardless of col/rowspan), such that * any column / row could be removed and the new grid constructed * @param {node} thead The header/footer element for the table * @returns {array} Calculated layout array * @memberof DataTable#oApi */ function _fnDetectHeader ( settings, thead, write ) { var columns = settings.aoColumns; var rows = $(thead).children('tr'); var row, cell; var i, k, l, iLen, shifted, column, colspan, rowspan; var isHeader = thead && thead.nodeName.toLowerCase() === 'thead'; var layout = []; var unique; var shift = function ( a, i, j ) { var k = a[i]; while ( k[j] ) { j++; } return j; }; // We know how many rows there are in the layout - so prep it for ( i=0, iLen=rows.length ; i<iLen ; i++ ) { layout.push( [] ); } for ( i=0, iLen=rows.length ; i<iLen ; i++ ) { row = rows[i]; column = 0; // For every cell in the row.. cell = row.firstChild; while ( cell ) { if ( cell.nodeName.toUpperCase() == 'TD' || cell.nodeName.toUpperCase() == 'TH' ) { var cols = []; // Get the col and rowspan attributes from the DOM and sanitise them colspan = cell.getAttribute('colspan') * 1; rowspan = cell.getAttribute('rowspan') * 1; colspan = (!colspan || colspan===0 || colspan===1) ? 1 : colspan; rowspan = (!rowspan || rowspan===0 || rowspan===1) ? 1 : rowspan; // There might be colspan cells already in this row, so shift our target // accordingly shifted = shift( layout, i, column ); // Cache calculation for unique columns unique = colspan === 1 ? true : false; // Perform header setup if ( write ) { if (unique) { // Allow column options to be set from HTML attributes _fnColumnOptions( settings, shifted, $(cell).data() ); // Get the width for the column. This can be defined from the // width attribute, style attribute or `columns.width` option var columnDef = columns[shifted]; var width = cell.getAttribute('width') || null; var t = cell.style.width.match(/width:\s*(\d+[pxem%]+)/); if ( t ) { width = t[1]; } columnDef.sWidthOrig = columnDef.sWidth || width; if (isHeader) { // Column title handling - can be user set, or read from the DOM // This happens before the render, so the original is still in place if ( columnDef.sTitle !== null && ! columnDef.autoTitle ) { cell.innerHTML = columnDef.sTitle; } if (! columnDef.sTitle && unique) { columnDef.sTitle = _stripHtml(cell.innerHTML); columnDef.autoTitle = true; } } else { // Footer specific operations if (columnDef.footer) { cell.innerHTML = columnDef.footer; } } // Fall back to the aria-label attribute on the table header if no ariaTitle is // provided. if (! columnDef.ariaTitle) { columnDef.ariaTitle = $(cell).attr("aria-label") || columnDef.sTitle; } // Column specific class names if ( columnDef.className ) { $(cell).addClass( columnDef.className ); } } // Wrap the column title so we can write to it in future if ( $('span.dt-column-title', cell).length === 0) { $('<span>') .addClass('dt-column-title') .append(cell.childNodes) .appendTo(cell); } if ( isHeader && $('span.dt-column-order', cell).length === 0) { $('<span>') .addClass('dt-column-order') .appendTo(cell); } } // If there is col / rowspan, copy the information into the layout grid for ( l=0 ; l<colspan ; l++ ) { for ( k=0 ; k<rowspan ; k++ ) { layout[i+k][shifted+l] = { cell: cell, unique: unique }; layout[i+k].row = row; } cols.push( shifted+l ); } // Assign an attribute so spanning cells can still be identified // as belonging to a column cell.setAttribute('data-dt-column', _unique(cols).join(',')); } cell = cell.nextSibling; } } return layout; } /** * Set the start position for draw * @param {object} oSettings dataTables settings object */ function _fnStart( oSettings ) { var bServerSide = _fnDataSource( oSettings ) == 'ssp'; var iInitDisplayStart = oSettings.iInitDisplayStart; // Check and see if we have an initial draw position from state saving if ( iInitDisplayStart !== undefined && iInitDisplayStart !== -1 ) { oSettings._iDisplayStart = bServerSide ? iInitDisplayStart : iInitDisplayStart >= oSettings.fnRecordsDisplay() ? 0 : iInitDisplayStart; oSettings.iInitDisplayStart = -1; } } /** * Create an Ajax call based on the table's settings, taking into account that * parameters can have multiple forms, and backwards compatibility. * * @param {object} oSettings dataTables settings object * @param {array} data Data to send to the server, required by * DataTables - may be augmented by developer callbacks * @param {function} fn Callback function to run when data is obtained */ function _fnBuildAjax( oSettings, data, fn ) { var ajaxData; var ajax = oSettings.ajax; var instance = oSettings.oInstance; var callback = function ( json ) { var status = oSettings.jqXHR ? oSettings.jqXHR.status : null; if ( json === null || (typeof status === 'number' && status == 204 ) ) { json = {}; _fnAjaxDataSrc( oSettings, json, [] ); } var error = json.error || json.sError; if ( error ) { _fnLog( oSettings, 0, error ); } oSettings.json = json; _fnCallbackFire( oSettings, null, 'xhr', [oSettings, json, oSettings.jqXHR], true ); fn( json ); }; if ( $.isPlainObject( ajax ) && ajax.data ) { ajaxData = ajax.data; var newData = typeof ajaxData === 'function' ? ajaxData( data, oSettings ) : // fn can manipulate data or return ajaxData; // an object object or array to merge // If the function returned something, use that alone data = typeof ajaxData === 'function' && newData ? newData : $.extend( true, data, newData ); // Remove the data property as we've resolved it already and don't want // jQuery to do it again (it is restored at the end of the function) delete ajax.data; } var baseAjax = { "url": typeof ajax === 'string' ? ajax : '', "data": data, "success": callback, "dataType": "json", "cache": false, "type": oSettings.sServerMethod, "error": function (xhr, error) { var ret = _fnCallbackFire( oSettings, null, 'xhr', [oSettings, null, oSettings.jqXHR], true ); if ( ret.indexOf(true) === -1 ) { if ( error == "parsererror" ) { _fnLog( oSettings, 0, 'Invalid JSON response', 1 ); } else if ( xhr.readyState === 4 ) { _fnLog( oSettings, 0, 'Ajax error', 7 ); } } _fnProcessingDisplay( oSettings, false ); } }; // If `ajax` option is an object, extend and override our default base if ( $.isPlainObject( ajax ) ) { $.extend( baseAjax, ajax ) } // Store the data submitted for the API oSettings.oAjaxData = data; // Allow plug-ins and external processes to modify the data _fnCallbackFire( oSettings, null, 'preXhr', [oSettings, data, baseAjax], true ); if ( typeof ajax === 'function' ) { // Is a function - let the caller define what needs to be done oSettings.jqXHR = ajax.call( instance, data, callback, oSettings ); } else if (ajax.url === '') { // No url, so don't load any data. Just apply an empty data array // to the object for the callback. var empty = {}; DataTable.util.set(ajax.dataSrc)(empty, []); callback(empty); } else { // Object to extend the base settings oSettings.jqXHR = $.ajax( baseAjax ); // Restore for next time around if ( ajaxData ) { ajax.data = ajaxData; } } } /** * Update the table using an Ajax call * @param {object} settings dataTables settings object * @returns {boolean} Block the table drawing or not * @memberof DataTable#oApi */ function _fnAjaxUpdate( settings ) { settings.iDraw++; _fnProcessingDisplay( settings, true ); _fnBuildAjax( settings, _fnAjaxParameters( settings ), function(json) { _fnAjaxUpdateDraw( settings, json ); } ); } /** * Build up the parameters in an object needed for a server-side processing * request. * @param {object} oSettings dataTables settings object * @returns {bool} block the table drawing or not * @memberof DataTable#oApi */ function _fnAjaxParameters( settings ) { var columns = settings.aoColumns, features = settings.oFeatures, preSearch = settings.oPreviousSearch, preColSearch = settings.aoPreSearchCols, colData = function ( idx, prop ) { return typeof columns[idx][prop] === 'function' ? 'function' : columns[idx][prop]; }; return { draw: settings.iDraw, columns: columns.map( function ( column, i ) { return { data: colData(i, 'mData'), name: column.sName, searchable: column.bSearchable, orderable: column.bSortable, search: { value: preColSearch[i].search, regex: preColSearch[i].regex, fixed: Object.keys(column.searchFixed).map( function(name) { return { name: name, term: column.searchFixed[name].toString() } }) } }; } ), order: _fnSortFlatten( settings ).map( function ( val ) { return { column: val.col, dir: val.dir, name: colData(val.col, 'sName') }; } ), start: settings._iDisplayStart, length: features.bPaginate ? settings._iDisplayLength : -1, search: { value: preSearch.search, regex: preSearch.regex, fixed: Object.keys(settings.searchFixed).map( function(name) { return { name: name, term: settings.searchFixed[name].toString() } }) } }; } /** * Data the data from the server (nuking the old) and redraw the table * @param {object} oSettings dataTables settings object * @param {object} json json data return from the server. * @param {string} json.sEcho Tracking flag for DataTables to match requests * @param {int} json.iTotalRecords Number of records in the data set, not accounting for filtering * @param {int} json.iTotalDisplayRecords Number of records in the data set, accounting for filtering * @param {array} json.aaData The data to display on this page * @param {string} [json.sColumns] Column ordering (sName, comma separated) * @memberof DataTable#oApi */ function _fnAjaxUpdateDraw ( settings, json ) { var data = _fnAjaxDataSrc(settings, json); var draw = _fnAjaxDataSrcParam(settings, 'draw', json); var recordsTotal = _fnAjaxDataSrcParam(settings, 'recordsTotal', json); var recordsFiltered = _fnAjaxDataSrcParam(settings, 'recordsFiltered', json); if ( draw !== undefined ) { // Protect against out of sequence returns if ( draw*1 < settings.iDraw ) { return; } settings.iDraw = draw * 1; } // No data in returned object, so rather than an array, we show an empty table if ( ! data ) { data = []; } _fnClearTable( settings ); settings._iRecordsTotal = parseInt(recordsTotal, 10); settings._iRecordsDisplay = parseInt(recordsFiltered, 10); for ( var i=0, ien=data.length ; i<ien ; i++ ) { _fnAddData( settings, data[i] ); } settings.aiDisplay = settings.aiDisplayMaster.slice(); _fnDraw( settings, true ); _fnInitComplete( settings ); _fnProcessingDisplay( settings, false ); } /** * Get the data from the JSON data source to use for drawing a table. Using * `_fnGetObjectDataFn` allows the data to be sourced from a property of the * source object, or from a processing function. * @param {object} settings dataTables settings object * @param {object} json Data source object / array from the server * @return {array} Array of data to use */ function _fnAjaxDataSrc ( settings, json, write ) { var dataProp = 'data'; if ($.isPlainObject( settings.ajax ) && settings.ajax.dataSrc !== undefined) { // Could in inside a `dataSrc` object, or not! var dataSrc = settings.ajax.dataSrc; // string, function and object are valid types if (typeof dataSrc === 'string' || typeof dataSrc === 'function') { dataProp = dataSrc; } else if (dataSrc.data !== undefined) { dataProp = dataSrc.data; } } if ( ! write ) { if ( dataProp === 'data' ) { // If the default, then we still want to support the old style, and safely ignore // it if possible return json.aaData || json[dataProp]; } return dataProp !== "" ? _fnGetObjectDataFn( dataProp )( json ) : json; } // set _fnSetObjectDataFn( dataProp )( json, write ); } /** * Very similar to _fnAjaxDataSrc, but for the other SSP properties * @param {*} settings DataTables settings object * @param {*} param Target parameter * @param {*} json JSON data * @returns Resolved value */ function _fnAjaxDataSrcParam (settings, param, json) { var dataSrc = $.isPlainObject( settings.ajax ) ? settings.ajax.dataSrc : null; if (dataSrc && dataSrc[param]) { // Get from custom location return _fnGetObjectDataFn( dataSrc[param] )( json ); } // else - Default behaviour var old = ''; // Legacy support if (param === 'draw') { old = 'sEcho'; } else if (param === 'recordsTotal') { old = 'iTotalRecords'; } else if (param === 'recordsFiltered') { old = 'iTotalDisplayRecords'; } return json[old] !== undefined ? json[old] : json[param]; } /** * Filter the table using both the global filter and column based filtering * @param {object} settings dataTables settings object * @param {object} input search information * @memberof DataTable#oApi */ function _fnFilterComplete ( settings, input ) { var columnsSearch = settings.aoPreSearchCols; // Resolve any column types that are unknown due to addition or invalidation // @todo As per sort - can this be moved into an event handler? _fnColumnTypes( settings ); // In server-side processing all filtering is done by the server, so no point hanging around here if ( _fnDataSource( settings ) != 'ssp' ) { // Check if any of the rows were invalidated _fnFilterData( settings ); // Start from the full data set settings.aiDisplay = settings.aiDisplayMaster.slice(); // Global filter first _fnFilter( settings.aiDisplay, settings, input.search, input ); $.each(settings.searchFixed, function (name, term) { _fnFilter(settings.aiDisplay, settings, term, {}); }); // Then individual column filters for ( var i=0 ; i<columnsSearch.length ; i++ ) { var col = columnsSearch[i]; _fnFilter( settings.aiDisplay, settings, col.search, col, i ); $.each(settings.aoColumns[i].searchFixed, function (name, term) { _fnFilter(settings.aiDisplay, settings, term, {}, i); }); } // And finally global filtering _fnFilterCustom( settings ); } // Tell the draw function we have been filtering settings.bFiltered = true; _fnCallbackFire( settings, null, 'search', [settings] ); } /** * Apply custom filtering functions * * This is legacy now that we have named functions, but it is widely used * from 1.x, so it is not yet deprecated. * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnFilterCustom( settings ) { var filters = DataTable.ext.search; var displayRows = settings.aiDisplay; var row, rowIdx; for ( var i=0, ien=filters.length ; i<ien ; i++ ) { var rows = []; // Loop over each row and see if it should be included for ( var j=0, jen=displayRows.length ; j<jen ; j++ ) { rowIdx = displayRows[ j ]; row = settings.aoData[ rowIdx ]; if ( filters[i]( settings, row._aFilterData, rowIdx, row._aData, j ) ) { rows.push( rowIdx ); } } // So the array reference doesn't break set the results into the // existing array displayRows.length = 0; displayRows.push.apply(displayRows, rows); } } /** * Filter the data table based on user input and draw the table */ function _fnFilter( searchRows, settings, input, options, column ) { if ( input === '' ) { return; } var i = 0; var matched = []; // Search term can be a function, regex or string - if a string we apply our // smart filtering regex (assuming the options require that) var searchFunc = typeof input === 'function' ? input : null; var rpSearch = input instanceof RegExp ? input : searchFunc ? null : _fnFilterCreateSearch( input, options ); // Then for each row, does the test pass. If not, lop the row from the array for (i=0 ; i<searchRows.length ; i++) { var row = settings.aoData[ searchRows[i] ]; var data = column === undefined ? row._sFilterRow : row._aFilterData[ column ]; if ( (searchFunc && searchFunc(data, row._aData, searchRows[i], column)) || (rpSearch && rpSearch.test(data)) ) { matched.push(searchRows[i]); } } // Mutate the searchRows array searchRows.length = matched.length; for (i=0 ; i<matched.length ; i++) { searchRows[i] = matched[i]; } } /** * Build a regular expression object suitable for searching a table * @param {string} sSearch string to search for * @param {bool} bRegex treat as a regular expression or not * @param {bool} bSmart perform smart filtering or not * @param {bool} bCaseInsensitive Do case insensitive matching or not * @returns {RegExp} constructed object * @memberof DataTable#oApi */ function _fnFilterCreateSearch( search, inOpts ) { var not = []; var options = $.extend({}, { boundary: false, caseInsensitive: true, exact: false, regex: false, smart: true }, inOpts); if (typeof search !== 'string') { search = search.toString(); } // Remove diacritics if normalize is set up to do so search = _normalize(search); if (options.exact) { return new RegExp( '^'+_fnEscapeRegex(search)+'$', options.caseInsensitive ? 'i' : '' ); } search = options.regex ? search : _fnEscapeRegex( search ); if ( options.smart ) { /* For smart filtering we want to allow the search to work regardless of * word order. We also want double quoted text to be preserved, so word * order is important - a la google. And a negative look around for * finding rows which don't contain a given string. * * So this is the sort of thing we want to generate: * * ^(?=.*?\bone\b)(?=.*?\btwo three\b)(?=.*?\bfour\b).*$ */ var parts = search.match( /!?["\u201C][^"\u201D]+["\u201D]|[^ ]+/g ) || ['']; var a = parts.map( function ( word ) { var negative = false; var m; // Determine if it is a "does not include" if ( word.charAt(0) === '!' ) { negative = true; word = word.substring(1); } // Strip the quotes from around matched phrases if ( word.charAt(0) === '"' ) { m = word.match( /^"(.*)"$/ ); word = m ? m[1] : word; } else if ( word.charAt(0) === '\u201C' ) { // Smart quote match (iPhone users) m = word.match( /^\u201C(.*)\u201D$/ ); word = m ? m[1] : word; } // For our "not" case, we need to modify the string that is // allowed to match at the end of the expression. if (negative) { if (word.length > 1) { not.push('(?!'+word+')'); } word = ''; } return word.replace(/"/g, ''); } ); var match = not.length ? not.join('') : ''; var boundary = options.boundary ? '\\b' : ''; search = '^(?=.*?'+boundary+a.join( ')(?=.*?'+boundary )+')('+match+'.)*$'; } return new RegExp( search, options.caseInsensitive ? 'i' : '' ); } /** * Escape a string such that it can be used in a regular expression * @param {string} sVal string to escape * @returns {string} escaped string * @memberof DataTable#oApi */ var _fnEscapeRegex = DataTable.util.escapeRegex; var __filter_div = $('<div>')[0]; var __filter_div_textContent = __filter_div.textContent !== undefined; // Update the filtering data for each row if needed (by invalidation or first run) function _fnFilterData ( settings ) { var columns = settings.aoColumns; var data = settings.aoData; var column; var j, jen, filterData, cellData, row; var wasInvalidated = false; for ( var rowIdx=0 ; rowIdx<data.length ; rowIdx++ ) { if (! data[rowIdx]) { continue; } row = data[rowIdx]; if ( ! row._aFilterData ) { filterData = []; for ( j=0, jen=columns.length ; j<jen ; j++ ) { column = columns[j]; if ( column.bSearchable ) { cellData = _fnGetCellData( settings, rowIdx, j, 'filter' ); // Search in DataTables is string based if ( cellData === null ) { cellData = ''; } if ( typeof cellData !== 'string' && cellData.toString ) { cellData = cellData.toString(); } } else { cellData = ''; } // If it looks like there is an HTML entity in the string, // attempt to decode it so sorting works as expected. Note that // we could use a single line of jQuery to do this, but the DOM // method used here is much faster https://jsperf.com/html-decode if ( cellData.indexOf && cellData.indexOf('&') !== -1 ) { __filter_div.innerHTML = cellData; cellData = __filter_div_textContent ? __filter_div.textContent : __filter_div.innerText; } if ( cellData.replace ) { cellData = cellData.replace(/[\r\n\u2028]/g, ''); } filterData.push( cellData ); } row._aFilterData = filterData; row._sFilterRow = filterData.join(' '); wasInvalidated = true; } } return wasInvalidated; } /** * Draw the table for the first time, adding all required features * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnInitialise ( settings ) { var i, iAjaxStart=settings.iInitDisplayStart; /* Ensure that the table data is fully initialised */ if ( ! settings.bInitialised ) { setTimeout( function(){ _fnInitialise( settings ); }, 200 ); return; } /* Build and draw the header / footer for the table */ _fnBuildHead( settings, 'header' ); _fnBuildHead( settings, 'footer' ); _fnDrawHead( settings, settings.aoHeader ); _fnDrawHead( settings, settings.aoFooter ); // Enable features _fnAddOptionsHtml( settings ); _fnSortInit( settings ); _colGroup( settings ); /* Okay to show that something is going on now */ _fnProcessingDisplay( settings, true ); _fnCallbackFire( settings, null, 'preInit', [settings], true ); // If there is default sorting required - let's do it. The sort function // will do the drawing for us. Otherwise we draw the table regardless of the // Ajax source - this allows the table to look initialised for Ajax sourcing // data (show 'loading' message possibly) _fnReDraw( settings ); var dataSrc = _fnDataSource( settings ); // Server-side processing init complete is done by _fnAjaxUpdateDraw if ( dataSrc != 'ssp' ) { // if there is an ajax source load the data if ( dataSrc == 'ajax' ) { _fnBuildAjax( settings, {}, function(json) { var aData = _fnAjaxDataSrc( settings, json ); // Got the data - add it to the table for ( i=0 ; i<aData.length ; i++ ) { _fnAddData( settings, aData[i] ); } // Reset the init display for cookie saving. We've already done // a filter, and therefore cleared it before. So we need to make // it appear 'fresh' settings.iInitDisplayStart = iAjaxStart; _fnReDraw( settings ); _fnProcessingDisplay( settings, false ); _fnInitComplete( settings ); }, settings ); } else { _fnInitComplete( settings ); _fnProcessingDisplay( settings, false ); } } } /** * Draw the table for the first time, adding all required features * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnInitComplete ( settings ) { if (settings._bInitComplete) { return; } var args = [settings, settings.json]; settings._bInitComplete = true; // Table is fully set up and we have data, so calculate the // column widths _fnAdjustColumnSizing( settings ); _fnCallbackFire( settings, null, 'plugin-init', args, true ); _fnCallbackFire( settings, 'aoInitComplete', 'init', args, true ); } function _fnLengthChange ( settings, val ) { var len = parseInt( val, 10 ); settings._iDisplayLength = len; _fnLengthOverflow( settings ); // Fire length change event _fnCallbackFire( settings, null, 'length', [settings, len] ); } /** * Alter the display settings to change the page * @param {object} settings DataTables settings object * @param {string|int} action Paging action to take: "first", "previous", * "next" or "last" or page number to jump to (integer) * @param [bool] redraw Automatically draw the update or not * @returns {bool} true page has changed, false - no change * @memberof DataTable#oApi */ function _fnPageChange ( settings, action, redraw ) { var start = settings._iDisplayStart, len = settings._iDisplayLength, records = settings.fnRecordsDisplay(); if ( records === 0 || len === -1 ) { start = 0; } else if ( typeof action === "number" ) { start = action * len; if ( start > records ) { start = 0; } } else if ( action == "first" ) { start = 0; } else if ( action == "previous" ) { start = len >= 0 ? start - len : 0; if ( start < 0 ) { start = 0; } } else if ( action == "next" ) { if ( start + len < records ) { start += len; } } else if ( action == "last" ) { start = Math.floor( (records-1) / len) * len; } else if ( action === 'ellipsis' ) { return; } else { _fnLog( settings, 0, "Unknown paging action: "+action, 5 ); } var changed = settings._iDisplayStart !== start; settings._iDisplayStart = start; _fnCallbackFire( settings, null, changed ? 'page' : 'page-nc', [settings] ); if ( changed && redraw ) { _fnDraw( settings ); } return changed; } /** * Generate the node required for the processing node * @param {object} settings DataTables settings object */ function _processingHtml ( settings ) { var table = settings.nTable; var scrolling = settings.oScroll.sX !== '' || settings.oScroll.sY !== ''; if ( settings.oFeatures.bProcessing ) { var n = $('<div/>', { 'id': settings.sTableId + '_processing', 'class': settings.oClasses.processing.container, 'role': 'status' } ) .html( settings.oLanguage.sProcessing ) .append('<div><div></div><div></div><div></div><div></div></div>'); // Different positioning depending on if scrolling is enabled or not if (scrolling) { n.prependTo( $('div.dt-scroll', settings.nTableWrapper) ); } else { n.insertBefore( table ); } $(table).on( 'processing.dt.DT', function (e, s, show) { n.css( 'display', show ? 'block' : 'none' ); } ); } } /** * Display or hide the processing indicator * @param {object} settings DataTables settings object * @param {bool} show Show the processing indicator (true) or not (false) */ function _fnProcessingDisplay ( settings, show ) { _fnCallbackFire( settings, null, 'processing', [settings, show] ); } /** * Add any control elements for the table - specifically scrolling * @param {object} settings dataTables settings object * @returns {node} Node to add to the DOM * @memberof DataTable#oApi */ function _fnFeatureHtmlTable ( settings ) { var table = $(settings.nTable); // Scrolling from here on in var scroll = settings.oScroll; if ( scroll.sX === '' && scroll.sY === '' ) { return settings.nTable; } var scrollX = scroll.sX; var scrollY = scroll.sY; var classes = settings.oClasses.scrolling; var caption = settings.captionNode; var captionSide = caption ? caption._captionSide : null; var headerClone = $( table[0].cloneNode(false) ); var footerClone = $( table[0].cloneNode(false) ); var footer = table.children('tfoot'); var _div = '<div/>'; var size = function ( s ) { return !s ? null : _fnStringToCss( s ); }; if ( ! footer.length ) { footer = null; } /* * The HTML structure that we want to generate in this function is: * div - scroller * div - scroll head * div - scroll head inner * table - scroll head table * thead - thead * div - scroll body * table - table (master table) * thead - thead clone for sizing * tbody - tbody * div - scroll foot * div - scroll foot inner * table - scroll foot table * tfoot - tfoot */ var scroller = $( _div, { 'class': classes.container } ) .append( $(_div, { 'class': classes.header.self } ) .css( { overflow: 'hidden', position: 'relative', border: 0, width: scrollX ? size(scrollX) : '100%' } ) .append( $(_div, { 'class': classes.header.inner } ) .css( { 'box-sizing': 'content-box', width: scroll.sXInner || '100%' } ) .append( headerClone .removeAttr('id') .css( 'margin-left', 0 ) .append( captionSide === 'top' ? caption : null ) .append( table.children('thead') ) ) ) ) .append( $(_div, { 'class': classes.body } ) .css( { position: 'relative', overflow: 'auto', width: size( scrollX ) } ) .append( table ) ); if ( footer ) { scroller.append( $(_div, { 'class': classes.footer.self } ) .css( { overflow: 'hidden', border: 0, width: scrollX ? size(scrollX) : '100%' } ) .append( $(_div, { 'class': classes.footer.inner } ) .append( footerClone .removeAttr('id') .css( 'margin-left', 0 ) .append( captionSide === 'bottom' ? caption : null ) .append( table.children('tfoot') ) ) ) ); } var children = scroller.children(); var scrollHead = children[0]; var scrollBody = children[1]; var scrollFoot = footer ? children[2] : null; // When the body is scrolled, then we also want to scroll the headers $(scrollBody).on( 'scroll.DT', function () { var scrollLeft = this.scrollLeft; scrollHead.scrollLeft = scrollLeft; if ( footer ) { scrollFoot.scrollLeft = scrollLeft; } } ); // When focus is put on the header cells, we might need to scroll the body $('th, td', scrollHead).on('focus', function () { var scrollLeft = scrollHead.scrollLeft; scrollBody.scrollLeft = scrollLeft; if ( footer ) { scrollBody.scrollLeft = scrollLeft; } }); $(scrollBody).css('max-height', scrollY); if (! scroll.bCollapse) { $(scrollBody).css('height', scrollY); } settings.nScrollHead = scrollHead; settings.nScrollBody = scrollBody; settings.nScrollFoot = scrollFoot; // On redraw - align columns settings.aoDrawCallback.push(_fnScrollDraw); return scroller[0]; } /** * Update the header, footer and body tables for resizing - i.e. column * alignment. * * Welcome to the most horrible function DataTables. The process that this * function follows is basically: * 1. Re-create the table inside the scrolling div * 2. Correct colgroup > col values if needed * 3. Copy colgroup > col over to header and footer * 4. Clean up * * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnScrollDraw ( settings ) { // Given that this is such a monster function, a lot of variables are use // to try and keep the minimised size as small as possible var scroll = settings.oScroll, barWidth = scroll.iBarWidth, divHeader = $(settings.nScrollHead), divHeaderInner = divHeader.children('div'), divHeaderTable = divHeaderInner.children('table'), divBodyEl = settings.nScrollBody, divBody = $(divBodyEl), divFooter = $(settings.nScrollFoot), divFooterInner = divFooter.children('div'), divFooterTable = divFooterInner.children('table'), header = $(settings.nTHead), table = $(settings.nTable), footer = settings.nTFoot && $('th, td', settings.nTFoot).length ? $(settings.nTFoot) : null, browser = settings.oBrowser, headerCopy, footerCopy; // If the scrollbar visibility has changed from the last draw, we need to // adjust the column sizes as the table width will have changed to account // for the scrollbar var scrollBarVis = divBodyEl.scrollHeight > divBodyEl.clientHeight; if ( settings.scrollBarVis !== scrollBarVis && settings.scrollBarVis !== undefined ) { settings.scrollBarVis = scrollBarVis; _fnAdjustColumnSizing( settings ); return; // adjust column sizing will call this function again } else { settings.scrollBarVis = scrollBarVis; } // 1. Re-create the table inside the scrolling div // Remove the old minimised thead and tfoot elements in the inner table table.children('thead, tfoot').remove(); // Clone the current header and footer elements and then place it into the inner table headerCopy = header.clone().prependTo( table ); headerCopy.find('th, td').removeAttr('tabindex'); headerCopy.find('[id]').removeAttr('id'); if ( footer ) { footerCopy = footer.clone().prependTo( table ); footerCopy.find('[id]').removeAttr('id'); } // 2. Correct colgroup > col values if needed // It is possible that the cell sizes are smaller than the content, so we need to // correct colgroup>col for such cases. This can happen if the auto width detection // uses a cell which has a longer string, but isn't the widest! For example // "Chief Executive Officer (CEO)" is the longest string in the demo, but // "Systems Administrator" is actually the widest string since it doesn't collapse. // Note the use of translating into a column index to get the `col` element. This // is because of Responsive which might remove `col` elements, knocking the alignment // of the indexes out. if (settings.aiDisplay.length) { // Get the column sizes from the first row in the table var colSizes = table.children('tbody').eq(0).children('tr').eq(0).children('th, td').map(function (vis) { return { idx: _fnVisibleToColumnIndex(settings, vis), width: $(this).outerWidth() } }); // Check against what the colgroup > col is set to and correct if needed for (var i=0 ; i<colSizes.length ; i++) { var colEl = settings.aoColumns[ colSizes[i].idx ].colEl[0]; var colWidth = colEl.style.width.replace('px', ''); if (colWidth !== colSizes[i].width) { colEl.style.width = colSizes[i].width + 'px'; } } } // 3. Copy the colgroup over to the header and footer divHeaderTable .find('colgroup') .remove(); divHeaderTable.append(settings.colgroup.clone()); if ( footer ) { divFooterTable .find('colgroup') .remove(); divFooterTable.append(settings.colgroup.clone()); } // "Hide" the header and footer that we used for the sizing. We need to keep // the content of the cell so that the width applied to the header and body // both match, but we want to hide it completely. $('th, td', headerCopy).each(function () { $(this.childNodes).wrapAll('<div class="dt-scroll-sizing">'); }); if ( footer ) { $('th, td', footerCopy).each(function () { $(this.childNodes).wrapAll('<div class="dt-scroll-sizing">'); }); } // 4. Clean up // Figure out if there are scrollbar present - if so then we need a the header and footer to // provide a bit more space to allow "overflow" scrolling (i.e. past the scrollbar) var isScrolling = Math.floor(table.height()) > divBodyEl.clientHeight || divBody.css('overflow-y') == "scroll"; var paddingSide = 'padding' + (browser.bScrollbarLeft ? 'Left' : 'Right' ); // Set the width's of the header and footer tables var outerWidth = table.outerWidth(); divHeaderTable.css('width', _fnStringToCss( outerWidth )); divHeaderInner .css('width', _fnStringToCss( outerWidth )) .css(paddingSide, isScrolling ? barWidth+"px" : "0px"); if ( footer ) { divFooterTable.css('width', _fnStringToCss( outerWidth )); divFooterInner .css('width', _fnStringToCss( outerWidth )) .css(paddingSide, isScrolling ? barWidth+"px" : "0px"); } // Correct DOM ordering for colgroup - comes before the thead table.children('colgroup').prependTo(table); // Adjust the position of the header in case we loose the y-scrollbar divBody.trigger('scroll'); // If sorting or filtering has occurred, jump the scrolling back to the top // only if we aren't holding the position if ( (settings.bSorted || settings.bFiltered) && ! settings._drawHold ) { divBodyEl.scrollTop = 0; } } /** * Calculate the width of columns for the table * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnCalculateColumnWidths ( settings ) { // Not interested in doing column width calculation if auto-width is disabled if (! settings.oFeatures.bAutoWidth) { return; } var table = settings.nTable, columns = settings.aoColumns, scroll = settings.oScroll, scrollY = scroll.sY, scrollX = scroll.sX, scrollXInner = scroll.sXInner, visibleColumns = _fnGetColumns( settings, 'bVisible' ), tableWidthAttr = table.getAttribute('width'), // from DOM element tableContainer = table.parentNode, i, column, columnIdx; var styleWidth = table.style.width; if ( styleWidth && styleWidth.indexOf('%') !== -1 ) { tableWidthAttr = styleWidth; } // Let plug-ins know that we are doing a recalc, in case they have changed any of the // visible columns their own way (e.g. Responsive uses display:none). _fnCallbackFire( settings, null, 'column-calc', {visible: visibleColumns}, false ); // Construct a single row, worst case, table with the widest // node in the data, assign any user defined widths, then insert it into // the DOM and allow the browser to do all the hard work of calculating // table widths var tmpTable = $(table.cloneNode()) .css( 'visibility', 'hidden' ) .removeAttr( 'id' ); // Clean up the table body tmpTable.append('<tbody>') var tr = $('<tr/>').appendTo( tmpTable.find('tbody') ); // Clone the table header and footer - we can't use the header / footer // from the cloned table, since if scrolling is active, the table's // real header and footer are contained in different table tags tmpTable .append( $(settings.nTHead).clone() ) .append( $(settings.nTFoot).clone() ); // Remove any assigned widths from the footer (from scrolling) tmpTable.find('tfoot th, tfoot td').css('width', ''); // Apply custom sizing to the cloned header tmpTable.find('thead th, thead td').each( function () { // Get the `width` from the header layout var width = _fnColumnsSumWidth( settings, this, true, false ); if ( width ) { this.style.width = width; // For scrollX we need to force the column width otherwise the // browser will collapse it. If this width is smaller than the // width the column requires, then it will have no effect if ( scrollX ) { $( this ).append( $('<div/>').css( { width: width, margin: 0, padding: 0, border: 0, height: 1 } ) ); } } else { this.style.width = ''; } } ); // Find the widest piece of data for each column and put it into the table for ( i=0 ; i<visibleColumns.length ; i++ ) { columnIdx = visibleColumns[i]; column = columns[ columnIdx ]; var longest = _fnGetMaxLenString(settings, columnIdx); var autoClass = _ext.type.className[column.sType]; var text = longest + column.sContentPadding; var insert = longest.indexOf('<') === -1 ? document.createTextNode(text) : text $('<td/>') .addClass(autoClass) .addClass(column.sClass) .append(insert) .appendTo(tr); } // Tidy the temporary table - remove name attributes so there aren't // duplicated in the dom (radio elements for example) $('[name]', tmpTable).removeAttr('name'); // Table has been built, attach to the document so we can work with it. // A holding element is used, positioned at the top of the container // with minimal height, so it has no effect on if the container scrolls // or not. Otherwise it might trigger scrolling when it actually isn't // needed var holder = $('<div/>').css( scrollX || scrollY ? { position: 'absolute', top: 0, left: 0, height: 1, right: 0, overflow: 'hidden' } : {} ) .append( tmpTable ) .appendTo( tableContainer ); // When scrolling (X or Y) we want to set the width of the table as // appropriate. However, when not scrolling leave the table width as it // is. This results in slightly different, but I think correct behaviour if ( scrollX && scrollXInner ) { tmpTable.width( scrollXInner ); } else if ( scrollX ) { tmpTable.css( 'width', 'auto' ); tmpTable.removeAttr('width'); // If there is no width attribute or style, then allow the table to // collapse if ( tmpTable.width() < tableContainer.clientWidth && tableWidthAttr ) { tmpTable.width( tableContainer.clientWidth ); } } else if ( scrollY ) { tmpTable.width( tableContainer.clientWidth ); } else if ( tableWidthAttr ) { tmpTable.width( tableWidthAttr ); } // Get the width of each column in the constructed table var total = 0; var bodyCells = tmpTable.find('tbody tr').eq(0).children(); for ( i=0 ; i<visibleColumns.length ; i++ ) { // Use getBounding for sub-pixel accuracy, which we then want to round up! var bounding = bodyCells[i].getBoundingClientRect().width; // Total is tracked to remove any sub-pixel errors as the outerWidth // of the table might not equal the total given here total += bounding; // Width for each column to use columns[ visibleColumns[i] ].sWidth = _fnStringToCss( bounding ); } table.style.width = _fnStringToCss( total ); // Finished with the table - ditch it holder.remove(); // If there is a width attr, we want to attach an event listener which // allows the table sizing to automatically adjust when the window is // resized. Use the width attr rather than CSS, since we can't know if the // CSS is a relative value or absolute - DOM read is always px. if ( tableWidthAttr ) { table.style.width = _fnStringToCss( tableWidthAttr ); } if ( (tableWidthAttr || scrollX) && ! settings._reszEvt ) { var bindResize = function () { $(window).on('resize.DT-'+settings.sInstance, DataTable.util.throttle( function () { if (! settings.bDestroying) { _fnAdjustColumnSizing( settings ); } } ) ); }; bindResize(); settings._reszEvt = true; } } /** * Get the maximum strlen for each data column * @param {object} settings dataTables settings object * @param {int} colIdx column of interest * @returns {string} string of the max length * @memberof DataTable#oApi */ function _fnGetMaxLenString( settings, colIdx ) { var column = settings.aoColumns[colIdx]; if (! column.maxLenString) { var s, max='', maxLen = -1; for ( var i=0, ien=settings.aiDisplayMaster.length ; i<ien ; i++ ) { var rowIdx = settings.aiDisplayMaster[i]; var data = _fnGetRowDisplay(settings, rowIdx)[colIdx]; var cellString = data && typeof data === 'object' && data.nodeType ? data.innerHTML : data+''; // Remove id / name attributes from elements so they // don't interfere with existing elements cellString = cellString .replace(/id=".*?"/g, '') .replace(/name=".*?"/g, ''); s = _stripHtml(cellString) .replace( / /g, ' ' ); if ( s.length > maxLen ) { // We want the HTML in the string, but the length that // is important is the stripped string max = cellString; maxLen = s.length; } } column.maxLenString = max; } return column.maxLenString; } /** * Append a CSS unit (only if required) to a string * @param {string} value to css-ify * @returns {string} value with css unit * @memberof DataTable#oApi */ function _fnStringToCss( s ) { if ( s === null ) { return '0px'; } if ( typeof s == 'number' ) { return s < 0 ? '0px' : s+'px'; } // Check it has a unit character already return s.match(/\d$/) ? s+'px' : s; } /** * Re-insert the `col` elements for current visibility * * @param {*} settings DT settings */ function _colGroup( settings ) { var cols = settings.aoColumns; settings.colgroup.empty(); for (i=0 ; i<cols.length ; i++) { if (cols[i].bVisible) { settings.colgroup.append(cols[i].colEl); } } } function _fnSortInit( settings ) { var target = settings.nTHead; var headerRows = target.querySelectorAll('tr'); var legacyTop = settings.bSortCellsTop; var notSelector = ':not([data-dt-order="disable"]):not([data-dt-order="icon-only"])'; // Legacy support for `orderCellsTop` if (legacyTop === true) { target = headerRows[0]; } else if (legacyTop === false) { target = headerRows[ headerRows.length - 1 ]; } _fnSortAttachListener( settings, target, target === settings.nTHead ? 'tr'+notSelector+' th'+notSelector+', tr'+notSelector+' td'+notSelector : 'th'+notSelector+', td'+notSelector ); // Need to resolve the user input array into our internal structure var order = []; _fnSortResolve( settings, order, settings.aaSorting ); settings.aaSorting = order; } function _fnSortAttachListener(settings, node, selector, column, callback) { _fnBindAction( node, selector, function (e) { var run = false; var columns = column === undefined ? _fnColumnsFromHeader( e.target ) : [column]; if ( columns.length ) { for ( var i=0, ien=columns.length ; i<ien ; i++ ) { var ret = _fnSortAdd( settings, columns[i], i, e.shiftKey ); if (ret !== false) { run = true; } // If the first entry is no sort, then subsequent // sort columns are ignored if (settings.aaSorting.length === 1 && settings.aaSorting[0][1] === '') { break; } } if (run) { _fnProcessingDisplay( settings, true ); // Allow the processing display to show setTimeout( function () { _fnSort( settings ); _fnSortDisplay( settings, settings.aiDisplay ); // Sort processing done - redraw has its own processing display _fnProcessingDisplay( settings, false ); _fnReDraw( settings, false, false ); if (callback) { callback(); } }, 0); } } } ); } /** * Sort the display array to match the master's order * @param {*} settings */ function _fnSortDisplay(settings, display) { if (display.length < 2) { return; } var master = settings.aiDisplayMaster; var masterMap = {}; var map = {}; var i; // Rather than needing an `indexOf` on master array, we can create a map for (i=0 ; i<master.length ; i++) { masterMap[master[i]] = i; } // And then cache what would be the indexOf fom the display for (i=0 ; i<display.length ; i++) { map[display[i]] = masterMap[display[i]]; } display.sort(function(a, b){ // Short version of this function is simply `master.indexOf(a) - master.indexOf(b);` return map[a] - map[b]; }); } function _fnSortResolve (settings, nestedSort, sort) { var push = function ( a ) { if ($.isPlainObject(a)) { if (a.idx !== undefined) { // Index based ordering nestedSort.push([a.idx, a.dir]); } else if (a.name) { // Name based ordering var cols = _pluck( settings.aoColumns, 'sName'); var idx = cols.indexOf(a.name); if (idx !== -1) { nestedSort.push([idx, a.dir]); } } } else { // Plain column index and direction pair nestedSort.push(a); } }; if ( $.isPlainObject(sort) ) { // Object push(sort); } else if ( sort.length && typeof sort[0] === 'number' ) { // 1D array push(sort); } else if ( sort.length ) { // 2D array for (var z=0; z<sort.length; z++) { push(sort[z]); // Object or array } } } function _fnSortFlatten ( settings ) { var i, k, kLen, aSort = [], extSort = DataTable.ext.type.order, aoColumns = settings.aoColumns, aDataSort, iCol, sType, srcCol, fixed = settings.aaSortingFixed, fixedObj = $.isPlainObject( fixed ), nestedSort = []; if ( ! settings.oFeatures.bSort ) { return aSort; } // Build the sort array, with pre-fix and post-fix options if they have been // specified if ( Array.isArray( fixed ) ) { _fnSortResolve( settings, nestedSort, fixed ); } if ( fixedObj && fixed.pre ) { _fnSortResolve( settings, nestedSort, fixed.pre ); } _fnSortResolve( settings, nestedSort, settings.aaSorting ); if (fixedObj && fixed.post ) { _fnSortResolve( settings, nestedSort, fixed.post ); } for ( i=0 ; i<nestedSort.length ; i++ ) { srcCol = nestedSort[i][0]; if ( aoColumns[ srcCol ] ) { aDataSort = aoColumns[ srcCol ].aDataSort; for ( k=0, kLen=aDataSort.length ; k<kLen ; k++ ) { iCol = aDataSort[k]; sType = aoColumns[ iCol ].sType || 'string'; if ( nestedSort[i]._idx === undefined ) { nestedSort[i]._idx = aoColumns[iCol].asSorting.indexOf(nestedSort[i][1]); } if ( nestedSort[i][1] ) { aSort.push( { src: srcCol, col: iCol, dir: nestedSort[i][1], index: nestedSort[i]._idx, type: sType, formatter: extSort[ sType+"-pre" ], sorter: extSort[ sType+"-"+nestedSort[i][1] ] } ); } } } } return aSort; } /** * Change the order of the table * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnSort ( oSettings, col, dir ) { var i, ien, iLen, aiOrig = [], extSort = DataTable.ext.type.order, aoData = oSettings.aoData, sortCol, displayMaster = oSettings.aiDisplayMaster, aSort; // Resolve any column types that are unknown due to addition or invalidation // @todo Can this be moved into a 'data-ready' handler which is called when // data is going to be used in the table? _fnColumnTypes( oSettings ); // Allow a specific column to be sorted, which will _not_ alter the display // master if (col !== undefined) { var srcCol = oSettings.aoColumns[col]; aSort = [{ src: col, col: col, dir: dir, index: 0, type: srcCol.sType, formatter: extSort[ srcCol.sType+"-pre" ], sorter: extSort[ srcCol.sType+"-"+dir ] }]; displayMaster = displayMaster.slice(); } else { aSort = _fnSortFlatten( oSettings ); } for ( i=0, ien=aSort.length ; i<ien ; i++ ) { sortCol = aSort[i]; // Load the data needed for the sort, for each cell _fnSortData( oSettings, sortCol.col ); } /* No sorting required if server-side or no sorting array */ if ( _fnDataSource( oSettings ) != 'ssp' && aSort.length !== 0 ) { // Reset the initial positions on each pass so we get a stable sort for ( i=0, iLen=displayMaster.length ; i<iLen ; i++ ) { aiOrig[ i ] = i; } // If the first sort is desc, then reverse the array to preserve original // order, just in reverse if (aSort.length && aSort[0].dir === 'desc') { aiOrig.reverse(); } /* Do the sort - here we want multi-column sorting based on a given data source (column) * and sorting function (from oSort) in a certain direction. It's reasonably complex to * follow on it's own, but this is what we want (example two column sorting): * fnLocalSorting = function(a,b){ * var test; * test = oSort['string-asc']('data11', 'data12'); * if (test !== 0) * return test; * test = oSort['numeric-desc']('data21', 'data22'); * if (test !== 0) * return test; * return oSort['numeric-asc']( aiOrig[a], aiOrig[b] ); * } * Basically we have a test for each sorting column, if the data in that column is equal, * test the next column. If all columns match, then we use a numeric sort on the row * positions in the original data array to provide a stable sort. */ displayMaster.sort( function ( a, b ) { var x, y, k, test, sort, len=aSort.length, dataA = aoData[a]._aSortData, dataB = aoData[b]._aSortData; for ( k=0 ; k<len ; k++ ) { sort = aSort[k]; // Data, which may have already been through a `-pre` function x = dataA[ sort.col ]; y = dataB[ sort.col ]; if (sort.sorter) { // If there is a custom sorter (`-asc` or `-desc`) for this // data type, use it test = sort.sorter(x, y); if ( test !== 0 ) { return test; } } else { // Otherwise, use generic sorting test = x<y ? -1 : x>y ? 1 : 0; if ( test !== 0 ) { return sort.dir === 'asc' ? test : -test; } } } x = aiOrig[a]; y = aiOrig[b]; return x<y ? -1 : x>y ? 1 : 0; } ); } else if ( aSort.length === 0 ) { // Apply index order displayMaster.sort(function (x, y) { return x<y ? -1 : x>y ? 1 : 0; }); } if (col === undefined) { // Tell the draw function that we have sorted the data oSettings.bSorted = true; _fnCallbackFire( oSettings, null, 'order', [oSettings, aSort] ); } return displayMaster; } /** * Function to run on user sort request * @param {object} settings dataTables settings object * @param {node} attachTo node to attach the handler to * @param {int} colIdx column sorting index * @param {int} addIndex Counter * @param {boolean} [shift=false] Shift click add * @param {function} [callback] callback function * @memberof DataTable#oApi */ function _fnSortAdd ( settings, colIdx, addIndex, shift ) { var col = settings.aoColumns[ colIdx ]; var sorting = settings.aaSorting; var asSorting = col.asSorting; var nextSortIdx; var next = function ( a, overflow ) { var idx = a._idx; if ( idx === undefined ) { idx = asSorting.indexOf(a[1]); } return idx+1 < asSorting.length ? idx+1 : overflow ? null : 0; }; if ( ! col.bSortable ) { return false; } // Convert to 2D array if needed if ( typeof sorting[0] === 'number' ) { sorting = settings.aaSorting = [ sorting ]; } // If appending the sort then we are multi-column sorting if ( (shift || addIndex) && settings.oFeatures.bSortMulti ) { // Are we already doing some kind of sort on this column? var sortIdx = _pluck(sorting, '0').indexOf(colIdx); if ( sortIdx !== -1 ) { // Yes, modify the sort nextSortIdx = next( sorting[sortIdx], true ); if ( nextSortIdx === null && sorting.length === 1 ) { nextSortIdx = 0; // can't remove sorting completely } if ( nextSortIdx === null ) { sorting.splice( sortIdx, 1 ); } else { sorting[sortIdx][1] = asSorting[ nextSortIdx ]; sorting[sortIdx]._idx = nextSortIdx; } } else if (shift) { // No sort on this column yet, being added by shift click // add it as itself sorting.push( [ colIdx, asSorting[0], 0 ] ); sorting[sorting.length-1]._idx = 0; } else { // No sort on this column yet, being added from a colspan // so add with same direction as first column sorting.push( [ colIdx, sorting[0][1], 0 ] ); sorting[sorting.length-1]._idx = 0; } } else if ( sorting.length && sorting[0][0] == colIdx ) { // Single column - already sorting on this column, modify the sort nextSortIdx = next( sorting[0] ); sorting.length = 1; sorting[0][1] = asSorting[ nextSortIdx ]; sorting[0]._idx = nextSortIdx; } else { // Single column - sort only on this column sorting.length = 0; sorting.push( [ colIdx, asSorting[0] ] ); sorting[0]._idx = 0; } } /** * Set the sorting classes on table's body, Note: it is safe to call this function * when bSort and bSortClasses are false * @param {object} oSettings dataTables settings object * @memberof DataTable#oApi */ function _fnSortingClasses( settings ) { var oldSort = settings.aLastSort; var sortClass = settings.oClasses.order.position; var sort = _fnSortFlatten( settings ); var features = settings.oFeatures; var i, ien, colIdx; if ( features.bSort && features.bSortClasses ) { // Remove old sorting classes for ( i=0, ien=oldSort.length ; i<ien ; i++ ) { colIdx = oldSort[i].src; // Remove column sorting $( _pluck( settings.aoData, 'anCells', colIdx ) ) .removeClass( sortClass + (i<2 ? i+1 : 3) ); } // Add new column sorting for ( i=0, ien=sort.length ; i<ien ; i++ ) { colIdx = sort[i].src; $( _pluck( settings.aoData, 'anCells', colIdx ) ) .addClass( sortClass + (i<2 ? i+1 : 3) ); } } settings.aLastSort = sort; } // Get the data to sort a column, be it from cache, fresh (populating the // cache), or from a sort formatter function _fnSortData( settings, colIdx ) { // Custom sorting function - provided by the sort data type var column = settings.aoColumns[ colIdx ]; var customSort = DataTable.ext.order[ column.sSortDataType ]; var customData; if ( customSort ) { customData = customSort.call( settings.oInstance, settings, colIdx, _fnColumnIndexToVisible( settings, colIdx ) ); } // Use / populate cache var row, cellData; var formatter = DataTable.ext.type.order[ column.sType+"-pre" ]; var data = settings.aoData; for ( var rowIdx=0 ; rowIdx<data.length ; rowIdx++ ) { // Sparse array if (! data[rowIdx]) { continue; } row = data[rowIdx]; if ( ! row._aSortData ) { row._aSortData = []; } if ( ! row._aSortData[colIdx] || customSort ) { cellData = customSort ? customData[rowIdx] : // If there was a custom sort function, use data from there _fnGetCellData( settings, rowIdx, colIdx, 'sort' ); row._aSortData[ colIdx ] = formatter ? formatter( cellData, settings ) : cellData; } } } /** * State information for a table * * @param {*} settings * @returns State object */ function _fnSaveState ( settings ) { if (settings._bLoadingState) { return; } /* Store the interesting variables */ var state = { time: +new Date(), start: settings._iDisplayStart, length: settings._iDisplayLength, order: $.extend( true, [], settings.aaSorting ), search: $.extend({}, settings.oPreviousSearch), columns: settings.aoColumns.map( function ( col, i ) { return { visible: col.bVisible, search: $.extend({}, settings.aoPreSearchCols[i]) }; } ) }; settings.oSavedState = state; _fnCallbackFire( settings, "aoStateSaveParams", 'stateSaveParams', [settings, state] ); if ( settings.oFeatures.bStateSave && !settings.bDestroying ) { settings.fnStateSaveCallback.call( settings.oInstance, settings, state ); } } /** * Attempt to load a saved table state * @param {object} oSettings dataTables settings object * @param {object} oInit DataTables init object so we can override settings * @param {function} callback Callback to execute when the state has been loaded * @memberof DataTable#oApi */ function _fnLoadState ( settings, init, callback ) { if ( ! settings.oFeatures.bStateSave ) { callback(); return; } var loaded = function(state) { _fnImplementState(settings, state, callback); } var state = settings.fnStateLoadCallback.call( settings.oInstance, settings, loaded ); if ( state !== undefined ) { _fnImplementState( settings, state, callback ); } // otherwise, wait for the loaded callback to be executed return true; } function _fnImplementState ( settings, s, callback) { var i, ien; var columns = settings.aoColumns; settings._bLoadingState = true; // When StateRestore was introduced the state could now be implemented at any time // Not just initialisation. To do this an api instance is required in some places var api = settings._bInitComplete ? new DataTable.Api(settings) : null; if ( ! s || ! s.time ) { settings._bLoadingState = false; callback(); return; } // Reject old data var duration = settings.iStateDuration; if ( duration > 0 && s.time < +new Date() - (duration*1000) ) { settings._bLoadingState = false; callback(); return; } // Allow custom and plug-in manipulation functions to alter the saved data set and // cancelling of loading by returning false var abStateLoad = _fnCallbackFire( settings, 'aoStateLoadParams', 'stateLoadParams', [settings, s] ); if ( abStateLoad.indexOf(false) !== -1 ) { settings._bLoadingState = false; callback(); return; } // Number of columns have changed - all bets are off, no restore of settings if ( s.columns && columns.length !== s.columns.length ) { settings._bLoadingState = false; callback(); return; } // Store the saved state so it might be accessed at any time settings.oLoadedState = $.extend( true, {}, s ); // This is needed for ColReorder, which has to happen first to allow all // the stored indexes to be usable. It is not publicly documented. _fnCallbackFire( settings, null, 'stateLoadInit', [settings, s], true ); // Page Length if ( s.length !== undefined ) { // If already initialised just set the value directly so that the select element is also updated if (api) { api.page.len(s.length) } else { settings._iDisplayLength = s.length; } } // Restore key features - todo - for 1.11 this needs to be done by // subscribed events if ( s.start !== undefined ) { if(api === null) { settings._iDisplayStart = s.start; settings.iInitDisplayStart = s.start; } else { _fnPageChange(settings, s.start/settings._iDisplayLength); } } // Order if ( s.order !== undefined ) { settings.aaSorting = []; $.each( s.order, function ( i, col ) { settings.aaSorting.push( col[0] >= columns.length ? [ 0, col[1] ] : col ); } ); } // Search if ( s.search !== undefined ) { $.extend( settings.oPreviousSearch, s.search ); } // Columns if ( s.columns ) { for ( i=0, ien=s.columns.length ; i<ien ; i++ ) { var col = s.columns[i]; // Visibility if ( col.visible !== undefined ) { // If the api is defined, the table has been initialised so we need to use it rather than internal settings if (api) { // Don't redraw the columns on every iteration of this loop, we will do this at the end instead api.column(i).visible(col.visible, false); } else { columns[i].bVisible = col.visible; } } // Search if ( col.search !== undefined ) { $.extend( settings.aoPreSearchCols[i], col.search ); } } // If the api is defined then we need to adjust the columns once the visibility has been changed if (api) { api.columns.adjust(); } } settings._bLoadingState = false; _fnCallbackFire( settings, 'aoStateLoaded', 'stateLoaded', [settings, s] ); callback(); } /** * Log an error message * @param {object} settings dataTables settings object * @param {int} level log error messages, or display them to the user * @param {string} msg error message * @param {int} tn Technical note id to get more information about the error. * @memberof DataTable#oApi */ function _fnLog( settings, level, msg, tn ) { msg = 'DataTables warning: '+ (settings ? 'table id='+settings.sTableId+' - ' : '')+msg; if ( tn ) { msg += '. For more information about this error, please see '+ 'https://datatables.net/tn/'+tn; } if ( ! level ) { // Backwards compatibility pre 1.10 var ext = DataTable.ext; var type = ext.sErrMode || ext.errMode; if ( settings ) { _fnCallbackFire( settings, null, 'dt-error', [ settings, tn, msg ], true ); } if ( type == 'alert' ) { alert( msg ); } else if ( type == 'throw' ) { throw new Error(msg); } else if ( typeof type == 'function' ) { type( settings, tn, msg ); } } else if ( window.console && console.log ) { console.log( msg ); } } /** * See if a property is defined on one object, if so assign it to the other object * @param {object} ret target object * @param {object} src source object * @param {string} name property * @param {string} [mappedName] name to map too - optional, name used if not given * @memberof DataTable#oApi */ function _fnMap( ret, src, name, mappedName ) { if ( Array.isArray( name ) ) { $.each( name, function (i, val) { if ( Array.isArray( val ) ) { _fnMap( ret, src, val[0], val[1] ); } else { _fnMap( ret, src, val ); } } ); return; } if ( mappedName === undefined ) { mappedName = name; } if ( src[name] !== undefined ) { ret[mappedName] = src[name]; } } /** * Extend objects - very similar to jQuery.extend, but deep copy objects, and * shallow copy arrays. The reason we need to do this, is that we don't want to * deep copy array init values (such as aaSorting) since the dev wouldn't be * able to override them, but we do want to deep copy arrays. * @param {object} out Object to extend * @param {object} extender Object from which the properties will be applied to * out * @param {boolean} breakRefs If true, then arrays will be sliced to take an * independent copy with the exception of the `data` or `aaData` parameters * if they are present. This is so you can pass in a collection to * DataTables and have that used as your data source without breaking the * references * @returns {object} out Reference, just for convenience - out === the return. * @memberof DataTable#oApi * @todo This doesn't take account of arrays inside the deep copied objects. */ function _fnExtend( out, extender, breakRefs ) { var val; for ( var prop in extender ) { if ( Object.prototype.hasOwnProperty.call(extender, prop) ) { val = extender[prop]; if ( $.isPlainObject( val ) ) { if ( ! $.isPlainObject( out[prop] ) ) { out[prop] = {}; } $.extend( true, out[prop], val ); } else if ( breakRefs && prop !== 'data' && prop !== 'aaData' && Array.isArray(val) ) { out[prop] = val.slice(); } else { out[prop] = val; } } } return out; } /** * Bind an event handers to allow a click or return key to activate the callback. * This is good for accessibility since a return on the keyboard will have the * same effect as a click, if the element has focus. * @param {element} n Element to bind the action to * @param {object|string} selector Selector (for delegated events) or data object * to pass to the triggered function * @param {function} fn Callback function for when the event is triggered * @memberof DataTable#oApi */ function _fnBindAction( n, selector, fn ) { $(n) .on( 'click.DT', selector, function (e) { fn(e); } ) .on( 'keypress.DT', selector, function (e){ if ( e.which === 13 ) { e.preventDefault(); fn(e); } } ) .on( 'selectstart.DT', selector, function () { // Don't want a double click resulting in text selection return false; } ); } /** * Register a callback function. Easily allows a callback function to be added to * an array store of callback functions that can then all be called together. * @param {object} settings dataTables settings object * @param {string} store Name of the array storage for the callbacks in oSettings * @param {function} fn Function to be called back * @memberof DataTable#oApi */ function _fnCallbackReg( settings, store, fn ) { if ( fn ) { settings[store].push(fn); } } /** * Fire callback functions and trigger events. Note that the loop over the * callback array store is done backwards! Further note that you do not want to * fire off triggers in time sensitive applications (for example cell creation) * as its slow. * @param {object} settings dataTables settings object * @param {string} callbackArr Name of the array storage for the callbacks in * oSettings * @param {string} eventName Name of the jQuery custom event to trigger. If * null no trigger is fired * @param {array} args Array of arguments to pass to the callback function / * trigger * @param {boolean} [bubbles] True if the event should bubble * @memberof DataTable#oApi */ function _fnCallbackFire( settings, callbackArr, eventName, args, bubbles ) { var ret = []; if ( callbackArr ) { ret = settings[callbackArr].slice().reverse().map( function (val) { return val.apply( settings.oInstance, args ); } ); } if ( eventName !== null) { var e = $.Event( eventName+'.dt' ); var table = $(settings.nTable); // Expose the DataTables API on the event object for easy access e.dt = settings.api; table[bubbles ? 'trigger' : 'triggerHandler']( e, args ); // If not yet attached to the document, trigger the event // on the body directly to sort of simulate the bubble if (bubbles && table.parents('body').length === 0) { $('body').trigger( e, args ); } ret.push( e.result ); } return ret; } function _fnLengthOverflow ( settings ) { var start = settings._iDisplayStart, end = settings.fnDisplayEnd(), len = settings._iDisplayLength; /* If we have space to show extra rows (backing up from the end point - then do so */ if ( start >= end ) { start = end - len; } // Keep the start record on the current page start -= (start % len); if ( len === -1 || start < 0 ) { start = 0; } settings._iDisplayStart = start; } function _fnRenderer( settings, type ) { var renderer = settings.renderer; var host = DataTable.ext.renderer[type]; if ( $.isPlainObject( renderer ) && renderer[type] ) { // Specific renderer for this type. If available use it, otherwise use // the default. return host[renderer[type]] || host._; } else if ( typeof renderer === 'string' ) { // Common renderer - if there is one available for this type use it, // otherwise use the default return host[renderer] || host._; } // Use the default return host._; } /** * Detect the data source being used for the table. Used to simplify the code * a little (ajax) and to make it compress a little smaller. * * @param {object} settings dataTables settings object * @returns {string} Data source * @memberof DataTable#oApi */ function _fnDataSource ( settings ) { if ( settings.oFeatures.bServerSide ) { return 'ssp'; } else if ( settings.ajax ) { return 'ajax'; } return 'dom'; } /** * Common replacement for language strings * * @param {*} settings DT settings object * @param {*} str String with values to replace * @param {*} entries Plural number for _ENTRIES_ - can be undefined * @returns String */ function _fnMacros ( settings, str, entries ) { // When infinite scrolling, we are always starting at 1. _iDisplayStart is // used only internally var formatter = settings.fnFormatNumber, start = settings._iDisplayStart+1, len = settings._iDisplayLength, vis = settings.fnRecordsDisplay(), max = settings.fnRecordsTotal(), all = len === -1; return str. replace(/_START_/g, formatter.call( settings, start ) ). replace(/_END_/g, formatter.call( settings, settings.fnDisplayEnd() ) ). replace(/_MAX_/g, formatter.call( settings, max ) ). replace(/_TOTAL_/g, formatter.call( settings, vis ) ). replace(/_PAGE_/g, formatter.call( settings, all ? 1 : Math.ceil( start / len ) ) ). replace(/_PAGES_/g, formatter.call( settings, all ? 1 : Math.ceil( vis / len ) ) ). replace(/_ENTRIES_/g, settings.api.i18n('entries', '', entries) ). replace(/_ENTRIES-MAX_/g, settings.api.i18n('entries', '', max) ). replace(/_ENTRIES-TOTAL_/g, settings.api.i18n('entries', '', vis) ); } /** * Computed structure of the DataTables API, defined by the options passed to * `DataTable.Api.register()` when building the API. * * The structure is built in order to speed creation and extension of the Api * objects since the extensions are effectively pre-parsed. * * The array is an array of objects with the following structure, where this * base array represents the Api prototype base: * * [ * { * name: 'data' -- string - Property name * val: function () {}, -- function - Api method (or undefined if just an object * methodExt: [ ... ], -- array - Array of Api object definitions to extend the method result * propExt: [ ... ] -- array - Array of Api object definitions to extend the property * }, * { * name: 'row' * val: {}, * methodExt: [ ... ], * propExt: [ * { * name: 'data' * val: function () {}, * methodExt: [ ... ], * propExt: [ ... ] * }, * ... * ] * } * ] * * @type {Array} * @ignore */ var __apiStruct = []; /** * `Array.prototype` reference. * * @type object * @ignore */ var __arrayProto = Array.prototype; /** * Abstraction for `context` parameter of the `Api` constructor to allow it to * take several different forms for ease of use. * * Each of the input parameter types will be converted to a DataTables settings * object where possible. * * @param {string|node|jQuery|object} mixed DataTable identifier. Can be one * of: * * * `string` - jQuery selector. Any DataTables' matching the given selector * with be found and used. * * `node` - `TABLE` node which has already been formed into a DataTable. * * `jQuery` - A jQuery object of `TABLE` nodes. * * `object` - DataTables settings object * * `DataTables.Api` - API instance * @return {array|null} Matching DataTables settings objects. `null` or * `undefined` is returned if no matching DataTable is found. * @ignore */ var _toSettings = function ( mixed ) { var idx, jq; var settings = DataTable.settings; var tables = _pluck(settings, 'nTable'); if ( ! mixed ) { return []; } else if ( mixed.nTable && mixed.oFeatures ) { // DataTables settings object return [ mixed ]; } else if ( mixed.nodeName && mixed.nodeName.toLowerCase() === 'table' ) { // Table node idx = tables.indexOf(mixed); return idx !== -1 ? [ settings[idx] ] : null; } else if ( mixed && typeof mixed.settings === 'function' ) { return mixed.settings().toArray(); } else if ( typeof mixed === 'string' ) { // jQuery selector jq = $(mixed).get(); } else if ( mixed instanceof $ ) { // jQuery object (also DataTables instance) jq = mixed.get(); } if ( jq ) { return settings.filter(function (v, idx) { return jq.includes(tables[idx]); }); } }; /** * DataTables API class - used to control and interface with one or more * DataTables enhanced tables. * * The API class is heavily based on jQuery, presenting a chainable interface * that you can use to interact with tables. Each instance of the API class has * a "context" - i.e. the tables that it will operate on. This could be a single * table, all tables on a page or a sub-set thereof. * * Additionally the API is designed to allow you to easily work with the data in * the tables, retrieving and manipulating it as required. This is done by * presenting the API class as an array like interface. The contents of the * array depend upon the actions requested by each method (for example * `rows().nodes()` will return an array of nodes, while `rows().data()` will * return an array of objects or arrays depending upon your table's * configuration). The API object has a number of array like methods (`push`, * `pop`, `reverse` etc) as well as additional helper methods (`each`, `pluck`, * `unique` etc) to assist your working with the data held in a table. * * Most methods (those which return an Api instance) are chainable, which means * the return from a method call also has all of the methods available that the * top level object had. For example, these two calls are equivalent: * * // Not chained * api.row.add( {...} ); * api.draw(); * * // Chained * api.row.add( {...} ).draw(); * * @class DataTable.Api * @param {array|object|string|jQuery} context DataTable identifier. This is * used to define which DataTables enhanced tables this API will operate on. * Can be one of: * * * `string` - jQuery selector. Any DataTables' matching the given selector * with be found and used. * * `node` - `TABLE` node which has already been formed into a DataTable. * * `jQuery` - A jQuery object of `TABLE` nodes. * * `object` - DataTables settings object * @param {array} [data] Data to initialise the Api instance with. * * @example * // Direct initialisation during DataTables construction * var api = $('#example').DataTable(); * * @example * // Initialisation using a DataTables jQuery object * var api = $('#example').dataTable().api(); * * @example * // Initialisation as a constructor * var api = new DataTable.Api( 'table.dataTable' ); */ _Api = function ( context, data ) { if ( ! (this instanceof _Api) ) { return new _Api( context, data ); } var settings = []; var ctxSettings = function ( o ) { var a = _toSettings( o ); if ( a ) { settings.push.apply( settings, a ); } }; if ( Array.isArray( context ) ) { for ( var i=0, ien=context.length ; i<ien ; i++ ) { ctxSettings( context[i] ); } } else { ctxSettings( context ); } // Remove duplicates this.context = settings.length > 1 ? _unique( settings ) : settings; // Initial data if ( data ) { this.push.apply(this, data); } // selector this.selector = { rows: null, cols: null, opts: null }; _Api.extend( this, this, __apiStruct ); }; DataTable.Api = _Api; // Don't destroy the existing prototype, just extend it. Required for jQuery 2's // isPlainObject. $.extend( _Api.prototype, { any: function () { return this.count() !== 0; }, context: [], // array of table settings objects count: function () { return this.flatten().length; }, each: function ( fn ) { for ( var i=0, ien=this.length ; i<ien; i++ ) { fn.call( this, this[i], i, this ); } return this; }, eq: function ( idx ) { var ctx = this.context; return ctx.length > idx ? new _Api( ctx[idx], this[idx] ) : null; }, filter: function ( fn ) { var a = __arrayProto.filter.call( this, fn, this ); return new _Api( this.context, a ); }, flatten: function () { var a = []; return new _Api( this.context, a.concat.apply( a, this.toArray() ) ); }, get: function ( idx ) { return this[ idx ]; }, join: __arrayProto.join, includes: function ( find ) { return this.indexOf( find ) === -1 ? false : true; }, indexOf: __arrayProto.indexOf, iterator: function ( flatten, type, fn, alwaysNew ) { var a = [], ret, i, ien, j, jen, context = this.context, rows, items, item, selector = this.selector; // Argument shifting if ( typeof flatten === 'string' ) { alwaysNew = fn; fn = type; type = flatten; flatten = false; } for ( i=0, ien=context.length ; i<ien ; i++ ) { var apiInst = new _Api( context[i] ); if ( type === 'table' ) { ret = fn.call( apiInst, context[i], i ); if ( ret !== undefined ) { a.push( ret ); } } else if ( type === 'columns' || type === 'rows' ) { // this has same length as context - one entry for each table ret = fn.call( apiInst, context[i], this[i], i ); if ( ret !== undefined ) { a.push( ret ); } } else if ( type === 'every' || type === 'column' || type === 'column-rows' || type === 'row' || type === 'cell' ) { // columns and rows share the same structure. // 'this' is an array of column indexes for each context items = this[i]; if ( type === 'column-rows' ) { rows = _selector_row_indexes( context[i], selector.opts ); } for ( j=0, jen=items.length ; j<jen ; j++ ) { item = items[j]; if ( type === 'cell' ) { ret = fn.call( apiInst, context[i], item.row, item.column, i, j ); } else { ret = fn.call( apiInst, context[i], item, i, j, rows ); } if ( ret !== undefined ) { a.push( ret ); } } } } if ( a.length || alwaysNew ) { var api = new _Api( context, flatten ? a.concat.apply( [], a ) : a ); var apiSelector = api.selector; apiSelector.rows = selector.rows; apiSelector.cols = selector.cols; apiSelector.opts = selector.opts; return api; } return this; }, lastIndexOf: __arrayProto.lastIndexOf, length: 0, map: function ( fn ) { var a = __arrayProto.map.call( this, fn, this ); return new _Api( this.context, a ); }, pluck: function ( prop ) { var fn = DataTable.util.get(prop); return this.map( function ( el ) { return fn(el); } ); }, pop: __arrayProto.pop, push: __arrayProto.push, reduce: __arrayProto.reduce, reduceRight: __arrayProto.reduceRight, reverse: __arrayProto.reverse, // Object with rows, columns and opts selector: null, shift: __arrayProto.shift, slice: function () { return new _Api( this.context, this ); }, sort: __arrayProto.sort, splice: __arrayProto.splice, toArray: function () { return __arrayProto.slice.call( this ); }, to$: function () { return $( this ); }, toJQuery: function () { return $( this ); }, unique: function () { return new _Api( this.context, _unique(this.toArray()) ); }, unshift: __arrayProto.unshift } ); function _api_scope( scope, fn, struc ) { return function () { var ret = fn.apply( scope || this, arguments ); // Method extension _Api.extend( ret, ret, struc.methodExt ); return ret; }; } function _api_find( src, name ) { for ( var i=0, ien=src.length ; i<ien ; i++ ) { if ( src[i].name === name ) { return src[i]; } } return null; } window.__apiStruct = __apiStruct; _Api.extend = function ( scope, obj, ext ) { // Only extend API instances and static properties of the API if ( ! ext.length || ! obj || ( ! (obj instanceof _Api) && ! obj.__dt_wrapper ) ) { return; } var i, ien, struct; for ( i=0, ien=ext.length ; i<ien ; i++ ) { struct = ext[i]; if (struct.name === '__proto__') { continue; } // Value obj[ struct.name ] = struct.type === 'function' ? _api_scope( scope, struct.val, struct ) : struct.type === 'object' ? {} : struct.val; obj[ struct.name ].__dt_wrapper = true; // Property extension _Api.extend( scope, obj[ struct.name ], struct.propExt ); } }; // [ // { // name: 'data' -- string - Property name // val: function () {}, -- function - Api method (or undefined if just an object // methodExt: [ ... ], -- array - Array of Api object definitions to extend the method result // propExt: [ ... ] -- array - Array of Api object definitions to extend the property // }, // { // name: 'row' // val: {}, // methodExt: [ ... ], // propExt: [ // { // name: 'data' // val: function () {}, // methodExt: [ ... ], // propExt: [ ... ] // }, // ... // ] // } // ] _Api.register = _api_register = function ( name, val ) { if ( Array.isArray( name ) ) { for ( var j=0, jen=name.length ; j<jen ; j++ ) { _Api.register( name[j], val ); } return; } var i, ien, heir = name.split('.'), struct = __apiStruct, key, method; for ( i=0, ien=heir.length ; i<ien ; i++ ) { method = heir[i].indexOf('()') !== -1; key = method ? heir[i].replace('()', '') : heir[i]; var src = _api_find( struct, key ); if ( ! src ) { src = { name: key, val: {}, methodExt: [], propExt: [], type: 'object' }; struct.push( src ); } if ( i === ien-1 ) { src.val = val; src.type = typeof val === 'function' ? 'function' : $.isPlainObject( val ) ? 'object' : 'other'; } else { struct = method ? src.methodExt : src.propExt; } } }; _Api.registerPlural = _api_registerPlural = function ( pluralName, singularName, val ) { _Api.register( pluralName, val ); _Api.register( singularName, function () { var ret = val.apply( this, arguments ); if ( ret === this ) { // Returned item is the API instance that was passed in, return it return this; } else if ( ret instanceof _Api ) { // New API instance returned, want the value from the first item // in the returned array for the singular result. return ret.length ? Array.isArray( ret[0] ) ? new _Api( ret.context, ret[0] ) : // Array results are 'enhanced' ret[0] : undefined; } // Non-API return - just fire it back return ret; } ); }; /** * Selector for HTML tables. Apply the given selector to the give array of * DataTables settings objects. * * @param {string|integer} [selector] jQuery selector string or integer * @param {array} Array of DataTables settings objects to be filtered * @return {array} * @ignore */ var __table_selector = function ( selector, a ) { if ( Array.isArray(selector) ) { var result = []; selector.forEach(function (sel) { var inner = __table_selector(sel, a); result.push.apply(result, inner); }); return result.filter( function (item) { return item; }); } // Integer is used to pick out a table by index if ( typeof selector === 'number' ) { return [ a[ selector ] ]; } // Perform a jQuery selector on the table nodes var nodes = a.map( function (el) { return el.nTable; } ); return $(nodes) .filter( selector ) .map( function () { // Need to translate back from the table node to the settings var idx = nodes.indexOf(this); return a[ idx ]; } ) .toArray(); }; /** * Context selector for the API's context (i.e. the tables the API instance * refers to. * * @name DataTable.Api#tables * @param {string|integer} [selector] Selector to pick which tables the iterator * should operate on. If not given, all tables in the current context are * used. This can be given as a jQuery selector (for example `':gt(0)'`) to * select multiple tables or as an integer to select a single table. * @returns {DataTable.Api} Returns a new API instance if a selector is given. */ _api_register( 'tables()', function ( selector ) { // A new instance is created if there was a selector specified return selector !== undefined && selector !== null ? new _Api( __table_selector( selector, this.context ) ) : this; } ); _api_register( 'table()', function ( selector ) { var tables = this.tables( selector ); var ctx = tables.context; // Truncate to the first matched table return ctx.length ? new _Api( ctx[0] ) : tables; } ); // Common methods, combined to reduce size [ ['nodes', 'node', 'nTable'], ['body', 'body', 'nTBody'], ['header', 'header', 'nTHead'], ['footer', 'footer', 'nTFoot'], ].forEach(function (item) { _api_registerPlural( 'tables().' + item[0] + '()', 'table().' + item[1] + '()' , function () { return this.iterator( 'table', function ( ctx ) { return ctx[item[2]]; }, 1 ); } ); }); // Structure methods [ ['header', 'aoHeader'], ['footer', 'aoFooter'], ].forEach(function (item) { _api_register( 'table().' + item[0] + '.structure()' , function (selector) { var indexes = this.columns(selector).indexes().flatten(); var ctx = this.context[0]; return _fnHeaderLayout(ctx, ctx[item[1]], indexes); } ); }) _api_registerPlural( 'tables().containers()', 'table().container()' , function () { return this.iterator( 'table', function ( ctx ) { return ctx.nTableWrapper; }, 1 ); } ); _api_register( 'tables().every()', function ( fn ) { var that = this; return this.iterator('table', function (s, i) { fn.call(that.table(i), i); }); }); _api_register( 'caption()', function ( value, side ) { var context = this.context; // Getter - return existing node's content if ( value === undefined ) { var caption = context[0].captionNode; return caption && context.length ? caption.innerHTML : null; } return this.iterator( 'table', function ( ctx ) { var table = $(ctx.nTable); var caption = $(ctx.captionNode); var container = $(ctx.nTableWrapper); // Create the node if it doesn't exist yet if ( ! caption.length ) { caption = $('<caption/>').html( value ); ctx.captionNode = caption[0]; // If side isn't set, we need to insert into the document to let the // CSS decide so we can read it back, otherwise there is no way to // know if the CSS would put it top or bottom for scrolling if (! side) { table.prepend(caption); side = caption.css('caption-side'); } } caption.html( value ); if ( side ) { caption.css( 'caption-side', side ); caption[0]._captionSide = side; } if (container.find('div.dataTables_scroll').length) { var selector = (side === 'top' ? 'Head' : 'Foot'); container.find('div.dataTables_scroll'+ selector +' table').prepend(caption); } else { table.prepend(caption); } }, 1 ); } ); _api_register( 'caption.node()', function () { var ctx = this.context; return ctx.length ? ctx[0].captionNode : null; } ); /** * Redraw the tables in the current context. */ _api_register( 'draw()', function ( paging ) { return this.iterator( 'table', function ( settings ) { if ( paging === 'page' ) { _fnDraw( settings ); } else { if ( typeof paging === 'string' ) { paging = paging === 'full-hold' ? false : true; } _fnReDraw( settings, paging===false ); } } ); } ); /** * 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.iterator( 'table', function ( settings ) { _fnPageChange( settings, action ); } ); } ); /** * 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 () { if ( this.context.length === 0 ) { return undefined; } var settings = this.context[0], start = settings._iDisplayStart, len = settings.oFeatures.bPaginate ? settings._iDisplayLength : -1, visRecords = settings.fnRecordsDisplay(), all = len === -1; return { "page": all ? 0 : Math.floor( start / len ), "pages": all ? 1 : Math.ceil( visRecords / len ), "start": start, "end": settings.fnDisplayEnd(), "length": len, "recordsTotal": settings.fnRecordsTotal(), "recordsDisplay": visRecords, "serverSide": _fnDataSource( settings ) === 'ssp' }; } ); /** * 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.iterator( 'table', function ( settings ) { _fnLengthChange( settings, len ); } ); } ); var __reload = function ( settings, holdPosition, callback ) { // Use the draw event to trigger a callback if ( callback ) { var api = new _Api( settings ); api.one( 'draw', function () { callback( api.ajax.json() ); } ); } if ( _fnDataSource( settings ) == 'ssp' ) { _fnReDraw( settings, holdPosition ); } else { _fnProcessingDisplay( settings, true ); // Cancel an existing request var xhr = settings.jqXHR; if ( xhr && xhr.readyState !== 4 ) { xhr.abort(); } // Trigger xhr _fnBuildAjax( settings, {}, function( json ) { _fnClearTable( settings ); var data = _fnAjaxDataSrc( settings, json ); for ( var i=0, ien=data.length ; i<ien ; i++ ) { _fnAddData( settings, data[i] ); } _fnReDraw( settings, holdPosition ); _fnInitComplete( settings ); _fnProcessingDisplay( settings, false ); } ); } }; /** * Get the JSON response from the last Ajax request that DataTables made to the * server. Note that this returns the JSON from the first table in the current * context. * * @return {object} JSON received from the server. */ _api_register( 'ajax.json()', function () { var ctx = this.context; if ( ctx.length > 0 ) { return ctx[0].json; } // else return undefined; } ); /** * Get the data submitted in the last Ajax request */ _api_register( 'ajax.params()', function () { var ctx = this.context; if ( ctx.length > 0 ) { return ctx[0].oAjaxData; } // else return undefined; } ); /** * Reload tables from the Ajax data source. Note that this function will * automatically re-draw the table when the remote data has been loaded. * * @param {boolean} [reset=true] Reset (default) or hold the current paging * position. A full re-sort and re-filter is performed when this method is * called, which is why the pagination reset is the default action. * @returns {DataTables.Api} this */ _api_register( 'ajax.reload()', function ( callback, resetPaging ) { return this.iterator( 'table', function (settings) { __reload( settings, resetPaging===false, callback ); } ); } ); /** * Get the current Ajax URL. Note that this returns the URL from the first * table in the current context. * * @return {string} Current Ajax source URL *//** * Set the Ajax URL. Note that this will set the URL for all tables in the * current context. * * @param {string} url URL to set. * @returns {DataTables.Api} this */ _api_register( 'ajax.url()', function ( url ) { var ctx = this.context; if ( url === undefined ) { // get if ( ctx.length === 0 ) { return undefined; } ctx = ctx[0]; return $.isPlainObject( ctx.ajax ) ? ctx.ajax.url : ctx.ajax; } // set return this.iterator( 'table', function ( settings ) { if ( $.isPlainObject( settings.ajax ) ) { settings.ajax.url = url; } else { settings.ajax = url; } } ); } ); /** * Load data from the newly set Ajax URL. Note that this method is only * available when `ajax.url()` is used to set a URL. Additionally, this method * has the same effect as calling `ajax.reload()` but is provided for * convenience when setting a new URL. Like `ajax.reload()` it will * automatically redraw the table once the remote data has been loaded. * * @returns {DataTables.Api} this */ _api_register( 'ajax.url().load()', function ( callback, resetPaging ) { // Same as a reload, but makes sense to present it for easy access after a // url change return this.iterator( 'table', function ( ctx ) { __reload( ctx, resetPaging===false, callback ); } ); } ); var _selector_run = function ( type, selector, selectFn, settings, opts ) { var out = [], res, a, i, ien, j, jen, selectorType = typeof selector; // Can't just check for isArray here, as an API or jQuery instance might be // given with their array like look if ( ! selector || selectorType === 'string' || selectorType === 'function' || selector.length === undefined ) { selector = [ selector ]; } for ( i=0, ien=selector.length ; i<ien ; i++ ) { // Only split on simple strings - complex expressions will be jQuery selectors a = selector[i] && selector[i].split && ! selector[i].match(/[[(:]/) ? selector[i].split(',') : [ selector[i] ]; for ( j=0, jen=a.length ; j<jen ; j++ ) { res = selectFn( typeof a[j] === 'string' ? (a[j]).trim() : a[j] ); // Remove empty items res = res.filter( function (item) { return item !== null && item !== undefined; }); if ( res && res.length ) { out = out.concat( res ); } } } // selector extensions var ext = _ext.selector[ type ]; if ( ext.length ) { for ( i=0, ien=ext.length ; i<ien ; i++ ) { out = ext[i]( settings, opts, out ); } } return _unique( out ); }; var _selector_opts = function ( opts ) { if ( ! opts ) { opts = {}; } // Backwards compatibility for 1.9- which used the terminology filter rather // than search if ( opts.filter && opts.search === undefined ) { opts.search = opts.filter; } return $.extend( { search: 'none', order: 'current', page: 'all' }, opts ); }; // Reduce the API instance to the first item found var _selector_first = function ( old ) { let inst = new _Api(old.context[0]); // Use a push rather than passing to the constructor, since it will // merge arrays down automatically, which isn't what is wanted here if (old.length) { inst.push( old[0] ); } inst.selector = old.selector; // Limit to a single row / column / cell if (inst.length && inst[0].length > 1) { inst[0].splice(1); } return inst; }; var _selector_row_indexes = function ( settings, opts ) { var i, ien, tmp, a=[], displayFiltered = settings.aiDisplay, displayMaster = settings.aiDisplayMaster; var search = opts.search, // none, applied, removed order = opts.order, // applied, current, index (original - compatibility with 1.9) page = opts.page; // all, current if ( _fnDataSource( settings ) == 'ssp' ) { // In server-side processing mode, most options are irrelevant since // rows not shown don't exist and the index order is the applied order // Removed is a special case - for consistency just return an empty // array return search === 'removed' ? [] : _range( 0, displayMaster.length ); } if ( page == 'current' ) { // Current page implies that order=current and filter=applied, since it is // fairly senseless otherwise, regardless of what order and search actually // are for ( i=settings._iDisplayStart, ien=settings.fnDisplayEnd() ; i<ien ; i++ ) { a.push( displayFiltered[i] ); } } else if ( order == 'current' || order == 'applied' ) { if ( search == 'none') { a = displayMaster.slice(); } else if ( search == 'applied' ) { a = displayFiltered.slice(); } else if ( search == 'removed' ) { // O(n+m) solution by creating a hash map var displayFilteredMap = {}; for ( i=0, ien=displayFiltered.length ; i<ien ; i++ ) { displayFilteredMap[displayFiltered[i]] = null; } displayMaster.forEach(function (item) { if (! Object.prototype.hasOwnProperty.call(displayFilteredMap, item)) { a.push(item); } }); } } else if ( order == 'index' || order == 'original' ) { for ( i=0, ien=settings.aoData.length ; i<ien ; i++ ) { if (! settings.aoData[i]) { continue; } if ( search == 'none' ) { a.push( i ); } else { // applied | removed tmp = displayFiltered.indexOf(i); if ((tmp === -1 && search == 'removed') || (tmp >= 0 && search == 'applied') ) { a.push( i ); } } } } else if ( typeof order === 'number' ) { // Order the rows by the given column var ordered = _fnSort(settings, order, 'asc'); if (search === 'none') { a = ordered; } else { // applied | removed for (i=0; i<ordered.length; i++) { tmp = displayFiltered.indexOf(ordered[i]); if ((tmp === -1 && search == 'removed') || (tmp >= 0 && search == 'applied') ) { a.push( ordered[i] ); } } } } return a; }; /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Rows * * {} - no selector - use all available rows * {integer} - row aoData index * {node} - TR node * {string} - jQuery selector to apply to the TR elements * {array} - jQuery array of nodes, or simply an array of TR nodes * */ var __row_selector = function ( settings, selector, opts ) { var rows; var run = function ( sel ) { var selInt = _intVal( sel ); var aoData = settings.aoData; // Short cut - selector is a number and no options provided (default is // all records, so no need to check if the index is in there, since it // must be - dev error if the index doesn't exist). if ( selInt !== null && ! opts ) { return [ selInt ]; } if ( ! rows ) { rows = _selector_row_indexes( settings, opts ); } if ( selInt !== null && rows.indexOf(selInt) !== -1 ) { // Selector - integer return [ selInt ]; } else if ( sel === null || sel === undefined || sel === '' ) { // Selector - none return rows; } // Selector - function if ( typeof sel === 'function' ) { return rows.map( function (idx) { var row = aoData[ idx ]; return sel( idx, row._aData, row.nTr ) ? idx : null; } ); } // Selector - node if ( sel.nodeName ) { var rowIdx = sel._DT_RowIndex; // Property added by DT for fast lookup var cellIdx = sel._DT_CellIndex; if ( rowIdx !== undefined ) { // Make sure that the row is actually still present in the table return aoData[ rowIdx ] && aoData[ rowIdx ].nTr === sel ? [ rowIdx ] : []; } else if ( cellIdx ) { return aoData[ cellIdx.row ] && aoData[ cellIdx.row ].nTr === sel.parentNode ? [ cellIdx.row ] : []; } else { var host = $(sel).closest('*[data-dt-row]'); return host.length ? [ host.data('dt-row') ] : []; } } // ID selector. Want to always be able to select rows by id, regardless // of if the tr element has been created or not, so can't rely upon // jQuery here - hence a custom implementation. This does not match // Sizzle's fast selector or HTML4 - in HTML5 the ID can be anything, // but to select it using a CSS selector engine (like Sizzle or // querySelect) it would need to need to be escaped for some characters. // DataTables simplifies this for row selectors since you can select // only a row. A # indicates an id any anything that follows is the id - // unescaped. if ( typeof sel === 'string' && sel.charAt(0) === '#' ) { // get row index from id var rowObj = settings.aIds[ sel.replace( /^#/, '' ) ]; if ( rowObj !== undefined ) { return [ rowObj.idx ]; } // need to fall through to jQuery in case there is DOM id that // matches } // Get nodes in the order from the `rows` array with null values removed var nodes = _removeEmpty( _pluck_order( settings.aoData, rows, 'nTr' ) ); // Selector - jQuery selector string, array of nodes or jQuery object/ // As jQuery's .filter() allows jQuery objects to be passed in filter, // it also allows arrays, so this will cope with all three options return $(nodes) .filter( sel ) .map( function () { return this._DT_RowIndex; } ) .toArray(); }; var matched = _selector_run( 'row', selector, run, settings, opts ); if (opts.order === 'current' || opts.order === 'applied') { _fnSortDisplay(settings, matched); } return matched; }; _api_register( 'rows()', function ( selector, opts ) { // argument shifting if ( selector === undefined ) { selector = ''; } else if ( $.isPlainObject( selector ) ) { opts = selector; selector = ''; } opts = _selector_opts( opts ); var inst = this.iterator( 'table', function ( settings ) { return __row_selector( settings, selector, opts ); }, 1 ); // Want argument shifting here and in __row_selector? inst.selector.rows = selector; inst.selector.opts = opts; return inst; } ); _api_register( 'rows().nodes()', function () { return this.iterator( 'row', function ( settings, row ) { return settings.aoData[ row ].nTr || undefined; }, 1 ); } ); _api_register( 'rows().data()', function () { return this.iterator( true, 'rows', function ( settings, rows ) { return _pluck_order( settings.aoData, rows, '_aData' ); }, 1 ); } ); _api_registerPlural( 'rows().cache()', 'row().cache()', function ( type ) { return this.iterator( 'row', function ( settings, row ) { var r = settings.aoData[ row ]; return type === 'search' ? r._aFilterData : r._aSortData; }, 1 ); } ); _api_registerPlural( 'rows().invalidate()', 'row().invalidate()', function ( src ) { return this.iterator( 'row', function ( settings, row ) { _fnInvalidate( settings, row, src ); } ); } ); _api_registerPlural( 'rows().indexes()', 'row().index()', function () { return this.iterator( 'row', function ( settings, row ) { return row; }, 1 ); } ); _api_registerPlural( 'rows().ids()', 'row().id()', function ( hash ) { var a = []; var context = this.context; // `iterator` will drop undefined values, but in this case we want them for ( var i=0, ien=context.length ; i<ien ; i++ ) { for ( var j=0, jen=this[i].length ; j<jen ; j++ ) { var id = context[i].rowIdFn( context[i].aoData[ this[i][j] ]._aData ); a.push( (hash === true ? '#' : '' )+ id ); } } return new _Api( context, a ); } ); _api_registerPlural( 'rows().remove()', 'row().remove()', function () { this.iterator( 'row', function ( settings, row ) { var data = settings.aoData; var rowData = data[ row ]; // Delete from the display arrays var idx = settings.aiDisplayMaster.indexOf(row); if (idx !== -1) { settings.aiDisplayMaster.splice(idx, 1); } // For server-side processing tables - subtract the deleted row from the count if ( settings._iRecordsDisplay > 0 ) { settings._iRecordsDisplay--; } // Check for an 'overflow' they case for displaying the table _fnLengthOverflow( settings ); // Remove the row's ID reference if there is one var id = settings.rowIdFn( rowData._aData ); if ( id !== undefined ) { delete settings.aIds[ id ]; } data[row] = null; } ); return this; } ); _api_register( 'rows.add()', function ( rows ) { var newRows = this.iterator( 'table', function ( settings ) { var row, i, ien; var out = []; for ( i=0, ien=rows.length ; i<ien ; i++ ) { row = rows[i]; if ( row.nodeName && row.nodeName.toUpperCase() === 'TR' ) { out.push( _fnAddTr( settings, row )[0] ); } else { out.push( _fnAddData( settings, row ) ); } } return out; }, 1 ); // Return an Api.rows() extended instance, so rows().nodes() etc can be used var modRows = this.rows( -1 ); modRows.pop(); modRows.push.apply(modRows, newRows); return modRows; } ); /** * */ _api_register( 'row()', function ( selector, opts ) { return _selector_first( this.rows( selector, opts ) ); } ); _api_register( 'row().data()', function ( data ) { var ctx = this.context; if ( data === undefined ) { // Get return ctx.length && this.length && this[0].length ? ctx[0].aoData[ this[0] ]._aData : undefined; } // Set var row = ctx[0].aoData[ this[0] ]; row._aData = data; // If the DOM has an id, and the data source is an array if ( Array.isArray( data ) && row.nTr && row.nTr.id ) { _fnSetObjectDataFn( ctx[0].rowId )( data, row.nTr.id ); } // Automatically invalidate _fnInvalidate( ctx[0], this[0], 'data' ); return this; } ); _api_register( 'row().node()', function () { var ctx = this.context; if (ctx.length && this.length && this[0].length) { var row = ctx[0].aoData[ this[0] ]; if (row && row.nTr) { return row.nTr; } } return null; } ); _api_register( 'row.add()', function ( row ) { // Allow a jQuery object to be passed in - only a single row is added from // it though - the first element in the set if ( row instanceof $ && row.length ) { row = row[0]; } var rows = this.iterator( 'table', function ( settings ) { if ( row.nodeName && row.nodeName.toUpperCase() === 'TR' ) { return _fnAddTr( settings, row )[0]; } return _fnAddData( settings, row ); } ); // Return an Api.rows() extended instance, with the newly added row selected return this.row( rows[0] ); } ); $(document).on('plugin-init.dt', function (e, context) { var api = new _Api( context ); api.on( 'stateSaveParams.DT', function ( e, settings, d ) { // This could be more compact with the API, but it is a lot faster as a simple // internal loop var idFn = settings.rowIdFn; var rows = settings.aiDisplayMaster; var ids = []; for (var i=0 ; i<rows.length ; i++) { var rowIdx = rows[i]; var data = settings.aoData[rowIdx]; if (data._detailsShow) { ids.push( '#' + idFn(data._aData) ); } } d.childRows = ids; }); // For future state loads (e.g. with StateRestore) api.on( 'stateLoaded.DT', function (e, settings, state) { __details_state_load( api, state ); }); // And the initial load state __details_state_load( api, api.state.loaded() ); }); var __details_state_load = function (api, state) { if ( state && state.childRows ) { api .rows( state.childRows.map(function (id) { // Escape any `:` characters from the row id. Accounts for // already escaped characters. return id.replace(/([^:\\]*(?:\\.[^:\\]*)*):/g, "$1\\:"); }) ) .every( function () { _fnCallbackFire( api.settings()[0], null, 'requestChild', [ this ] ) }); } } var __details_add = function ( ctx, row, data, klass ) { // Convert to array of TR elements var rows = []; var addRow = function ( r, k ) { // Recursion to allow for arrays of jQuery objects if ( Array.isArray( r ) || r instanceof $ ) { for ( var i=0, ien=r.length ; i<ien ; i++ ) { addRow( r[i], k ); } return; } // If we get a TR element, then just add it directly - up to the dev // to add the correct number of columns etc if ( r.nodeName && r.nodeName.toLowerCase() === 'tr' ) { r.setAttribute( 'data-dt-row', row.idx ); rows.push( r ); } else { // Otherwise create a row with a wrapper var created = $('<tr><td></td></tr>') .attr( 'data-dt-row', row.idx ) .addClass( k ); $('td', created) .addClass( k ) .html( r )[0].colSpan = _fnVisbleColumns( ctx ); rows.push( created[0] ); } }; addRow( data, klass ); if ( row._details ) { row._details.detach(); } row._details = $(rows); // If the children were already shown, that state should be retained if ( row._detailsShow ) { row._details.insertAfter( row.nTr ); } }; // Make state saving of child row details async to allow them to be batch processed var __details_state = DataTable.util.throttle( function (ctx) { _fnSaveState( ctx[0] ) }, 500 ); var __details_remove = function ( api, idx ) { var ctx = api.context; if ( ctx.length ) { var row = ctx[0].aoData[ idx !== undefined ? idx : api[0] ]; if ( row && row._details ) { row._details.remove(); row._detailsShow = undefined; row._details = undefined; $( row.nTr ).removeClass( 'dt-hasChild' ); __details_state( ctx ); } } }; var __details_display = function ( api, show ) { var ctx = api.context; if ( ctx.length && api.length ) { var row = ctx[0].aoData[ api[0] ]; if ( row._details ) { row._detailsShow = show; if ( show ) { row._details.insertAfter( row.nTr ); $( row.nTr ).addClass( 'dt-hasChild' ); } else { row._details.detach(); $( row.nTr ).removeClass( 'dt-hasChild' ); } _fnCallbackFire( ctx[0], null, 'childRow', [ show, api.row( api[0] ) ] ) __details_events( ctx[0] ); __details_state( ctx ); } } }; var __details_events = function ( settings ) { var api = new _Api( settings ); var namespace = '.dt.DT_details'; var drawEvent = 'draw'+namespace; var colvisEvent = 'column-sizing'+namespace; var destroyEvent = 'destroy'+namespace; var data = settings.aoData; api.off( drawEvent +' '+ colvisEvent +' '+ destroyEvent ); if ( _pluck( data, '_details' ).length > 0 ) { // On each draw, insert the required elements into the document api.on( drawEvent, function ( e, ctx ) { if ( settings !== ctx ) { return; } api.rows( {page:'current'} ).eq(0).each( function (idx) { // Internal data grab var row = data[ idx ]; if ( row._detailsShow ) { row._details.insertAfter( row.nTr ); } } ); } ); // Column visibility change - update the colspan api.on( colvisEvent, function ( e, ctx ) { if ( settings !== ctx ) { return; } // Update the colspan for the details rows (note, only if it already has // a colspan) var row, visible = _fnVisbleColumns( ctx ); for ( var i=0, ien=data.length ; i<ien ; i++ ) { row = data[i]; if ( row && row._details ) { row._details.each(function () { var el = $(this).children('td'); if (el.length == 1) { el.attr('colspan', visible); } }); } } } ); // Table destroyed - nuke any child rows api.on( destroyEvent, function ( e, ctx ) { if ( settings !== ctx ) { return; } for ( var i=0, ien=data.length ; i<ien ; i++ ) { if ( data[i] && data[i]._details ) { __details_remove( api, i ); } } } ); } }; // Strings for the method names to help minification var _emp = ''; var _child_obj = _emp+'row().child'; var _child_mth = _child_obj+'()'; // data can be: // tr // string // jQuery or array of any of the above _api_register( _child_mth, function ( data, klass ) { var ctx = this.context; if ( data === undefined ) { // get return ctx.length && this.length && ctx[0].aoData[ this[0] ] ? ctx[0].aoData[ this[0] ]._details : undefined; } else if ( data === true ) { // show this.child.show(); } else if ( data === false ) { // remove __details_remove( this ); } else if ( ctx.length && this.length ) { // set __details_add( ctx[0], ctx[0].aoData[ this[0] ], data, klass ); } return this; } ); _api_register( [ _child_obj+'.show()', _child_mth+'.show()' // only when `child()` was called with parameters (without ], function () { // it returns an object and this method is not executed) __details_display( this, true ); return this; } ); _api_register( [ _child_obj+'.hide()', _child_mth+'.hide()' // only when `child()` was called with parameters (without ], function () { // it returns an object and this method is not executed) __details_display( this, false ); return this; } ); _api_register( [ _child_obj+'.remove()', _child_mth+'.remove()' // only when `child()` was called with parameters (without ], function () { // it returns an object and this method is not executed) __details_remove( this ); return this; } ); _api_register( _child_obj+'.isShown()', function () { var ctx = this.context; if ( ctx.length && this.length && ctx[0].aoData[ this[0] ] ) { // _detailsShown as false or undefined will fall through to return false return ctx[0].aoData[ this[0] ]._detailsShow || false; } return false; } ); /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Columns * * {integer} - column index (>=0 count from left, <0 count from right) * "{integer}:visIdx" - visible column index (i.e. translate to column index) (>=0 count from left, <0 count from right) * "{integer}:visible" - alias for {integer}:visIdx (>=0 count from left, <0 count from right) * "{string}:name" - column name * "{string}" - jQuery selector on column header nodes * */ // can be an array of these items, comma separated list, or an array of comma // separated lists var __re_column_selector = /^([^:]+)?:(name|title|visIdx|visible)$/; // r1 and r2 are redundant - but it means that the parameters match for the // iterator callback in columns().data() var __columnData = function ( settings, column, r1, r2, rows, type ) { var a = []; for ( var row=0, ien=rows.length ; row<ien ; row++ ) { a.push( _fnGetCellData( settings, rows[row], column, type ) ); } return a; }; var __column_header = function ( settings, column, row ) { var header = settings.aoHeader; var target = row !== undefined ? row : settings.bSortCellsTop // legacy support ? 0 : header.length - 1; return header[target][column].cell; }; var __column_selector = function ( settings, selector, opts ) { var columns = settings.aoColumns, names = _pluck( columns, 'sName' ), titles = _pluck( columns, 'sTitle' ), cells = DataTable.util.get('[].[].cell')(settings.aoHeader), nodes = _unique( _flatten([], cells) ); var run = function ( s ) { var selInt = _intVal( s ); // Selector - all if ( s === '' ) { return _range( columns.length ); } // Selector - index if ( selInt !== null ) { return [ selInt >= 0 ? selInt : // Count from left columns.length + selInt // Count from right (+ because its a negative value) ]; } // Selector = function if ( typeof s === 'function' ) { var rows = _selector_row_indexes( settings, opts ); return columns.map(function (col, idx) { return s( idx, __columnData( settings, idx, 0, 0, rows ), __column_header( settings, idx ) ) ? idx : null; }); } // jQuery or string selector var match = typeof s === 'string' ? s.match( __re_column_selector ) : ''; if ( match ) { switch( match[2] ) { case 'visIdx': case 'visible': if (match[1]) { var idx = parseInt( match[1], 10 ); // Visible index given, convert to column index if ( idx < 0 ) { // Counting from the right var visColumns = columns.map( function (col,i) { return col.bVisible ? i : null; } ); return [ visColumns[ visColumns.length + idx ] ]; } // Counting from the left return [ _fnVisibleToColumnIndex( settings, idx ) ]; } // `:visible` on its own return columns.map( function (col, i) { return col.bVisible ? i : null; } ); case 'name': // match by name. `names` is column index complete and in order return names.map( function (name, i) { return name === match[1] ? i : null; } ); case 'title': // match by column title return titles.map( function (title, i) { return title === match[1] ? i : null; } ); default: return []; } } // Cell in the table body if ( s.nodeName && s._DT_CellIndex ) { return [ s._DT_CellIndex.column ]; } // jQuery selector on the TH elements for the columns var jqResult = $( nodes ) .filter( s ) .map( function () { return _fnColumnsFromHeader( this ); // `nodes` is column index complete and in order } ) .toArray(); if ( jqResult.length || ! s.nodeName ) { return jqResult; } // Otherwise a node which might have a `dt-column` data attribute, or be // a child or such an element var host = $(s).closest('*[data-dt-column]'); return host.length ? [ host.data('dt-column') ] : []; }; return _selector_run( 'column', selector, run, settings, opts ); }; var __setColumnVis = function ( settings, column, vis ) { var cols = settings.aoColumns, col = cols[ column ], data = settings.aoData, cells, i, ien, tr; // Get if ( vis === undefined ) { return col.bVisible; } // Set // No change if ( col.bVisible === vis ) { return false; } if ( vis ) { // Insert column // Need to decide if we should use appendChild or insertBefore var insertBefore = _pluck(cols, 'bVisible').indexOf(true, column+1); for ( i=0, ien=data.length ; i<ien ; i++ ) { if (data[i]) { tr = data[i].nTr; cells = data[i].anCells; if ( tr ) { // insertBefore can act like appendChild if 2nd arg is null tr.insertBefore( cells[ column ], cells[ insertBefore ] || null ); } } } } else { // Remove column $( _pluck( settings.aoData, 'anCells', column ) ).detach(); } // Common actions col.bVisible = vis; _colGroup(settings); return true; }; _api_register( 'columns()', function ( selector, opts ) { // argument shifting if ( selector === undefined ) { selector = ''; } else if ( $.isPlainObject( selector ) ) { opts = selector; selector = ''; } opts = _selector_opts( opts ); var inst = this.iterator( 'table', function ( settings ) { return __column_selector( settings, selector, opts ); }, 1 ); // Want argument shifting here and in _row_selector? inst.selector.cols = selector; inst.selector.opts = opts; return inst; } ); _api_registerPlural( 'columns().header()', 'column().header()', function ( row ) { return this.iterator( 'column', function (settings, column) { return __column_header(settings, column, row); }, 1 ); } ); _api_registerPlural( 'columns().footer()', 'column().footer()', function ( row ) { return this.iterator( 'column', function ( settings, column ) { var footer = settings.aoFooter; if (! footer.length) { return null; } return settings.aoFooter[row !== undefined ? row : 0][column].cell; }, 1 ); } ); _api_registerPlural( 'columns().data()', 'column().data()', function () { return this.iterator( 'column-rows', __columnData, 1 ); } ); _api_registerPlural( 'columns().render()', 'column().render()', function ( type ) { return this.iterator( 'column-rows', function ( settings, column, i, j, rows ) { return __columnData( settings, column, i, j, rows, type ); }, 1 ); } ); _api_registerPlural( 'columns().dataSrc()', 'column().dataSrc()', function () { return this.iterator( 'column', function ( settings, column ) { return settings.aoColumns[column].mData; }, 1 ); } ); _api_registerPlural( 'columns().cache()', 'column().cache()', function ( type ) { return this.iterator( 'column-rows', function ( settings, column, i, j, rows ) { return _pluck_order( settings.aoData, rows, type === 'search' ? '_aFilterData' : '_aSortData', column ); }, 1 ); } ); _api_registerPlural( 'columns().init()', 'column().init()', function () { return this.iterator( 'column', function ( settings, column ) { return settings.aoColumns[column]; }, 1 ); } ); _api_registerPlural( 'columns().nodes()', 'column().nodes()', function () { return this.iterator( 'column-rows', function ( settings, column, i, j, rows ) { return _pluck_order( settings.aoData, rows, 'anCells', column ) ; }, 1 ); } ); _api_registerPlural( 'columns().titles()', 'column().title()', function (title, row) { return this.iterator( 'column', function ( settings, column ) { // Argument shifting if (typeof title === 'number') { row = title; title = undefined; } var span = $('span.dt-column-title', this.column(column).header(row)); if (title !== undefined) { span.html(title); return this; } return span.html(); }, 1 ); } ); _api_registerPlural( 'columns().types()', 'column().type()', function () { return this.iterator( 'column', function ( settings, column ) { var type = settings.aoColumns[column].sType; // If the type was invalidated, then resolve it. This actually does // all columns at the moment. Would only happen once if getting all // column's data types. if (! type) { _fnColumnTypes(settings); } return type; }, 1 ); } ); _api_registerPlural( 'columns().visible()', 'column().visible()', function ( vis, calc ) { var that = this; var changed = []; var ret = this.iterator( 'column', function ( settings, column ) { if ( vis === undefined ) { return settings.aoColumns[ column ].bVisible; } // else if (__setColumnVis( settings, column, vis )) { changed.push(column); } } ); // Group the column visibility changes if ( vis !== undefined ) { this.iterator( 'table', function ( settings ) { // Redraw the header after changes _fnDrawHead( settings, settings.aoHeader ); _fnDrawHead( settings, settings.aoFooter ); // Update colspan for no records display. Child rows and extensions will use their own // listeners to do this - only need to update the empty table item here if ( ! settings.aiDisplay.length ) { $(settings.nTBody).find('td[colspan]').attr('colspan', _fnVisbleColumns(settings)); } _fnSaveState( settings ); // Second loop once the first is done for events that.iterator( 'column', function ( settings, column ) { if (changed.includes(column)) { _fnCallbackFire( settings, null, 'column-visibility', [settings, column, vis, calc] ); } } ); if ( changed.length && (calc === undefined || calc) ) { that.columns.adjust(); } }); } return ret; } ); _api_registerPlural( 'columns().widths()', 'column().width()', function () { // Injects a fake row into the table for just a moment so the widths can // be read, regardless of colspan in the header and rows being present in // the body var columns = this.columns(':visible').count(); var row = $('<tr>').html('<td>' + Array(columns).join('</td><td>') + '</td>'); $(this.table().body()).append(row); var widths = row.children().map(function () { return $(this).outerWidth(); }); row.remove(); return this.iterator( 'column', function ( settings, column ) { var visIdx = _fnColumnIndexToVisible( settings, column ); return visIdx !== null ? widths[visIdx] : 0; }, 1); } ); _api_registerPlural( 'columns().indexes()', 'column().index()', function ( type ) { return this.iterator( 'column', function ( settings, column ) { return type === 'visible' ? _fnColumnIndexToVisible( settings, column ) : column; }, 1 ); } ); _api_register( 'columns.adjust()', function () { return this.iterator( 'table', function ( settings ) { _fnAdjustColumnSizing( settings ); }, 1 ); } ); _api_register( 'column.index()', function ( type, idx ) { if ( this.context.length !== 0 ) { var ctx = this.context[0]; if ( type === 'fromVisible' || type === 'toData' ) { return _fnVisibleToColumnIndex( ctx, idx ); } else if ( type === 'fromData' || type === 'toVisible' ) { return _fnColumnIndexToVisible( ctx, idx ); } } } ); _api_register( 'column()', function ( selector, opts ) { return _selector_first( this.columns( selector, opts ) ); } ); var __cell_selector = function ( settings, selector, opts ) { var data = settings.aoData; var rows = _selector_row_indexes( settings, opts ); var cells = _removeEmpty( _pluck_order( data, rows, 'anCells' ) ); var allCells = $(_flatten( [], cells )); var row; var columns = settings.aoColumns.length; var a, i, ien, j, o, host; var run = function ( s ) { var fnSelector = typeof s === 'function'; if ( s === null || s === undefined || fnSelector ) { // All cells and function selectors a = []; for ( i=0, ien=rows.length ; i<ien ; i++ ) { row = rows[i]; for ( j=0 ; j<columns ; j++ ) { o = { row: row, column: j }; if ( fnSelector ) { // Selector - function host = data[ row ]; if ( s( o, _fnGetCellData(settings, row, j), host.anCells ? host.anCells[j] : null ) ) { a.push( o ); } } else { // Selector - all a.push( o ); } } } return a; } // Selector - index if ( $.isPlainObject( s ) ) { // Valid cell index and its in the array of selectable rows return s.column !== undefined && s.row !== undefined && rows.indexOf(s.row) !== -1 ? [s] : []; } // Selector - jQuery filtered cells var jqResult = allCells .filter( s ) .map( function (i, el) { return { // use a new object, in case someone changes the values row: el._DT_CellIndex.row, column: el._DT_CellIndex.column }; } ) .toArray(); if ( jqResult.length || ! s.nodeName ) { return jqResult; } // Otherwise the selector is a node, and there is one last option - the // element might be a child of an element which has dt-row and dt-column // data attributes host = $(s).closest('*[data-dt-row]'); return host.length ? [ { row: host.data('dt-row'), column: host.data('dt-column') } ] : []; }; return _selector_run( 'cell', selector, run, settings, opts ); }; _api_register( 'cells()', function ( rowSelector, columnSelector, opts ) { // Argument shifting if ( $.isPlainObject( rowSelector ) ) { // Indexes if ( rowSelector.row === undefined ) { // Selector options in first parameter opts = rowSelector; rowSelector = null; } else { // Cell index objects in first parameter opts = columnSelector; columnSelector = null; } } if ( $.isPlainObject( columnSelector ) ) { opts = columnSelector; columnSelector = null; } // Cell selector if ( columnSelector === null || columnSelector === undefined ) { return this.iterator( 'table', function ( settings ) { return __cell_selector( settings, rowSelector, _selector_opts( opts ) ); } ); } // The default built in options need to apply to row and columns var internalOpts = opts ? { page: opts.page, order: opts.order, search: opts.search } : {}; // Row + column selector var columns = this.columns( columnSelector, internalOpts ); var rows = this.rows( rowSelector, internalOpts ); var i, ien, j, jen; var cellsNoOpts = this.iterator( 'table', function ( settings, idx ) { var a = []; for ( i=0, ien=rows[idx].length ; i<ien ; i++ ) { for ( j=0, jen=columns[idx].length ; j<jen ; j++ ) { a.push( { row: rows[idx][i], column: columns[idx][j] } ); } } return a; }, 1 ); // There is currently only one extension which uses a cell selector extension // It is a _major_ performance drag to run this if it isn't needed, so this is // an extension specific check at the moment var cells = opts && opts.selected ? this.cells( cellsNoOpts, opts ) : cellsNoOpts; $.extend( cells.selector, { cols: columnSelector, rows: rowSelector, opts: opts } ); return cells; } ); _api_registerPlural( 'cells().nodes()', 'cell().node()', function () { return this.iterator( 'cell', function ( settings, row, column ) { var data = settings.aoData[ row ]; return data && data.anCells ? data.anCells[ column ] : undefined; }, 1 ); } ); _api_register( 'cells().data()', function () { return this.iterator( 'cell', function ( settings, row, column ) { return _fnGetCellData( settings, row, column ); }, 1 ); } ); _api_registerPlural( 'cells().cache()', 'cell().cache()', function ( type ) { type = type === 'search' ? '_aFilterData' : '_aSortData'; return this.iterator( 'cell', function ( settings, row, column ) { return settings.aoData[ row ][ type ][ column ]; }, 1 ); } ); _api_registerPlural( 'cells().render()', 'cell().render()', function ( type ) { return this.iterator( 'cell', function ( settings, row, column ) { return _fnGetCellData( settings, row, column, type ); }, 1 ); } ); _api_registerPlural( 'cells().indexes()', 'cell().index()', function () { return this.iterator( 'cell', function ( settings, row, column ) { return { row: row, column: column, columnVisible: _fnColumnIndexToVisible( settings, column ) }; }, 1 ); } ); _api_registerPlural( 'cells().invalidate()', 'cell().invalidate()', function ( src ) { return this.iterator( 'cell', function ( settings, row, column ) { _fnInvalidate( settings, row, src, column ); } ); } ); _api_register( 'cell()', function ( rowSelector, columnSelector, opts ) { return _selector_first( this.cells( rowSelector, columnSelector, opts ) ); } ); _api_register( 'cell().data()', function ( data ) { var ctx = this.context; var cell = this[0]; if ( data === undefined ) { // Get return ctx.length && cell.length ? _fnGetCellData( ctx[0], cell[0].row, cell[0].column ) : undefined; } // Set _fnSetCellData( ctx[0], cell[0].row, cell[0].column, data ); _fnInvalidate( ctx[0], cell[0].row, 'data', cell[0].column ); return this; } ); /** * Get current ordering (sorting) that has been applied to the table. * * @returns {array} 2D array containing the sorting information for the first * table in the current context. Each element in the parent array represents * a column being sorted upon (i.e. multi-sorting with two columns would have * 2 inner arrays). The inner arrays may have 2 or 3 elements. The first is * the column index that the sorting condition applies to, the second is the * direction of the sort (`desc` or `asc`) and, optionally, the third is the * index of the sorting order from the `column.sorting` initialisation array. *//** * Set the ordering for the table. * * @param {integer} order Column index to sort upon. * @param {string} direction Direction of the sort to be applied (`asc` or `desc`) * @returns {DataTables.Api} this *//** * Set the ordering for the table. * * @param {array} order 1D array of sorting information to be applied. * @param {array} [...] Optional additional sorting conditions * @returns {DataTables.Api} this *//** * Set the ordering for the table. * * @param {array} order 2D array of sorting information to be applied. * @returns {DataTables.Api} this */ _api_register( 'order()', function ( order, dir ) { var ctx = this.context; var args = Array.prototype.slice.call( arguments ); if ( order === undefined ) { // get return ctx.length !== 0 ? ctx[0].aaSorting : undefined; } // set if ( typeof order === 'number' ) { // Simple column / direction passed in order = [ [ order, dir ] ]; } else if ( args.length > 1 ) { // Arguments passed in (list of 1D arrays) order = args; } // otherwise a 2D array was passed in return this.iterator( 'table', function ( settings ) { settings.aaSorting = Array.isArray(order) ? order.slice() : order; } ); } ); /** * Attach a sort listener to an element for a given column * * @param {node|jQuery|string} node Identifier for the element(s) to attach the * listener to. This can take the form of a single DOM node, a jQuery * collection of nodes or a jQuery selector which will identify the node(s). * @param {integer} column the column that a click on this node will sort on * @param {function} [callback] callback function when sort is run * @returns {DataTables.Api} this */ _api_register( 'order.listener()', function ( node, column, callback ) { return this.iterator( 'table', function ( settings ) { _fnSortAttachListener(settings, node, {}, column, callback); } ); } ); _api_register( 'order.fixed()', function ( set ) { if ( ! set ) { var ctx = this.context; var fixed = ctx.length ? ctx[0].aaSortingFixed : undefined; return Array.isArray( fixed ) ? { pre: fixed } : fixed; } return this.iterator( 'table', function ( settings ) { settings.aaSortingFixed = $.extend( true, {}, set ); } ); } ); // Order by the selected column(s) _api_register( [ 'columns().order()', 'column().order()' ], function ( dir ) { var that = this; if ( ! dir ) { return this.iterator( 'column', function ( settings, idx ) { var sort = _fnSortFlatten( settings ); for ( var i=0, ien=sort.length ; i<ien ; i++ ) { if ( sort[i].col === idx ) { return sort[i].dir; } } return null; }, 1 ); } else { return this.iterator( 'table', function ( settings, i ) { settings.aaSorting = that[i].map( function (col) { return [ col, dir ]; } ); } ); } } ); _api_registerPlural('columns().orderable()', 'column().orderable()', function ( directions ) { return this.iterator( 'column', function ( settings, idx ) { var col = settings.aoColumns[idx]; return directions ? col.asSorting : col.bSortable; }, 1 ); } ); _api_register( 'processing()', function ( show ) { return this.iterator( 'table', function ( ctx ) { _fnProcessingDisplay( ctx, show ); } ); } ); _api_register( 'search()', function ( input, regex, smart, caseInsen ) { var ctx = this.context; if ( input === undefined ) { // get return ctx.length !== 0 ? ctx[0].oPreviousSearch.search : undefined; } // set return this.iterator( 'table', function ( settings ) { if ( ! settings.oFeatures.bFilter ) { return; } if (typeof regex === 'object') { // New style options to pass to the search builder _fnFilterComplete( settings, $.extend( settings.oPreviousSearch, regex, { search: input } ) ); } else { // Compat for the old options _fnFilterComplete( settings, $.extend( settings.oPreviousSearch, { search: input, regex: regex === null ? false : regex, smart: smart === null ? true : smart, caseInsensitive: caseInsen === null ? true : caseInsen } ) ); } } ); } ); _api_register( 'search.fixed()', function ( name, search ) { var ret = this.iterator( true, 'table', function ( settings ) { var fixed = settings.searchFixed; if (! name) { return Object.keys(fixed) } else if (search === undefined) { return fixed[name]; } else if (search === null) { delete fixed[name]; } else { fixed[name] = search; } return this; } ); return name !== undefined && search === undefined ? ret[0] : ret; } ); _api_registerPlural( 'columns().search()', 'column().search()', function ( input, regex, smart, caseInsen ) { return this.iterator( 'column', function ( settings, column ) { var preSearch = settings.aoPreSearchCols; if ( input === undefined ) { // get return preSearch[ column ].search; } // set if ( ! settings.oFeatures.bFilter ) { return; } if (typeof regex === 'object') { // New style options to pass to the search builder $.extend( preSearch[ column ], regex, { search: input } ); } else { // Old style (with not all options available) $.extend( preSearch[ column ], { search: input, regex: regex === null ? false : regex, smart: smart === null ? true : smart, caseInsensitive: caseInsen === null ? true : caseInsen } ); } _fnFilterComplete( settings, settings.oPreviousSearch ); } ); } ); _api_register([ 'columns().search.fixed()', 'column().search.fixed()' ], function ( name, search ) { var ret = this.iterator( true, 'column', function ( settings, colIdx ) { var fixed = settings.aoColumns[colIdx].searchFixed; if (! name) { return Object.keys(fixed) } else if (search === undefined) { return fixed[name]; } else if (search === null) { delete fixed[name]; } else { fixed[name] = search; } return this; } ); return name !== undefined && search === undefined ? ret[0] : ret; } ); /* * State API methods */ _api_register( 'state()', function ( set, ignoreTime ) { // getter if ( ! set ) { return this.context.length ? this.context[0].oSavedState : null; } var setMutate = $.extend( true, {}, set ); // setter return this.iterator( 'table', function ( settings ) { if ( ignoreTime !== false ) { setMutate.time = +new Date() + 100; } _fnImplementState( settings, setMutate, function(){} ); } ); } ); _api_register( 'state.clear()', function () { return this.iterator( 'table', function ( settings ) { // Save an empty object settings.fnStateSaveCallback.call( settings.oInstance, settings, {} ); } ); } ); _api_register( 'state.loaded()', function () { return this.context.length ? this.context[0].oLoadedState : null; } ); _api_register( 'state.save()', function () { return this.iterator( 'table', function ( settings ) { _fnSaveState( settings ); } ); } ); /** * Set the jQuery or window object to be used by DataTables * * @param {*} module Library / container object * @param {string} [type] Library or container type `lib`, `win` or `datetime`. * If not provided, automatic detection is attempted. */ DataTable.use = function (module, type) { if (type === 'lib' || module.fn) { $ = module; } else if (type == 'win' || module.document) { window = module; document = module.document; } else if (type === 'datetime' || module.type === 'DateTime') { DataTable.DateTime = module; } } /** * CommonJS factory function pass through. This will check if the arguments * given are a window object or a jQuery object. If so they are set * accordingly. * @param {*} root Window * @param {*} jq jQUery * @returns {boolean} Indicator */ DataTable.factory = function (root, jq) { var is = false; // Test if the first parameter is a window object if (root && root.document) { window = root; document = root.document; } // Test if the second parameter is a jQuery object if (jq && jq.fn && jq.fn.jquery) { $ = jq; is = true; } return is; } /** * Provide a common method for plug-ins to check the version of DataTables being * used, in order to ensure compatibility. * * @param {string} version Version string to check for, in the format "X.Y.Z". * Note that the formats "X" and "X.Y" are also acceptable. * @param {string} [version2=current DataTables version] As above, but optional. * If not given the current DataTables version will be used. * @returns {boolean} true if this version of DataTables is greater or equal to * the required version, or false if this version of DataTales is not * suitable * @static * @dtopt API-Static * * @example * alert( $.fn.dataTable.versionCheck( '1.9.0' ) ); */ DataTable.versionCheck = function( version, version2 ) { var aThis = version2 ? version2.split('.') : DataTable.version.split('.'); var aThat = version.split('.'); var iThis, iThat; for ( var i=0, iLen=aThat.length ; i<iLen ; i++ ) { iThis = parseInt( aThis[i], 10 ) || 0; iThat = parseInt( aThat[i], 10 ) || 0; // Parts are the same, keep comparing if (iThis === iThat) { continue; } // Parts are different, return immediately return iThis > iThat; } return true; }; /** * Check if a `<table>` node is a DataTable table already or not. * * @param {node|jquery|string} table Table node, jQuery object or jQuery * selector for the table to test. Note that if more than more than one * table is passed on, only the first will be checked * @returns {boolean} true the table given is a DataTable, or false otherwise * @static * @dtopt API-Static * * @example * if ( ! $.fn.DataTable.isDataTable( '#example' ) ) { * $('#example').dataTable(); * } */ DataTable.isDataTable = function ( table ) { var t = $(table).get(0); var is = false; if ( table instanceof DataTable.Api ) { return true; } $.each( DataTable.settings, function (i, o) { var head = o.nScrollHead ? $('table', o.nScrollHead)[0] : null; var foot = o.nScrollFoot ? $('table', o.nScrollFoot)[0] : null; if ( o.nTable === t || head === t || foot === t ) { is = true; } } ); return is; }; /** * Get all DataTable tables that have been initialised - optionally you can * select to get only currently visible tables. * * @param {boolean} [visible=false] Flag to indicate if you want all (default) * or visible tables only. * @returns {array} Array of `table` nodes (not DataTable instances) which are * DataTables * @static * @dtopt API-Static * * @example * $.each( $.fn.dataTable.tables(true), function () { * $(table).DataTable().columns.adjust(); * } ); */ DataTable.tables = function ( visible ) { var api = false; if ( $.isPlainObject( visible ) ) { api = visible.api; visible = visible.visible; } var a = DataTable.settings .filter( function (o) { return !visible || (visible && $(o.nTable).is(':visible')) ? true : false; } ) .map( function (o) { return o.nTable; }); return api ? new _Api( a ) : a; }; /** * Convert from camel case parameters to Hungarian notation. This is made public * for the extensions to provide the same ability as DataTables core to accept * either the 1.9 style Hungarian notation, or the 1.10+ style camelCase * parameters. * * @param {object} src The model object which holds all parameters that can be * mapped. * @param {object} user The object to convert from camel case to Hungarian. * @param {boolean} force When set to `true`, properties which already have a * Hungarian value in the `user` object will be overwritten. Otherwise they * won't be. */ DataTable.camelToHungarian = _fnCamelToHungarian; /** * */ _api_register( '$()', function ( selector, opts ) { var rows = this.rows( opts ).nodes(), // Get all rows jqRows = $(rows); return $( [].concat( jqRows.filter( selector ).toArray(), jqRows.find( selector ).toArray() ) ); } ); // jQuery functions to operate on the tables $.each( [ 'on', 'one', 'off' ], function (i, key) { _api_register( key+'()', function ( /* event, handler */ ) { var args = Array.prototype.slice.call(arguments); // Add the `dt` namespace automatically if it isn't already present args[0] = args[0].split( /\s/ ).map( function ( e ) { return ! e.match(/\.dt\b/) ? e+'.dt' : e; } ).join( ' ' ); var inst = $( this.tables().nodes() ); inst[key].apply( inst, args ); return this; } ); } ); _api_register( 'clear()', function () { return this.iterator( 'table', function ( settings ) { _fnClearTable( settings ); } ); } ); _api_register( 'error()', function (msg) { return this.iterator( 'table', function ( settings ) { _fnLog( settings, 0, msg ); } ); } ); _api_register( 'settings()', function () { return new _Api( this.context, this.context ); } ); _api_register( 'init()', function () { var ctx = this.context; return ctx.length ? ctx[0].oInit : null; } ); _api_register( 'data()', function () { return this.iterator( 'table', function ( settings ) { return _pluck( settings.aoData, '_aData' ); } ).flatten(); } ); _api_register( 'trigger()', function ( name, args, bubbles ) { return this.iterator( 'table', function ( settings ) { return _fnCallbackFire( settings, null, name, args, bubbles ); } ).flatten(); } ); _api_register( 'ready()', function ( fn ) { var ctx = this.context; // Get status of first table if (! fn) { return ctx.length ? (ctx[0]._bInitComplete || false) : null; } // Function to run either once the table becomes ready or // immediately if it is already ready. return this.tables().every(function () { if (this.context[0]._bInitComplete) { fn.call(this); } else { this.on('init', function () { fn.call(this); }); } } ); } ); _api_register( 'destroy()', function ( remove ) { remove = remove || false; return this.iterator( 'table', function ( settings ) { var classes = settings.oClasses; var table = settings.nTable; var tbody = settings.nTBody; var thead = settings.nTHead; var tfoot = settings.nTFoot; var jqTable = $(table); var jqTbody = $(tbody); var jqWrapper = $(settings.nTableWrapper); var rows = settings.aoData.map( function (r) { return r ? r.nTr : null; } ); var orderClasses = classes.order; // Flag to note that the table is currently being destroyed - no action // should be taken settings.bDestroying = true; // Fire off the destroy callbacks for plug-ins etc _fnCallbackFire( settings, "aoDestroyCallback", "destroy", [settings], true ); // If not being removed from the document, make all columns visible if ( ! remove ) { new _Api( settings ).columns().visible( true ); } // Blitz all `DT` namespaced events (these are internal events, the // lowercase, `dt` events are user subscribed and they are responsible // for removing them jqWrapper.off('.DT').find(':not(tbody *)').off('.DT'); $(window).off('.DT-'+settings.sInstance); // When scrolling we had to break the table up - restore it if ( table != thead.parentNode ) { jqTable.children('thead').detach(); jqTable.append( thead ); } if ( tfoot && table != tfoot.parentNode ) { jqTable.children('tfoot').detach(); jqTable.append( tfoot ); } settings.colgroup.remove(); settings.aaSorting = []; settings.aaSortingFixed = []; _fnSortingClasses( settings ); $('th, td', thead) .removeClass( orderClasses.canAsc + ' ' + orderClasses.canDesc + ' ' + orderClasses.isAsc + ' ' + orderClasses.isDesc ) .css('width', ''); // Add the TR elements back into the table in their original order jqTbody.children().detach(); jqTbody.append( rows ); var orig = settings.nTableWrapper.parentNode; var insertBefore = settings.nTableWrapper.nextSibling; // Remove the DataTables generated nodes, events and classes var removedMethod = remove ? 'remove' : 'detach'; jqTable[ removedMethod ](); jqWrapper[ removedMethod ](); // If we need to reattach the table to the document if ( ! remove && orig ) { // insertBefore acts like appendChild if !arg[1] orig.insertBefore( table, insertBefore ); // Restore the width of the original table - was read from the style property, // so we can restore directly to that jqTable .css( 'width', settings.sDestroyWidth ) .removeClass( classes.table ); } /* Remove the settings object from the settings array */ var idx = DataTable.settings.indexOf(settings); if ( idx !== -1 ) { DataTable.settings.splice( idx, 1 ); } } ); } ); // Add the `every()` method for rows, columns and cells in a compact form $.each( [ 'column', 'row', 'cell' ], function ( i, type ) { _api_register( type+'s().every()', function ( fn ) { var opts = this.selector.opts; var api = this; var inst; var counter = 0; return this.iterator( 'every', function ( settings, selectedIdx, tableIdx ) { inst = api[ type ](selectedIdx, opts); if (type === 'cell') { fn.call(inst, inst[0][0].row, inst[0][0].column, tableIdx, counter); } else { fn.call(inst, selectedIdx, tableIdx, counter); } counter++; } ); } ); } ); // i18n method for extensions to be able to use the language object from the // DataTable _api_register( 'i18n()', function ( token, def, plural ) { var ctx = this.context[0]; var resolved = _fnGetObjectDataFn( token )( ctx.oLanguage ); if ( resolved === undefined ) { resolved = def; } if ( $.isPlainObject( resolved ) ) { resolved = plural !== undefined && resolved[ plural ] !== undefined ? resolved[ plural ] : resolved._; } return typeof resolved === 'string' ? resolved.replace( '%d', plural ) // nb: plural might be undefined, : resolved; } ); /** * Version string for plug-ins to check compatibility. Allowed format is * `a.b.c-d` where: a:int, b:int, c:int, d:string(dev|beta|alpha). `d` is used * only for non-release builds. See https://semver.org/ for more information. * @member * @type string * @default Version number */ DataTable.version = "2.0.8"; /** * Private data store, containing all of the settings objects that are * created for the tables on a given page. * * Note that the `DataTable.settings` object is aliased to * `jQuery.fn.dataTableExt` through which it may be accessed and * manipulated, or `jQuery.fn.dataTable.settings`. * @member * @type array * @default [] * @private */ DataTable.settings = []; /** * Object models container, for the various models that DataTables has * available to it. These models define the objects that are used to hold * the active state and configuration of the table. * @namespace */ DataTable.models = {}; /** * Template object for the way in which DataTables holds information about * search information for the global filter and individual column filters. * @namespace */ DataTable.models.oSearch = { /** * Flag to indicate if the filtering should be case insensitive or not */ "caseInsensitive": true, /** * Applied search term */ "search": "", /** * Flag to indicate if the search term should be interpreted as a * regular expression (true) or not (false) and therefore and special * regex characters escaped. */ "regex": false, /** * Flag to indicate if DataTables is to use its smart filtering or not. */ "smart": true, /** * Flag to indicate if DataTables should only trigger a search when * the return key is pressed. */ "return": false }; /** * Template object for the way in which DataTables holds information about * each individual row. This is the object format used for the settings * aoData array. * @namespace */ DataTable.models.oRow = { /** * TR element for the row */ "nTr": null, /** * Array of TD elements for each row. This is null until the row has been * created. */ "anCells": null, /** * Data object from the original data source for the row. This is either * an array if using the traditional form of DataTables, or an object if * using mData options. The exact type will depend on the passed in * data from the data source, or will be an array if using DOM a data * source. */ "_aData": [], /** * Sorting data cache - this array is ostensibly the same length as the * number of columns (although each index is generated only as it is * needed), and holds the data that is used for sorting each column in the * row. We do this cache generation at the start of the sort in order that * the formatting of the sort data need be done only once for each cell * per sort. This array should not be read from or written to by anything * other than the master sorting methods. */ "_aSortData": null, /** * Per cell filtering data cache. As per the sort data cache, used to * increase the performance of the filtering in DataTables */ "_aFilterData": null, /** * Filtering data cache. This is the same as the cell filtering cache, but * in this case a string rather than an array. This is easily computed with * a join on `_aFilterData`, but is provided as a cache so the join isn't * needed on every search (memory traded for performance) */ "_sFilterRow": null, /** * Denote if the original data source was from the DOM, or the data source * object. This is used for invalidating data, so DataTables can * automatically read data from the original source, unless uninstructed * otherwise. */ "src": null, /** * Index in the aoData array. This saves an indexOf lookup when we have the * object, but want to know the index */ "idx": -1, /** * Cached display value */ displayData: null }; /** * Template object for the column information object in DataTables. This object * is held in the settings aoColumns array and contains all the information that * DataTables needs about each individual column. * * Note that this object is related to {@link DataTable.defaults.column} * but this one is the internal data store for DataTables's cache of columns. * It should NOT be manipulated outside of DataTables. Any configuration should * be done through the initialisation options. * @namespace */ DataTable.models.oColumn = { /** * Column index. */ "idx": null, /** * A list of the columns that sorting should occur on when this column * is sorted. That this property is an array allows multi-column sorting * to be defined for a column (for example first name / last name columns * would benefit from this). The values are integers pointing to the * columns to be sorted on (typically it will be a single integer pointing * at itself, but that doesn't need to be the case). */ "aDataSort": null, /** * Define the sorting directions that are applied to the column, in sequence * as the column is repeatedly sorted upon - i.e. the first value is used * as the sorting direction when the column if first sorted (clicked on). * Sort it again (click again) and it will move on to the next index. * Repeat until loop. */ "asSorting": null, /** * Flag to indicate if the column is searchable, and thus should be included * in the filtering or not. */ "bSearchable": null, /** * Flag to indicate if the column is sortable or not. */ "bSortable": null, /** * Flag to indicate if the column is currently visible in the table or not */ "bVisible": null, /** * Store for manual type assignment using the `column.type` option. This * is held in store so we can manipulate the column's `sType` property. */ "_sManualType": null, /** * Flag to indicate if HTML5 data attributes should be used as the data * source for filtering or sorting. True is either are. */ "_bAttrSrc": false, /** * Developer definable function that is called whenever a cell is created (Ajax source, * etc) or processed for input (DOM source). This can be used as a compliment to mRender * allowing you to modify the DOM element (add background colour for example) when the * element is available. */ "fnCreatedCell": null, /** * Function to get data from a cell in a column. You should <b>never</b> * access data directly through _aData internally in DataTables - always use * the method attached to this property. It allows mData to function as * required. This function is automatically assigned by the column * initialisation method */ "fnGetData": null, /** * Function to set data for a cell in the column. You should <b>never</b> * set the data directly to _aData internally in DataTables - always use * this method. It allows mData to function as required. This function * is automatically assigned by the column initialisation method */ "fnSetData": null, /** * Property to read the value for the cells in the column from the data * source array / object. If null, then the default content is used, if a * function is given then the return from the function is used. */ "mData": null, /** * Partner property to mData which is used (only when defined) to get * the data - i.e. it is basically the same as mData, but without the * 'set' option, and also the data fed to it is the result from mData. * This is the rendering method to match the data method of mData. */ "mRender": null, /** * The class to apply to all TD elements in the table's TBODY for the column */ "sClass": null, /** * When DataTables calculates the column widths to assign to each column, * it finds the longest string in each column and then constructs a * temporary table and reads the widths from that. The problem with this * is that "mmm" is much wider then "iiii", but the latter is a longer * string - thus the calculation can go wrong (doing it properly and putting * it into an DOM object and measuring that is horribly(!) slow). Thus as * a "work around" we provide this option. It will append its value to the * text that is found to be the longest string for the column - i.e. padding. */ "sContentPadding": null, /** * Allows a default value to be given for a column's data, and will be used * whenever a null data source is encountered (this can be because mData * is set to null, or because the data source itself is null). */ "sDefaultContent": null, /** * Name for the column, allowing reference to the column by name as well as * by index (needs a lookup to work by name). */ "sName": null, /** * Custom sorting data type - defines which of the available plug-ins in * afnSortData the custom sorting will use - if any is defined. */ "sSortDataType": 'std', /** * Class to be applied to the header element when sorting on this column */ "sSortingClass": null, /** * Title of the column - what is seen in the TH element (nTh). */ "sTitle": null, /** * Column sorting and filtering type */ "sType": null, /** * Width of the column */ "sWidth": null, /** * Width of the column when it was first "encountered" */ "sWidthOrig": null, /** Cached string which is the longest in the column */ maxLenString: null, /** * Store for named searches */ searchFixed: null }; /* * Developer note: The properties of the object below are given in Hungarian * notation, that was used as the interface for DataTables prior to v1.10, however * from v1.10 onwards the primary interface is camel case. In order to avoid * breaking backwards compatibility utterly with this change, the Hungarian * version is still, internally the primary interface, but is is not documented * - hence the @name tags in each doc comment. This allows a Javascript function * to create a map from Hungarian notation to camel case (going the other direction * would require each property to be listed, which would add around 3K to the size * of DataTables, while this method is about a 0.5K hit). * * Ultimately this does pave the way for Hungarian notation to be dropped * completely, but that is a massive amount of work and will break current * installs (therefore is on-hold until v2). */ /** * Initialisation options that can be given to DataTables at initialisation * time. * @namespace */ DataTable.defaults = { /** * An array of data to use for the table, passed in at initialisation which * will be used in preference to any data which is already in the DOM. This is * particularly useful for constructing tables purely in Javascript, for * example with a custom Ajax call. */ "aaData": null, /** * If ordering is enabled, then DataTables will perform a first pass sort on * initialisation. You can define which column(s) the sort is performed * upon, and the sorting direction, with this variable. The `sorting` array * should contain an array for each column to be sorted initially containing * the column's index and a direction string ('asc' or 'desc'). */ "aaSorting": [[0,'asc']], /** * This parameter is basically identical to the `sorting` parameter, but * cannot be overridden by user interaction with the table. What this means * is that you could have a column (visible or hidden) which the sorting * will always be forced on first - any sorting after that (from the user) * will then be performed as required. This can be useful for grouping rows * together. */ "aaSortingFixed": [], /** * DataTables can be instructed to load data to display in the table from a * Ajax source. This option defines how that Ajax call is made and where to. * * The `ajax` property has three different modes of operation, depending on * how it is defined. These are: * * * `string` - Set the URL from where the data should be loaded from. * * `object` - Define properties for `jQuery.ajax`. * * `function` - Custom data get function * * `string` * -------- * * As a string, the `ajax` property simply defines the URL from which * DataTables will load data. * * `object` * -------- * * As an object, the parameters in the object are passed to * [jQuery.ajax](https://api.jquery.com/jQuery.ajax/) allowing fine control * of the Ajax request. DataTables has a number of default parameters which * you can override using this option. Please refer to the jQuery * documentation for a full description of the options available, although * the following parameters provide additional options in DataTables or * require special consideration: * * * `data` - As with jQuery, `data` can be provided as an object, but it * can also be used as a function to manipulate the data DataTables sends * to the server. The function takes a single parameter, an object of * parameters with the values that DataTables has readied for sending. An * object may be returned which will be merged into the DataTables * defaults, or you can add the items to the object that was passed in and * not return anything from the function. This supersedes `fnServerParams` * from DataTables 1.9-. * * * `dataSrc` - By default DataTables will look for the property `data` (or * `aaData` for compatibility with DataTables 1.9-) when obtaining data * from an Ajax source or for server-side processing - this parameter * allows that property to be changed. You can use Javascript dotted * object notation to get a data source for multiple levels of nesting, or * it my be used as a function. As a function it takes a single parameter, * the JSON returned from the server, which can be manipulated as * required, with the returned value being that used by DataTables as the * data source for the table. * * * `success` - Should not be overridden it is used internally in * DataTables. To manipulate / transform the data returned by the server * use `ajax.dataSrc`, or use `ajax` as a function (see below). * * `function` * ---------- * * As a function, making the Ajax call is left up to yourself allowing * complete control of the Ajax request. Indeed, if desired, a method other * than Ajax could be used to obtain the required data, such as Web storage * or an AIR database. * * The function is given four parameters and no return is required. The * parameters are: * * 1. _object_ - Data to send to the server * 2. _function_ - Callback function that must be executed when the required * data has been obtained. That data should be passed into the callback * as the only parameter * 3. _object_ - DataTables settings object for the table */ "ajax": null, /** * This parameter allows you to readily specify the entries in the length drop * down menu that DataTables shows when pagination is enabled. It can be * either a 1D array of options which will be used for both the displayed * option and the value, or a 2D array which will use the array in the first * position as the value, and the array in the second position as the * displayed options (useful for language strings such as 'All'). * * Note that the `pageLength` property will be automatically set to the * first value given in this array, unless `pageLength` is also provided. */ "aLengthMenu": [ 10, 25, 50, 100 ], /** * The `columns` option in the initialisation parameter allows you to define * details about the way individual columns behave. For a full list of * column options that can be set, please see * {@link DataTable.defaults.column}. Note that if you use `columns` to * define your columns, you must have an entry in the array for every single * column that you have in your table (these can be null if you don't which * to specify any options). */ "aoColumns": null, /** * Very similar to `columns`, `columnDefs` allows you to target a specific * column, multiple columns, or all columns, using the `targets` property of * each object in the array. This allows great flexibility when creating * tables, as the `columnDefs` arrays can be of any length, targeting the * columns you specifically want. `columnDefs` may use any of the column * options available: {@link DataTable.defaults.column}, but it _must_ * have `targets` defined in each object in the array. Values in the `targets` * array may be: * <ul> * <li>a string - class name will be matched on the TH for the column</li> * <li>0 or a positive integer - column index counting from the left</li> * <li>a negative integer - column index counting from the right</li> * <li>the string "_all" - all columns (i.e. assign a default)</li> * </ul> */ "aoColumnDefs": null, /** * Basically the same as `search`, this parameter defines the individual column * filtering state at initialisation time. The array must be of the same size * as the number of columns, and each element be an object with the parameters * `search` and `escapeRegex` (the latter is optional). 'null' is also * accepted and the default will be used. */ "aoSearchCols": [], /** * Enable or disable automatic column width calculation. This can be disabled * as an optimisation (it takes some time to calculate the widths) if the * tables widths are passed in using `columns`. */ "bAutoWidth": true, /** * Deferred rendering can provide DataTables with a huge speed boost when you * are using an Ajax or JS data source for the table. This option, when set to * true, will cause DataTables to defer the creation of the table elements for * each row until they are needed for a draw - saving a significant amount of * time. */ "bDeferRender": true, /** * Replace a DataTable which matches the given selector and replace it with * one which has the properties of the new initialisation object passed. If no * table matches the selector, then the new DataTable will be constructed as * per normal. */ "bDestroy": false, /** * Enable or disable filtering of data. Filtering in DataTables is "smart" in * that it allows the end user to input multiple words (space separated) and * will match a row containing those words, even if not in the order that was * specified (this allow matching across multiple columns). Note that if you * wish to use filtering in DataTables this must remain 'true' - to remove the * default filtering input box and retain filtering abilities, please use * {@link DataTable.defaults.dom}. */ "bFilter": true, /** * Used only for compatiblity with DT1 * @deprecated */ "bInfo": true, /** * Used only for compatiblity with DT1 * @deprecated */ "bLengthChange": true, /** * Enable or disable pagination. */ "bPaginate": true, /** * Enable or disable the display of a 'processing' indicator when the table is * being processed (e.g. a sort). This is particularly useful for tables with * large amounts of data where it can take a noticeable amount of time to sort * the entries. */ "bProcessing": false, /** * Retrieve the DataTables object for the given selector. Note that if the * table has already been initialised, this parameter will cause DataTables * to simply return the object that has already been set up - it will not take * account of any changes you might have made to the initialisation object * passed to DataTables (setting this parameter to true is an acknowledgement * that you understand this). `destroy` can be used to reinitialise a table if * you need. */ "bRetrieve": false, /** * When vertical (y) scrolling is enabled, DataTables will force the height of * the table's viewport to the given height at all times (useful for layout). * However, this can look odd when filtering data down to a small data set, * and the footer is left "floating" further down. This parameter (when * enabled) will cause DataTables to collapse the table's viewport down when * the result set will fit within the given Y height. */ "bScrollCollapse": false, /** * Configure DataTables to use server-side processing. Note that the * `ajax` parameter must also be given in order to give DataTables a * source to obtain the required data for each draw. */ "bServerSide": false, /** * Enable or disable sorting of columns. Sorting of individual columns can be * disabled by the `sortable` option for each column. */ "bSort": true, /** * Enable or display DataTables' ability to sort multiple columns at the * same time (activated by shift-click by the user). */ "bSortMulti": true, /** * Allows control over whether DataTables should use the top (true) unique * cell that is found for a single column, or the bottom (false - default). * This is useful when using complex headers. */ "bSortCellsTop": null, /** * Enable or disable the addition of the classes `sorting\_1`, `sorting\_2` and * `sorting\_3` to the columns which are currently being sorted on. This is * presented as a feature switch as it can increase processing time (while * classes are removed and added) so for large data sets you might want to * turn this off. */ "bSortClasses": true, /** * Enable or disable state saving. When enabled HTML5 `localStorage` will be * used to save table display information such as pagination information, * display length, filtering and sorting. As such when the end user reloads * the page the display display will match what thy had previously set up. */ "bStateSave": false, /** * This function is called when a TR element is created (and all TD child * elements have been inserted), or registered if using a DOM source, allowing * manipulation of the TR element (adding classes etc). */ "fnCreatedRow": null, /** * This function is called on every 'draw' event, and allows you to * dynamically modify any aspect you want about the created DOM. */ "fnDrawCallback": null, /** * Identical to fnHeaderCallback() but for the table footer this function * allows you to modify the table footer on every 'draw' event. */ "fnFooterCallback": null, /** * When rendering large numbers in the information element for the table * (i.e. "Showing 1 to 10 of 57 entries") DataTables will render large numbers * to have a comma separator for the 'thousands' units (e.g. 1 million is * rendered as "1,000,000") to help readability for the end user. This * function will override the default method DataTables uses. */ "fnFormatNumber": function ( toFormat ) { return toFormat.toString().replace( /\B(?=(\d{3})+(?!\d))/g, this.oLanguage.sThousands ); }, /** * This function is called on every 'draw' event, and allows you to * dynamically modify the header row. This can be used to calculate and * display useful information about the table. */ "fnHeaderCallback": null, /** * The information element can be used to convey information about the current * state of the table. Although the internationalisation options presented by * DataTables are quite capable of dealing with most customisations, there may * be times where you wish to customise the string further. This callback * allows you to do exactly that. */ "fnInfoCallback": null, /** * Called when the table has been initialised. Normally DataTables will * initialise sequentially and there will be no need for this function, * however, this does not hold true when using external language information * since that is obtained using an async XHR call. */ "fnInitComplete": null, /** * Called at the very start of each table draw and can be used to cancel the * draw by returning false, any other return (including undefined) results in * the full draw occurring). */ "fnPreDrawCallback": null, /** * This function allows you to 'post process' each row after it have been * generated for each table draw, but before it is rendered on screen. This * function might be used for setting the row class name etc. */ "fnRowCallback": null, /** * Load the table state. With this function you can define from where, and how, the * state of a table is loaded. By default DataTables will load from `localStorage` * but you might wish to use a server-side database or cookies. */ "fnStateLoadCallback": function ( settings ) { try { return JSON.parse( (settings.iStateDuration === -1 ? sessionStorage : localStorage).getItem( 'DataTables_'+settings.sInstance+'_'+location.pathname ) ); } catch (e) { return {}; } }, /** * Callback which allows modification of the saved state prior to loading that state. * This callback is called when the table is loading state from the stored data, but * prior to the settings object being modified by the saved state. Note that for * plug-in authors, you should use the `stateLoadParams` event to load parameters for * a plug-in. */ "fnStateLoadParams": null, /** * Callback that is called when the state has been loaded from the state saving method * and the DataTables settings object has been modified as a result of the loaded state. */ "fnStateLoaded": null, /** * Save the table state. This function allows you to define where and how the state * information for the table is stored By default DataTables will use `localStorage` * but you might wish to use a server-side database or cookies. */ "fnStateSaveCallback": function ( settings, data ) { try { (settings.iStateDuration === -1 ? sessionStorage : localStorage).setItem( 'DataTables_'+settings.sInstance+'_'+location.pathname, JSON.stringify( data ) ); } catch (e) { // noop } }, /** * Callback which allows modification of the state to be saved. Called when the table * has changed state a new state save is required. This method allows modification of * the state saving object prior to actually doing the save, including addition or * other state properties or modification. Note that for plug-in authors, you should * use the `stateSaveParams` event to save parameters for a plug-in. */ "fnStateSaveParams": null, /** * Duration for which the saved state information is considered valid. After this period * has elapsed the state will be returned to the default. * Value is given in seconds. */ "iStateDuration": 7200, /** * Number of rows to display on a single page when using pagination. If * feature enabled (`lengthChange`) then the end user will be able to override * this to a custom setting using a pop-up menu. */ "iDisplayLength": 10, /** * Define the starting point for data display when using DataTables with * pagination. Note that this parameter is the number of records, rather than * the page number, so if you have 10 records per page and want to start on * the third page, it should be "20". */ "iDisplayStart": 0, /** * By default DataTables allows keyboard navigation of the table (sorting, paging, * and filtering) by adding a `tabindex` attribute to the required elements. This * allows you to tab through the controls and press the enter key to activate them. * The tabindex is default 0, meaning that the tab follows the flow of the document. * You can overrule this using this parameter if you wish. Use a value of -1 to * disable built-in keyboard navigation. */ "iTabIndex": 0, /** * Classes that DataTables assigns to the various components and features * that it adds to the HTML table. This allows classes to be configured * during initialisation in addition to through the static * {@link DataTable.ext.oStdClasses} object). */ "oClasses": {}, /** * All strings that DataTables uses in the user interface that it creates * are defined in this object, allowing you to modified them individually or * completely replace them all as required. */ "oLanguage": { /** * Strings that are used for WAI-ARIA labels and controls only (these are not * actually visible on the page, but will be read by screenreaders, and thus * must be internationalised as well). */ "oAria": { /** * ARIA label that is added to the table headers when the column may be sorted */ "orderable": ": Activate to sort", /** * ARIA label that is added to the table headers when the column is currently being sorted */ "orderableReverse": ": Activate to invert sorting", /** * ARIA label that is added to the table headers when the column is currently being * sorted and next step is to remove sorting */ "orderableRemove": ": Activate to remove sorting", paginate: { first: 'First', last: 'Last', next: 'Next', previous: 'Previous' } }, /** * Pagination string used by DataTables for the built-in pagination * control types. */ "oPaginate": { /** * Label and character for first page button («) */ "sFirst": "\u00AB", /** * Last page button (») */ "sLast": "\u00BB", /** * Next page button (›) */ "sNext": "\u203A", /** * Previous page button (‹) */ "sPrevious": "\u2039", }, /** * Plural object for the data type the table is showing */ entries: { _: "entries", 1: "entry" }, /** * This string is shown in preference to `zeroRecords` when the table is * empty of data (regardless of filtering). Note that this is an optional * parameter - if it is not given, the value of `zeroRecords` will be used * instead (either the default or given value). */ "sEmptyTable": "No data available in table", /** * This string gives information to the end user about the information * that is current on display on the page. The following tokens can be * used in the string and will be dynamically replaced as the table * display updates. This tokens can be placed anywhere in the string, or * removed as needed by the language requires: * * * `\_START\_` - Display index of the first record on the current page * * `\_END\_` - Display index of the last record on the current page * * `\_TOTAL\_` - Number of records in the table after filtering * * `\_MAX\_` - Number of records in the table without filtering * * `\_PAGE\_` - Current page number * * `\_PAGES\_` - Total number of pages of data in the table */ "sInfo": "Showing _START_ to _END_ of _TOTAL_ _ENTRIES-TOTAL_", /** * Display information string for when the table is empty. Typically the * format of this string should match `info`. */ "sInfoEmpty": "Showing 0 to 0 of 0 _ENTRIES-TOTAL_", /** * When a user filters the information in a table, this string is appended * to the information (`info`) to give an idea of how strong the filtering * is. The variable _MAX_ is dynamically updated. */ "sInfoFiltered": "(filtered from _MAX_ total _ENTRIES-MAX_)", /** * If can be useful to append extra information to the info string at times, * and this variable does exactly that. This information will be appended to * the `info` (`infoEmpty` and `infoFiltered` in whatever combination they are * being used) at all times. */ "sInfoPostFix": "", /** * This decimal place operator is a little different from the other * language options since DataTables doesn't output floating point * numbers, so it won't ever use this for display of a number. Rather, * what this parameter does is modify the sort methods of the table so * that numbers which are in a format which has a character other than * a period (`.`) as a decimal place will be sorted numerically. * * Note that numbers with different decimal places cannot be shown in * the same table and still be sortable, the table must be consistent. * However, multiple different tables on the page can use different * decimal place characters. */ "sDecimal": "", /** * DataTables has a build in number formatter (`formatNumber`) which is * used to format large numbers that are used in the table information. * By default a comma is used, but this can be trivially changed to any * character you wish with this parameter. */ "sThousands": ",", /** * Detail the action that will be taken when the drop down menu for the * pagination length option is changed. The '_MENU_' variable is replaced * with a default select list of 10, 25, 50 and 100, and can be replaced * with a custom select box if required. */ "sLengthMenu": "_MENU_ _ENTRIES_ per page", /** * When using Ajax sourced data and during the first draw when DataTables is * gathering the data, this message is shown in an empty row in the table to * indicate to the end user the the data is being loaded. Note that this * parameter is not used when loading data by server-side processing, just * Ajax sourced data with client-side processing. */ "sLoadingRecords": "Loading...", /** * Text which is displayed when the table is processing a user action * (usually a sort command or similar). */ "sProcessing": "", /** * Details the actions that will be taken when the user types into the * filtering input text box. The variable "_INPUT_", if used in the string, * is replaced with the HTML text box for the filtering input allowing * control over where it appears in the string. If "_INPUT_" is not given * then the input box is appended to the string automatically. */ "sSearch": "Search:", /** * Assign a `placeholder` attribute to the search `input` element * @type string * @default * * @dtopt Language * @name DataTable.defaults.language.searchPlaceholder */ "sSearchPlaceholder": "", /** * All of the language information can be stored in a file on the * server-side, which DataTables will look up if this parameter is passed. * It must store the URL of the language file, which is in a JSON format, * and the object has the same properties as the oLanguage object in the * initialiser object (i.e. the above parameters). Please refer to one of * the example language files to see how this works in action. */ "sUrl": "", /** * Text shown inside the table records when the is no information to be * displayed after filtering. `emptyTable` is shown when there is simply no * information in the table at all (regardless of filtering). */ "sZeroRecords": "No matching records found" }, /** * This parameter allows you to have define the global filtering state at * initialisation time. As an object the `search` parameter must be * defined, but all other parameters are optional. When `regex` is true, * the search string will be treated as a regular expression, when false * (default) it will be treated as a straight string. When `smart` * DataTables will use it's smart filtering methods (to word match at * any point in the data), when false this will not be done. */ "oSearch": $.extend( {}, DataTable.models.oSearch ), /** * Table and control layout. This replaces the legacy `dom` option. */ layout: { topStart: 'pageLength', topEnd: 'search', bottomStart: 'info', bottomEnd: 'paging' }, /** * Legacy DOM layout option */ "sDom": null, /** * Search delay option. This will throttle full table searches that use the * DataTables provided search input element (it does not effect calls to * `dt-api search()`, providing a delay before the search is made. */ "searchDelay": null, /** * DataTables features six different built-in options for the buttons to * display for pagination control: * * * `numbers` - Page number buttons only * * `simple` - 'Previous' and 'Next' buttons only * * 'simple_numbers` - 'Previous' and 'Next' buttons, plus page numbers * * `full` - 'First', 'Previous', 'Next' and 'Last' buttons * * `full_numbers` - 'First', 'Previous', 'Next' and 'Last' buttons, plus page numbers * * `first_last_numbers` - 'First' and 'Last' buttons, plus page numbers */ "sPaginationType": "full_numbers", /** * Enable horizontal scrolling. When a table is too wide to fit into a * certain layout, or you have a large number of columns in the table, you * can enable x-scrolling to show the table in a viewport, which can be * scrolled. This property can be `true` which will allow the table to * scroll horizontally when needed, or any CSS unit, or a number (in which * case it will be treated as a pixel measurement). Setting as simply `true` * is recommended. */ "sScrollX": "", /** * This property can be used to force a DataTable to use more width than it * might otherwise do when x-scrolling is enabled. For example if you have a * table which requires to be well spaced, this parameter is useful for * "over-sizing" the table, and thus forcing scrolling. This property can by * any CSS unit, or a number (in which case it will be treated as a pixel * measurement). */ "sScrollXInner": "", /** * Enable vertical scrolling. Vertical scrolling will constrain the DataTable * to the given height, and enable scrolling for any data which overflows the * current viewport. This can be used as an alternative to paging to display * a lot of data in a small area (although paging and scrolling can both be * enabled at the same time). This property can be any CSS unit, or a number * (in which case it will be treated as a pixel measurement). */ "sScrollY": "", /** * __Deprecated__ The functionality provided by this parameter has now been * superseded by that provided through `ajax`, which should be used instead. * * Set the HTTP method that is used to make the Ajax call for server-side * processing or Ajax sourced data. */ "sServerMethod": "GET", /** * DataTables makes use of renderers when displaying HTML elements for * a table. These renderers can be added or modified by plug-ins to * generate suitable mark-up for a site. For example the Bootstrap * integration plug-in for DataTables uses a paging button renderer to * display pagination buttons in the mark-up required by Bootstrap. * * For further information about the renderers available see * DataTable.ext.renderer */ "renderer": null, /** * Set the data property name that DataTables should use to get a row's id * to set as the `id` property in the node. */ "rowId": "DT_RowId", /** * Caption value */ "caption": null }; _fnHungarianMap( DataTable.defaults ); /* * Developer note - See note in model.defaults.js about the use of Hungarian * notation and camel case. */ /** * Column options that can be given to DataTables at initialisation time. * @namespace */ DataTable.defaults.column = { /** * Define which column(s) an order will occur on for this column. This * allows a column's ordering to take multiple columns into account when * doing a sort or use the data from a different column. For example first * name / last name columns make sense to do a multi-column sort over the * two columns. */ "aDataSort": null, "iDataSort": -1, ariaTitle: '', /** * You can control the default ordering direction, and even alter the * behaviour of the sort handler (i.e. only allow ascending ordering etc) * using this parameter. */ "asSorting": [ 'asc', 'desc', '' ], /** * Enable or disable filtering on the data in this column. */ "bSearchable": true, /** * Enable or disable ordering on this column. */ "bSortable": true, /** * Enable or disable the display of this column. */ "bVisible": true, /** * Developer definable function that is called whenever a cell is created (Ajax source, * etc) or processed for input (DOM source). This can be used as a compliment to mRender * allowing you to modify the DOM element (add background colour for example) when the * element is available. */ "fnCreatedCell": null, /** * This property can be used to read data from any data source property, * including deeply nested objects / properties. `data` can be given in a * number of different ways which effect its behaviour: * * * `integer` - treated as an array index for the data source. This is the * default that DataTables uses (incrementally increased for each column). * * `string` - read an object property from the data source. There are * three 'special' options that can be used in the string to alter how * DataTables reads the data from the source object: * * `.` - Dotted Javascript notation. Just as you use a `.` in * Javascript to read from nested objects, so to can the options * specified in `data`. For example: `browser.version` or * `browser.name`. If your object parameter name contains a period, use * `\\` to escape it - i.e. `first\\.name`. * * `[]` - Array notation. DataTables can automatically combine data * from and array source, joining the data with the characters provided * between the two brackets. For example: `name[, ]` would provide a * comma-space separated list from the source array. If no characters * are provided between the brackets, the original array source is * returned. * * `()` - Function notation. Adding `()` to the end of a parameter will * execute a function of the name given. For example: `browser()` for a * simple function on the data source, `browser.version()` for a * function in a nested property or even `browser().version` to get an * object property if the function called returns an object. Note that * function notation is recommended for use in `render` rather than * `data` as it is much simpler to use as a renderer. * * `null` - use the original data source for the row rather than plucking * data directly from it. This action has effects on two other * initialisation options: * * `defaultContent` - When null is given as the `data` option and * `defaultContent` is specified for the column, the value defined by * `defaultContent` will be used for the cell. * * `render` - When null is used for the `data` option and the `render` * option is specified for the column, the whole data source for the * row is used for the renderer. * * `function` - the function given will be executed whenever DataTables * needs to set or get the data for a cell in the column. The function * takes three parameters: * * Parameters: * * `{array|object}` The data source for the row * * `{string}` The type call data requested - this will be 'set' when * setting data or 'filter', 'display', 'type', 'sort' or undefined * when gathering data. Note that when `undefined` is given for the * type DataTables expects to get the raw data for the object back< * * `{*}` Data to set when the second parameter is 'set'. * * Return: * * The return value from the function is not required when 'set' is * the type of call, but otherwise the return is what will be used * for the data requested. * * Note that `data` is a getter and setter option. If you just require * formatting of data for output, you will likely want to use `render` which * is simply a getter and thus simpler to use. * * Note that prior to DataTables 1.9.2 `data` was called `mDataProp`. The * name change reflects the flexibility of this property and is consistent * with the naming of mRender. If 'mDataProp' is given, then it will still * be used by DataTables, as it automatically maps the old name to the new * if required. */ "mData": null, /** * This property is the rendering partner to `data` and it is suggested that * when you want to manipulate data for display (including filtering, * sorting etc) without altering the underlying data for the table, use this * property. `render` can be considered to be the the read only companion to * `data` which is read / write (then as such more complex). Like `data` * this option can be given in a number of different ways to effect its * behaviour: * * * `integer` - treated as an array index for the data source. This is the * default that DataTables uses (incrementally increased for each column). * * `string` - read an object property from the data source. There are * three 'special' options that can be used in the string to alter how * DataTables reads the data from the source object: * * `.` - Dotted Javascript notation. Just as you use a `.` in * Javascript to read from nested objects, so to can the options * specified in `data`. For example: `browser.version` or * `browser.name`. If your object parameter name contains a period, use * `\\` to escape it - i.e. `first\\.name`. * * `[]` - Array notation. DataTables can automatically combine data * from and array source, joining the data with the characters provided * between the two brackets. For example: `name[, ]` would provide a * comma-space separated list from the source array. If no characters * are provided between the brackets, the original array source is * returned. * * `()` - Function notation. Adding `()` to the end of a parameter will * execute a function of the name given. For example: `browser()` for a * simple function on the data source, `browser.version()` for a * function in a nested property or even `browser().version` to get an * object property if the function called returns an object. * * `object` - use different data for the different data types requested by * DataTables ('filter', 'display', 'type' or 'sort'). The property names * of the object is the data type the property refers to and the value can * defined using an integer, string or function using the same rules as * `render` normally does. Note that an `_` option _must_ be specified. * This is the default value to use if you haven't specified a value for * the data type requested by DataTables. * * `function` - the function given will be executed whenever DataTables * needs to set or get the data for a cell in the column. The function * takes three parameters: * * Parameters: * * {array|object} The data source for the row (based on `data`) * * {string} The type call data requested - this will be 'filter', * 'display', 'type' or 'sort'. * * {array|object} The full data source for the row (not based on * `data`) * * Return: * * The return value from the function is what will be used for the * data requested. */ "mRender": null, /** * Change the cell type created for the column - either TD cells or TH cells. This * can be useful as TH cells have semantic meaning in the table body, allowing them * to act as a header for a row (you may wish to add scope='row' to the TH elements). */ "sCellType": "td", /** * Class to give to each cell in this column. */ "sClass": "", /** * When DataTables calculates the column widths to assign to each column, * it finds the longest string in each column and then constructs a * temporary table and reads the widths from that. The problem with this * is that "mmm" is much wider then "iiii", but the latter is a longer * string - thus the calculation can go wrong (doing it properly and putting * it into an DOM object and measuring that is horribly(!) slow). Thus as * a "work around" we provide this option. It will append its value to the * text that is found to be the longest string for the column - i.e. padding. * Generally you shouldn't need this! */ "sContentPadding": "", /** * Allows a default value to be given for a column's data, and will be used * whenever a null data source is encountered (this can be because `data` * is set to null, or because the data source itself is null). */ "sDefaultContent": null, /** * This parameter is only used in DataTables' server-side processing. It can * be exceptionally useful to know what columns are being displayed on the * client side, and to map these to database fields. When defined, the names * also allow DataTables to reorder information from the server if it comes * back in an unexpected order (i.e. if you switch your columns around on the * client-side, your server-side code does not also need updating). */ "sName": "", /** * Defines a data source type for the ordering which can be used to read * real-time information from the table (updating the internally cached * version) prior to ordering. This allows ordering to occur on user * editable elements such as form inputs. */ "sSortDataType": "std", /** * The title of this column. */ "sTitle": null, /** * The type allows you to specify how the data for this column will be * ordered. Four types (string, numeric, date and html (which will strip * HTML tags before ordering)) are currently available. Note that only date * formats understood by Javascript's Date() object will be accepted as type * date. For example: "Mar 26, 2008 5:03 PM". May take the values: 'string', * 'numeric', 'date' or 'html' (by default). Further types can be adding * through plug-ins. */ "sType": null, /** * Defining the width of the column, this parameter may take any CSS value * (3em, 20px etc). DataTables applies 'smart' widths to columns which have not * been given a specific width through this interface ensuring that the table * remains readable. */ "sWidth": null }; _fnHungarianMap( DataTable.defaults.column ); /** * DataTables settings object - this holds all the information needed for a * given table, including configuration, data and current application of the * table options. DataTables does not have a single instance for each DataTable * with the settings attached to that instance, but rather instances of the * DataTable "class" are created on-the-fly as needed (typically by a * $().dataTable() call) and the settings object is then applied to that * instance. * * Note that this object is related to {@link DataTable.defaults} but this * one is the internal data store for DataTables's cache of columns. It should * NOT be manipulated outside of DataTables. Any configuration should be done * through the initialisation options. */ DataTable.models.oSettings = { /** * Primary features of DataTables and their enablement state. */ "oFeatures": { /** * Flag to say if DataTables should automatically try to calculate the * optimum table and columns widths (true) or not (false). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bAutoWidth": null, /** * Delay the creation of TR and TD elements until they are actually * needed by a driven page draw. This can give a significant speed * increase for Ajax source and Javascript source data, but makes no * difference at all for DOM and server-side processing tables. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bDeferRender": null, /** * Enable filtering on the table or not. Note that if this is disabled * then there is no filtering at all on the table, including fnFilter. * To just remove the filtering input use sDom and remove the 'f' option. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bFilter": null, /** * Used only for compatiblity with DT1 * @deprecated */ "bInfo": true, /** * Used only for compatiblity with DT1 * @deprecated */ "bLengthChange": true, /** * Pagination enabled or not. Note that if this is disabled then length * changing must also be disabled. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bPaginate": null, /** * Processing indicator enable flag whenever DataTables is enacting a * user request - typically an Ajax request for server-side processing. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bProcessing": null, /** * Server-side processing enabled flag - when enabled DataTables will * get all data from the server for every draw - there is no filtering, * sorting or paging done on the client-side. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bServerSide": null, /** * Sorting enablement flag. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSort": null, /** * Multi-column sorting * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSortMulti": null, /** * Apply a class to the columns which are being sorted to provide a * visual highlight or not. This can slow things down when enabled since * there is a lot of DOM interaction. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSortClasses": null, /** * State saving enablement flag. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bStateSave": null }, /** * Scrolling settings for a table. */ "oScroll": { /** * When the table is shorter in height than sScrollY, collapse the * table container down to the height of the table (when true). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bCollapse": null, /** * Width of the scrollbar for the web-browser's platform. Calculated * during table initialisation. */ "iBarWidth": 0, /** * Viewport width for horizontal scrolling. Horizontal scrolling is * disabled if an empty string. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sX": null, /** * Width to expand the table to when using x-scrolling. Typically you * should not need to use this. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. * @deprecated */ "sXInner": null, /** * Viewport height for vertical scrolling. Vertical scrolling is disabled * if an empty string. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sY": null }, /** * Language information for the table. */ "oLanguage": { /** * Information callback function. See * {@link DataTable.defaults.fnInfoCallback} */ "fnInfoCallback": null }, /** * Browser support parameters */ "oBrowser": { /** * Determine if the vertical scrollbar is on the right or left of the * scrolling container - needed for rtl language layout, although not * all browsers move the scrollbar (Safari). */ "bScrollbarLeft": false, /** * Browser scrollbar width */ "barWidth": 0 }, "ajax": null, /** * Array referencing the nodes which are used for the features. The * parameters of this object match what is allowed by sDom - i.e. * <ul> * <li>'l' - Length changing</li> * <li>'f' - Filtering input</li> * <li>'t' - The table!</li> * <li>'i' - Information</li> * <li>'p' - Pagination</li> * <li>'r' - pRocessing</li> * </ul> */ "aanFeatures": [], /** * Store data information - see {@link DataTable.models.oRow} for detailed * information. */ "aoData": [], /** * Array of indexes which are in the current display (after filtering etc) */ "aiDisplay": [], /** * Array of indexes for display - no filtering */ "aiDisplayMaster": [], /** * Map of row ids to data indexes */ "aIds": {}, /** * Store information about each column that is in use */ "aoColumns": [], /** * Store information about the table's header */ "aoHeader": [], /** * Store information about the table's footer */ "aoFooter": [], /** * Store the applied global search information in case we want to force a * research or compare the old search to a new one. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "oPreviousSearch": {}, /** * Store for named searches */ searchFixed: {}, /** * Store the applied search for each column - see * {@link DataTable.models.oSearch} for the format that is used for the * filtering information for each column. */ "aoPreSearchCols": [], /** * Sorting that is applied to the table. Note that the inner arrays are * used in the following manner: * <ul> * <li>Index 0 - column number</li> * <li>Index 1 - current sorting direction</li> * </ul> * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "aaSorting": null, /** * Sorting that is always applied to the table (i.e. prefixed in front of * aaSorting). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "aaSortingFixed": [], /** * If restoring a table - we should restore its width */ "sDestroyWidth": 0, /** * Callback functions array for every time a row is inserted (i.e. on a draw). */ "aoRowCallback": [], /** * Callback functions for the header on each draw. */ "aoHeaderCallback": [], /** * Callback function for the footer on each draw. */ "aoFooterCallback": [], /** * Array of callback functions for draw callback functions */ "aoDrawCallback": [], /** * Array of callback functions for row created function */ "aoRowCreatedCallback": [], /** * Callback functions for just before the table is redrawn. A return of * false will be used to cancel the draw. */ "aoPreDrawCallback": [], /** * Callback functions for when the table has been initialised. */ "aoInitComplete": [], /** * Callbacks for modifying the settings to be stored for state saving, prior to * saving state. */ "aoStateSaveParams": [], /** * Callbacks for modifying the settings that have been stored for state saving * prior to using the stored values to restore the state. */ "aoStateLoadParams": [], /** * Callbacks for operating on the settings object once the saved state has been * loaded */ "aoStateLoaded": [], /** * Cache the table ID for quick access */ "sTableId": "", /** * The TABLE node for the main table */ "nTable": null, /** * Permanent ref to the thead element */ "nTHead": null, /** * Permanent ref to the tfoot element - if it exists */ "nTFoot": null, /** * Permanent ref to the tbody element */ "nTBody": null, /** * Cache the wrapper node (contains all DataTables controlled elements) */ "nTableWrapper": null, /** * Indicate if all required information has been read in */ "bInitialised": false, /** * Information about open rows. Each object in the array has the parameters * 'nTr' and 'nParent' */ "aoOpenRows": [], /** * Dictate the positioning of DataTables' control elements - see * {@link DataTable.model.oInit.sDom}. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sDom": null, /** * Search delay (in mS) */ "searchDelay": null, /** * Which type of pagination should be used. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sPaginationType": "two_button", /** * Number of paging controls on the page. Only used for backwards compatibility */ pagingControls: 0, /** * The state duration (for `stateSave`) in seconds. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "iStateDuration": 0, /** * Array of callback functions for state saving. Each array element is an * object with the following parameters: * <ul> * <li>function:fn - function to call. Takes two parameters, oSettings * and the JSON string to save that has been thus far created. Returns * a JSON string to be inserted into a json object * (i.e. '"param": [ 0, 1, 2]')</li> * <li>string:sName - name of callback</li> * </ul> */ "aoStateSave": [], /** * Array of callback functions for state loading. Each array element is an * object with the following parameters: * <ul> * <li>function:fn - function to call. Takes two parameters, oSettings * and the object stored. May return false to cancel state loading</li> * <li>string:sName - name of callback</li> * </ul> */ "aoStateLoad": [], /** * State that was saved. Useful for back reference */ "oSavedState": null, /** * State that was loaded. Useful for back reference */ "oLoadedState": null, /** * Note if draw should be blocked while getting data */ "bAjaxDataGet": true, /** * The last jQuery XHR object that was used for server-side data gathering. * This can be used for working with the XHR information in one of the * callbacks */ "jqXHR": null, /** * JSON returned from the server in the last Ajax request */ "json": undefined, /** * Data submitted as part of the last Ajax request */ "oAjaxData": undefined, /** * Send the XHR HTTP method - GET or POST (could be PUT or DELETE if * required). * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "sServerMethod": null, /** * Format numbers for display. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "fnFormatNumber": null, /** * List of options that can be used for the user selectable length menu. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "aLengthMenu": null, /** * Counter for the draws that the table does. Also used as a tracker for * server-side processing */ "iDraw": 0, /** * Indicate if a redraw is being done - useful for Ajax */ "bDrawing": false, /** * Draw index (iDraw) of the last error when parsing the returned data */ "iDrawError": -1, /** * Paging display length */ "_iDisplayLength": 10, /** * Paging start point - aiDisplay index */ "_iDisplayStart": 0, /** * Server-side processing - number of records in the result set * (i.e. before filtering), Use fnRecordsTotal rather than * this property to get the value of the number of records, regardless of * the server-side processing setting. */ "_iRecordsTotal": 0, /** * Server-side processing - number of records in the current display set * (i.e. after filtering). Use fnRecordsDisplay rather than * this property to get the value of the number of records, regardless of * the server-side processing setting. */ "_iRecordsDisplay": 0, /** * The classes to use for the table */ "oClasses": {}, /** * Flag attached to the settings object so you can check in the draw * callback if filtering has been done in the draw. Deprecated in favour of * events. * @deprecated */ "bFiltered": false, /** * Flag attached to the settings object so you can check in the draw * callback if sorting has been done in the draw. Deprecated in favour of * events. * @deprecated */ "bSorted": false, /** * Indicate that if multiple rows are in the header and there is more than * one unique cell per column, if the top one (true) or bottom one (false) * should be used for sorting / title by DataTables. * Note that this parameter will be set by the initialisation routine. To * set a default use {@link DataTable.defaults}. */ "bSortCellsTop": null, /** * Initialisation object that is used for the table */ "oInit": null, /** * Destroy callback functions - for plug-ins to attach themselves to the * destroy so they can clean up markup and events. */ "aoDestroyCallback": [], /** * Get the number of records in the current record set, before filtering */ "fnRecordsTotal": function () { return _fnDataSource( this ) == 'ssp' ? this._iRecordsTotal * 1 : this.aiDisplayMaster.length; }, /** * Get the number of records in the current record set, after filtering */ "fnRecordsDisplay": function () { return _fnDataSource( this ) == 'ssp' ? this._iRecordsDisplay * 1 : this.aiDisplay.length; }, /** * Get the display end point - aiDisplay index */ "fnDisplayEnd": function () { var len = this._iDisplayLength, start = this._iDisplayStart, calc = start + len, records = this.aiDisplay.length, features = this.oFeatures, paginate = features.bPaginate; if ( features.bServerSide ) { return paginate === false || len === -1 ? start + records : Math.min( start+len, this._iRecordsDisplay ); } else { return ! paginate || calc>records || len===-1 ? records : calc; } }, /** * The DataTables object for this table */ "oInstance": null, /** * Unique identifier for each instance of the DataTables object. If there * is an ID on the table node, then it takes that value, otherwise an * incrementing internal counter is used. */ "sInstance": null, /** * tabindex attribute value that is added to DataTables control elements, allowing * keyboard navigation of the table and its controls. */ "iTabIndex": 0, /** * DIV container for the footer scrolling table if scrolling */ "nScrollHead": null, /** * DIV container for the footer scrolling table if scrolling */ "nScrollFoot": null, /** * Last applied sort */ "aLastSort": [], /** * Stored plug-in instances */ "oPlugins": {}, /** * Function used to get a row's id from the row's data */ "rowIdFn": null, /** * Data location where to store a row's id */ "rowId": null, caption: '', captionNode: null, colgroup: null }; /** * Extension object for DataTables that is used to provide all extension * options. * * Note that the `DataTable.ext` object is available through * `jQuery.fn.dataTable.ext` where it may be accessed and manipulated. It is * also aliased to `jQuery.fn.dataTableExt` for historic reasons. * @namespace * @extends DataTable.models.ext */ var extPagination = DataTable.ext.pager; // Paging buttons configuration $.extend( extPagination, { simple: function () { return [ 'previous', 'next' ]; }, full: function () { return [ 'first', 'previous', 'next', 'last' ]; }, numbers: function () { return [ 'numbers' ]; }, simple_numbers: function () { return [ 'previous', 'numbers', 'next' ]; }, full_numbers: function () { return [ 'first', 'previous', 'numbers', 'next', 'last' ]; }, first_last: function () { return ['first', 'last']; }, first_last_numbers: function () { return ['first', 'numbers', 'last']; }, // For testing and plug-ins to use _numbers: _pagingNumbers, // Number of number buttons - legacy, use `numbers` option for paging feature numbers_length: 7 } ); $.extend( true, DataTable.ext.renderer, { pagingButton: { _: function (settings, buttonType, content, active, disabled) { var classes = settings.oClasses.paging; var btnClasses = [classes.button]; var btn; if (active) { btnClasses.push(classes.active); } if (disabled) { btnClasses.push(classes.disabled) } if (buttonType === 'ellipsis') { btn = $('<span class="ellipsis"></span>').html(content)[0]; } else { btn = $('<button>', { class: btnClasses.join(' '), role: 'link', type: 'button' }).html(content); } return { display: btn, clicker: btn } } }, pagingContainer: { _: function (settings, buttons) { // No wrapping element - just append directly to the host return buttons; } } } ); // Common function to remove new lines, strip HTML and diacritic control var _filterString = function (stripHtml, normalize) { return function (str) { if (_empty(str) || typeof str !== 'string') { return str; } str = str.replace( _re_new_lines, " " ); if (stripHtml) { str = _stripHtml(str); } if (normalize) { str = _normalize(str, false); } return str; }; } /* * Public helper functions. These aren't used internally by DataTables, or * called by any of the options passed into DataTables, but they can be used * externally by developers working with DataTables. They are helper functions * to make working with DataTables a little bit easier. */ function __mldFnName(name) { return name.replace(/[\W]/g, '_') } // Common logic for moment, luxon or a date action function __mld( dt, momentFn, luxonFn, dateFn, arg1 ) { if (window.moment) { return dt[momentFn]( arg1 ); } else if (window.luxon) { return dt[luxonFn]( arg1 ); } return dateFn ? dt[dateFn]( arg1 ) : dt; } var __mlWarning = false; function __mldObj (d, format, locale) { var dt; if (window.moment) { dt = window.moment.utc( d, format, locale, true ); if (! dt.isValid()) { return null; } } else if (window.luxon) { dt = format && typeof d === 'string' ? window.luxon.DateTime.fromFormat( d, format ) : window.luxon.DateTime.fromISO( d ); if (! dt.isValid) { return null; } dt.setLocale(locale); } else if (! format) { // No format given, must be ISO dt = new Date(d); } else { if (! __mlWarning) { alert('DataTables warning: Formatted date without Moment.js or Luxon - https://datatables.net/tn/17'); } __mlWarning = true; } return dt; } // Wrapper for date, datetime and time which all operate the same way with the exception of // the output string for auto locale support function __mlHelper (localeString) { return function ( from, to, locale, def ) { // Luxon and Moment support // Argument shifting if ( arguments.length === 0 ) { locale = 'en'; to = null; // means toLocaleString from = null; // means iso8601 } else if ( arguments.length === 1 ) { locale = 'en'; to = from; from = null; } else if ( arguments.length === 2 ) { locale = to; to = from; from = null; } var typeName = 'datetime' + (to ? '-' + __mldFnName(to) : ''); // Add type detection and sorting specific to this date format - we need to be able to identify // date type columns as such, rather than as numbers in extensions. Hence the need for this. if (! DataTable.ext.type.order[typeName]) { DataTable.type(typeName, { detect: function (d) { // The renderer will give the value to type detect as the type! return d === typeName ? typeName : false; }, order: { pre: function (d) { // The renderer gives us Moment, Luxon or Date obects for the sorting, all of which have a // `valueOf` which gives milliseconds epoch return d.valueOf(); } }, className: 'dt-right' }); } return function ( d, type ) { // Allow for a default value if (d === null || d === undefined) { if (def === '--now') { // We treat everything as UTC further down, so no changes are // made, as such need to get the local date / time as if it were // UTC var local = new Date(); d = new Date( Date.UTC( local.getFullYear(), local.getMonth(), local.getDate(), local.getHours(), local.getMinutes(), local.getSeconds() ) ); } else { d = ''; } } if (type === 'type') { // Typing uses the type name for fast matching return typeName; } if (d === '') { return type !== 'sort' ? '' : __mldObj('0000-01-01 00:00:00', null, locale); } // Shortcut. If `from` and `to` are the same, we are using the renderer to // format for ordering, not display - its already in the display format. if ( to !== null && from === to && type !== 'sort' && type !== 'type' && ! (d instanceof Date) ) { return d; } var dt = __mldObj(d, from, locale); if (dt === null) { return d; } if (type === 'sort') { return dt; } var formatted = to === null ? __mld(dt, 'toDate', 'toJSDate', '')[localeString]() : __mld(dt, 'format', 'toFormat', 'toISOString', to); // XSS protection return type === 'display' ? _escapeHtml( formatted ) : formatted; }; } } // Based on locale, determine standard number formatting // Fallback for legacy browsers is US English var __thousands = ','; var __decimal = '.'; if (window.Intl !== undefined) { try { var num = new Intl.NumberFormat().formatToParts(100000.1); for (var i=0 ; i<num.length ; i++) { if (num[i].type === 'group') { __thousands = num[i].value; } else if (num[i].type === 'decimal') { __decimal = num[i].value; } } } catch (e) { // noop } } // Formatted date time detection - use by declaring the formats you are going to use DataTable.datetime = function ( format, locale ) { var typeName = 'datetime-detect-' + __mldFnName(format); if (! locale) { locale = 'en'; } if (! DataTable.ext.type.order[typeName]) { DataTable.type(typeName, { detect: function (d) { var dt = __mldObj(d, format, locale); return d === '' || dt ? typeName : false; }, order: { pre: function (d) { return __mldObj(d, format, locale) || 0; } }, className: 'dt-right' }); } } /** * Helpers for `columns.render`. * * The options defined here can be used with the `columns.render` initialisation * option to provide a display renderer. The following functions are defined: * * * `moment` - Uses the MomentJS library to convert from a given format into another. * This renderer has three overloads: * * 1 parameter: * * `string` - Format to convert to (assumes input is ISO8601 and locale is `en`) * * 2 parameters: * * `string` - Format to convert from * * `string` - Format to convert to. Assumes `en` locale * * 3 parameters: * * `string` - Format to convert from * * `string` - Format to convert to * * `string` - Locale * * `number` - Will format numeric data (defined by `columns.data`) for * display, retaining the original unformatted data for sorting and filtering. * It takes 5 parameters: * * `string` - Thousands grouping separator * * `string` - Decimal point indicator * * `integer` - Number of decimal points to show * * `string` (optional) - Prefix. * * `string` (optional) - Postfix (/suffix). * * `text` - Escape HTML to help prevent XSS attacks. It has no optional * parameters. * * @example * // Column definition using the number renderer * { * data: "salary", * render: $.fn.dataTable.render.number( '\'', '.', 0, '$' ) * } * * @namespace */ DataTable.render = { date: __mlHelper('toLocaleDateString'), datetime: __mlHelper('toLocaleString'), time: __mlHelper('toLocaleTimeString'), number: function ( thousands, decimal, precision, prefix, postfix ) { // Auto locale detection if (thousands === null || thousands === undefined) { thousands = __thousands; } if (decimal === null || decimal === undefined) { decimal = __decimal; } return { display: function ( d ) { if ( typeof d !== 'number' && typeof d !== 'string' ) { return d; } if (d === '' || d === null) { return d; } var negative = d < 0 ? '-' : ''; var flo = parseFloat( d ); var abs = Math.abs(flo); // Scientific notation for large and small numbers if (abs >= 100000000000 || (abs < 0.0001 && abs !== 0) ) { var exp = flo.toExponential(precision).split(/e\+?/); return exp[0] + ' x 10<sup>' + exp[1] + '</sup>'; } // If NaN then there isn't much formatting that we can do - just // return immediately, escaping any HTML (this was supposed to // be a number after all) if ( isNaN( flo ) ) { return _escapeHtml( d ); } flo = flo.toFixed( precision ); d = Math.abs( flo ); var intPart = parseInt( d, 10 ); var floatPart = precision ? decimal+(d - intPart).toFixed( precision ).substring( 2 ): ''; // If zero, then can't have a negative prefix if (intPart === 0 && parseFloat(floatPart) === 0) { negative = ''; } return negative + (prefix||'') + intPart.toString().replace( /\B(?=(\d{3})+(?!\d))/g, thousands ) + floatPart + (postfix||''); } }; }, text: function () { return { display: _escapeHtml, filter: _escapeHtml }; } }; var _extTypes = DataTable.ext.type; // Get / set type DataTable.type = function (name, prop, val) { if (! prop) { return { className: _extTypes.className[name], detect: _extTypes.detect.find(function (fn) { return fn.name === name; }), order: { pre: _extTypes.order[name + '-pre'], asc: _extTypes.order[name + '-asc'], desc: _extTypes.order[name + '-desc'] }, render: _extTypes.render[name], search: _extTypes.search[name] }; } var setProp = function(prop, propVal) { _extTypes[prop][name] = propVal; }; var setDetect = function (fn) { // Wrap to allow the function to return `true` rather than // specifying the type name. var cb = function (d, s) { var ret = fn(d, s); return ret === true ? name : ret; }; Object.defineProperty(cb, "name", {value: name}); var idx = _extTypes.detect.findIndex(function (fn) { return fn.name === name; }); if (idx === -1) { _extTypes.detect.unshift(cb); } else { _extTypes.detect.splice(idx, 1, cb); } }; var setOrder = function (obj) { _extTypes.order[name + '-pre'] = obj.pre; // can be undefined _extTypes.order[name + '-asc'] = obj.asc; // can be undefined _extTypes.order[name + '-desc'] = obj.desc; // can be undefined }; // prop is optional if (val === undefined) { val = prop; prop = null; } if (prop === 'className') { setProp('className', val); } else if (prop === 'detect') { setDetect(val); } else if (prop === 'order') { setOrder(val); } else if (prop === 'render') { setProp('render', val); } else if (prop === 'search') { setProp('search', val); } else if (! prop) { if (val.className) { setProp('className', val.className); } if (val.detect !== undefined) { setDetect(val.detect); } if (val.order) { setOrder(val.order); } if (val.render !== undefined) { setProp('render', val.render); } if (val.search !== undefined) { setProp('search', val.search); } } } // Get a list of types DataTable.types = function () { return _extTypes.detect.map(function (fn) { return fn.name; }); }; // // Built in data types // DataTable.type('string', { detect: function () { return 'string'; }, order: { pre: function ( a ) { // This is a little complex, but faster than always calling toString, // http://jsperf.com/tostring-v-check return _empty(a) ? '' : typeof a === 'string' ? a.toLowerCase() : ! a.toString ? '' : a.toString(); } }, search: _filterString(false, true) }); DataTable.type('html', { detect: function ( d ) { return _empty( d ) || (typeof d === 'string' && d.indexOf('<') !== -1) ? 'html' : null; }, order: { pre: function ( a ) { return _empty(a) ? '' : a.replace ? _stripHtml(a).trim().toLowerCase() : a+''; } }, search: _filterString(true, true) }); DataTable.type('date', { className: 'dt-type-date', detect: function ( d ) { // V8 tries _very_ hard to make a string passed into `Date.parse()` // valid, so we need to use a regex to restrict date formats. Use a // plug-in for anything other than ISO8601 style strings if ( d && !(d instanceof Date) && ! _re_date.test(d) ) { return null; } var parsed = Date.parse(d); return (parsed !== null && !isNaN(parsed)) || _empty(d) ? 'date' : null; }, order: { pre: function ( d ) { var ts = Date.parse( d ); return isNaN(ts) ? -Infinity : ts; } } }); DataTable.type('html-num-fmt', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _htmlNumeric( d, decimal, true ) ? 'html-num-fmt' : null; }, order: { pre: function ( d, s ) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp, _re_html, _re_formatted_numeric ); } }, search: _filterString(true, true) }); DataTable.type('html-num', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _htmlNumeric( d, decimal ) ? 'html-num' : null; }, order: { pre: function ( d, s ) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp, _re_html ); } }, search: _filterString(true, true) }); DataTable.type('num-fmt', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _isNumber( d, decimal, true ) ? 'num-fmt' : null; }, order: { pre: function ( d, s ) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp, _re_formatted_numeric ); } } }); DataTable.type('num', { className: 'dt-type-numeric', detect: function ( d, settings ) { var decimal = settings.oLanguage.sDecimal; return _isNumber( d, decimal ) ? 'num' : null; }, order: { pre: function (d, s) { var dp = s.oLanguage.sDecimal; return __numericReplace( d, dp ); } } }); var __numericReplace = function ( d, decimalPlace, re1, re2 ) { if ( d !== 0 && (!d || d === '-') ) { return -Infinity; } var type = typeof d; if (type === 'number' || type === 'bigint') { return d; } // If a decimal place other than `.` is used, it needs to be given to the // function so we can detect it and replace with a `.` which is the only // decimal place Javascript recognises - it is not locale aware. if ( decimalPlace ) { d = _numToDecimal( d, decimalPlace ); } if ( d.replace ) { if ( re1 ) { d = d.replace( re1, '' ); } if ( re2 ) { d = d.replace( re2, '' ); } } return d * 1; }; $.extend( true, DataTable.ext.renderer, { footer: { _: function ( settings, cell, classes ) { cell.addClass(classes.tfoot.cell); } }, header: { _: function ( settings, cell, classes ) { cell.addClass(classes.thead.cell); if (! settings.oFeatures.bSort) { cell.addClass(classes.order.none); } var legacyTop = settings.bSortCellsTop; var headerRows = cell.closest('thead').find('tr'); var rowIdx = cell.parent().index(); // Conditions to not apply the ordering icons if ( // Cells and rows which have the attribute to disable the icons cell.attr('data-dt-order') === 'disable' || cell.parent().attr('data-dt-order') === 'disable' || // Legacy support for `orderCellsTop`. If it is set, then cells // which are not in the top or bottom row of the header (depending // on the value) do not get the sorting classes applied to them (legacyTop === true && rowIdx !== 0) || (legacyTop === false && rowIdx !== headerRows.length - 1) ) { return; } // No additional mark-up required // Attach a sort listener to update on sort - note that using the // `DT` namespace will allow the event to be removed automatically // on destroy, while the `dt` namespaced event is the one we are // listening for $(settings.nTable).on( 'order.dt.DT', function ( e, ctx, sorting ) { if ( settings !== ctx ) { // need to check this this is the host return; // table, not a nested one } var orderClasses = classes.order; var columns = ctx.api.columns( cell ); var col = settings.aoColumns[columns.flatten()[0]]; var orderable = columns.orderable().includes(true); var ariaType = ''; var indexes = columns.indexes(); var sortDirs = columns.orderable(true).flatten(); var orderedColumns = ',' + sorting.map( function (val) { return val.col; } ).join(',') + ','; cell .removeClass( orderClasses.isAsc +' '+ orderClasses.isDesc ) .toggleClass( orderClasses.none, ! orderable ) .toggleClass( orderClasses.canAsc, orderable && sortDirs.includes('asc') ) .toggleClass( orderClasses.canDesc, orderable && sortDirs.includes('desc') ); var sortIdx = orderedColumns.indexOf( ',' + indexes.toArray().join(',') + ',' ); if ( sortIdx !== -1 ) { // Get the ordering direction for the columns under this cell // Note that it is possible for a cell to be asc and desc sorting // (column spanning cells) var orderDirs = columns.order(); cell.addClass( orderDirs.includes('asc') ? orderClasses.isAsc : '' + orderDirs.includes('desc') ? orderClasses.isDesc : '' ); } // The ARIA spec says that only one column should be marked with aria-sort if ( sortIdx === 0 ) { var firstSort = sorting[0]; var sortOrder = col.asSorting; cell.attr('aria-sort', firstSort.dir === 'asc' ? 'ascending' : 'descending'); // Determine if the next click will remove sorting or change the sort ariaType = ! sortOrder[firstSort.index + 1] ? 'Remove' : 'Reverse'; } else { cell.removeAttr('aria-sort'); } cell.attr('aria-label', orderable ? col.ariaTitle + ctx.api.i18n('oAria.orderable' + ariaType) : col.ariaTitle ); if (orderable) { cell.find('.dt-column-title').attr('role', 'button'); cell.attr('tabindex', 0) } } ); } }, layout: { _: function ( settings, container, items ) { var row = $('<div/>') .addClass('dt-layout-row') .appendTo( container ); $.each( items, function (key, val) { var klass = ! val.table ? 'dt-'+key+' ' : ''; if (val.table) { row.addClass('dt-layout-table'); } $('<div/>') .attr({ id: val.id || null, "class": 'dt-layout-cell '+klass+(val.className || '') }) .append( val.contents ) .appendTo( row ); } ); } } } ); DataTable.feature = {}; // Third parameter is internal only! DataTable.feature.register = function ( name, cb, legacy ) { DataTable.ext.features[ name ] = cb; if (legacy) { _ext.feature.push({ cFeature: legacy, fnInit: cb }); } }; DataTable.feature.register( 'info', function ( settings, opts ) { // For compatibility with the legacy `info` top level option if (! settings.oFeatures.bInfo) { return null; } var lang = settings.oLanguage, tid = settings.sTableId, n = $('<div/>', { 'class': settings.oClasses.info.container, } ); opts = $.extend({ callback: lang.fnInfoCallback, empty: lang.sInfoEmpty, postfix: lang.sInfoPostFix, search: lang.sInfoFiltered, text: lang.sInfo, }, opts); // Update display on each draw settings.aoDrawCallback.push(function (s) { _fnUpdateInfo(s, opts, n); }); // For the first info display in the table, we add a callback and aria information. if (! settings._infoEl) { n.attr({ 'aria-live': 'polite', id: tid+'_info', role: 'status' }); // Table is described by our info div $(settings.nTable).attr( 'aria-describedby', tid+'_info' ); settings._infoEl = n; } return n; }, 'i' ); /** * Update the information elements in the display * @param {object} settings dataTables settings object * @memberof DataTable#oApi */ function _fnUpdateInfo ( settings, opts, node ) { var start = settings._iDisplayStart+1, end = settings.fnDisplayEnd(), max = settings.fnRecordsTotal(), total = settings.fnRecordsDisplay(), out = total ? opts.text : opts.empty; if ( total !== max ) { // Record set after filtering out += ' ' + opts.search; } // Convert the macros out += opts.postfix; out = _fnMacros( settings, out ); if ( opts.callback ) { out = opts.callback.call( settings.oInstance, settings, start, end, max, total, out ); } node.html( out ); _fnCallbackFire(settings, null, 'info', [settings, node[0], out]); } var __searchCounter = 0; // opts // - text // - placeholder DataTable.feature.register( 'search', function ( settings, opts ) { // Don't show the input if filtering isn't available on the table if (! settings.oFeatures.bFilter) { return null; } var classes = settings.oClasses.search; var tableId = settings.sTableId; var language = settings.oLanguage; var previousSearch = settings.oPreviousSearch; var input = '<input type="search" class="'+classes.input+'"/>'; opts = $.extend({ placeholder: language.sSearchPlaceholder, text: language.sSearch }, opts); // The _INPUT_ is optional - is appended if not present if (opts.text.indexOf('_INPUT_') === -1) { opts.text += '_INPUT_'; } opts.text = _fnMacros(settings, opts.text); // We can put the <input> outside of the label if it is at the start or end // which helps improve accessability (not all screen readers like implicit // for elements). var end = opts.text.match(/_INPUT_$/); var start = opts.text.match(/^_INPUT_/); var removed = opts.text.replace(/_INPUT_/, ''); var str = '<label>' + opts.text + '</label>'; if (start) { str = '_INPUT_<label>' + removed + '</label>'; } else if (end) { str = '<label>' + removed + '</label>_INPUT_'; } var filter = $('<div>') .addClass(classes.container) .append(str.replace(/_INPUT_/, input)); // add for and id to label and input filter.find('label').attr('for', 'dt-search-' + __searchCounter); filter.find('input').attr('id', 'dt-search-' + __searchCounter); __searchCounter++; var searchFn = function(event) { var val = this.value; if(previousSearch.return && event.key !== "Enter") { return; } /* Now do the filter */ if ( val != previousSearch.search ) { previousSearch.search = val; _fnFilterComplete( settings, previousSearch ); // Need to redraw, without resorting settings._iDisplayStart = 0; _fnDraw( settings ); } }; var searchDelay = settings.searchDelay !== null ? settings.searchDelay : 0; var jqFilter = $('input', filter) .val( previousSearch.search ) .attr( 'placeholder', opts.placeholder ) .on( 'keyup.DT search.DT input.DT paste.DT cut.DT', searchDelay ? DataTable.util.debounce( searchFn, searchDelay ) : searchFn ) .on( 'mouseup.DT', function(e) { // Edge fix! Edge 17 does not trigger anything other than mouse events when clicking // on the clear icon (Edge bug 17584515). This is safe in other browsers as `searchFn` // checks the value to see if it has changed. In other browsers it won't have. setTimeout( function () { searchFn.call(jqFilter[0], e); }, 10); } ) .on( 'keypress.DT', function(e) { /* Prevent form submission */ if ( e.keyCode == 13 ) { return false; } } ) .attr('aria-controls', tableId); // Update the input elements whenever the table is filtered $(settings.nTable).on( 'search.dt.DT', function ( ev, s ) { if ( settings === s && jqFilter[0] !== document.activeElement ) { jqFilter.val( typeof previousSearch.search !== 'function' ? previousSearch.search : '' ); } } ); return filter; }, 'f' ); // opts // - type - button configuration // - buttons - number of buttons to show - must be odd DataTable.feature.register( 'paging', function ( settings, opts ) { // Don't show the paging input if the table doesn't have paging enabled if (! settings.oFeatures.bPaginate) { return null; } opts = $.extend({ buttons: DataTable.ext.pager.numbers_length, type: settings.sPaginationType, boundaryNumbers: true }, opts); // To be removed in 2.1 if (opts.numbers) { opts.buttons = opts.numbers; } var host = $('<div/>').addClass( settings.oClasses.paging.container + ' paging_' + opts.type ); var draw = function () { _pagingDraw(settings, host, opts); }; settings.aoDrawCallback.push(draw); // Responsive redraw of paging control $(settings.nTable).on('column-sizing.dt.DT', draw); return host; }, 'p' ); function _pagingDraw(settings, host, opts) { if (! settings._bInitComplete) { return; } var plugin = DataTable.ext.pager[ opts.type ], aria = settings.oLanguage.oAria.paginate || {}, start = settings._iDisplayStart, len = settings._iDisplayLength, visRecords = settings.fnRecordsDisplay(), all = len === -1, page = all ? 0 : Math.ceil( start / len ), pages = all ? 1 : Math.ceil( visRecords / len ), buttons = plugin() .map(function (val) { return val === 'numbers' ? _pagingNumbers(page, pages, opts.buttons, opts.boundaryNumbers) : val; }) .flat(); var buttonEls = []; for (var i=0 ; i<buttons.length ; i++) { var button = buttons[i]; var btnInfo = _pagingButtonInfo(settings, button, page, pages); var btn = _fnRenderer( settings, 'pagingButton' )( settings, button, btnInfo.display, btnInfo.active, btnInfo.disabled ); // Common attributes $(btn.clicker).attr({ 'aria-controls': settings.sTableId, 'aria-disabled': btnInfo.disabled ? 'true' : null, 'aria-current': btnInfo.active ? 'page' : null, 'aria-label': aria[ button ], 'data-dt-idx': button, 'tabIndex': btnInfo.disabled ? -1 : settings.iTabIndex, }); if (typeof button !== 'number') { $(btn.clicker).addClass(button); } _fnBindAction( btn.clicker, {action: button}, function(e) { e.preventDefault(); _fnPageChange( settings, e.data.action, true ); } ); buttonEls.push(btn.display); } var wrapped = _fnRenderer(settings, 'pagingContainer')( settings, buttonEls ); var activeEl = host.find(document.activeElement).data('dt-idx'); host.empty().append(wrapped); if ( activeEl !== undefined ) { host.find( '[data-dt-idx='+activeEl+']' ).trigger('focus'); } // Responsive - check if the buttons are over two lines based on the // height of the buttons and the container. if ( buttonEls.length && // any buttons opts.numbers > 1 && // prevent infinite $(host).height() >= ($(buttonEls[0]).outerHeight() * 2) - 10 ) { _pagingDraw(settings, host, $.extend({}, opts, { numbers: opts.numbers - 2 })); } } /** * Get properties for a button based on the current paging state of the table * * @param {*} settings DT settings object * @param {*} button The button type in question * @param {*} page Table's current page * @param {*} pages Number of pages * @returns Info object */ function _pagingButtonInfo(settings, button, page, pages) { var lang = settings.oLanguage.oPaginate; var o = { display: '', active: false, disabled: false }; switch ( button ) { case 'ellipsis': o.display = '…'; o.disabled = true; break; case 'first': o.display = lang.sFirst; if (page === 0) { o.disabled = true; } break; case 'previous': o.display = lang.sPrevious; if ( page === 0 ) { o.disabled = true; } break; case 'next': o.display = lang.sNext; if ( pages === 0 || page === pages-1 ) { o.disabled = true; } break; case 'last': o.display = lang.sLast; if ( pages === 0 || page === pages-1 ) { o.disabled = true; } break; default: if ( typeof button === 'number' ) { o.display = settings.fnFormatNumber( button + 1 ); if (page === button) { o.active = true; } } break; } return o; } /** * Compute what number buttons to show in the paging control * * @param {*} page Current page * @param {*} pages Total number of pages * @param {*} buttons Target number of number buttons * @param {boolean} addFirstLast Indicate if page 1 and end should be included * @returns Buttons to show */ function _pagingNumbers ( page, pages, buttons, addFirstLast ) { var numbers = [], half = Math.floor(buttons / 2), before = addFirstLast ? 2 : 1, after = addFirstLast ? 1 : 0; if ( pages <= buttons ) { numbers = _range(0, pages); } else if (buttons === 1) { // Single button - current page only numbers = [page]; } else if (buttons === 3) { // Special logic for just three buttons if (page <= 1) { numbers = [0, 1, 'ellipsis']; } else if (page >= pages - 2) { numbers = _range(pages-2, pages); numbers.unshift('ellipsis'); } else { numbers = ['ellipsis', page, 'ellipsis']; } } else if ( page <= half ) { numbers = _range(0, buttons-before); numbers.push('ellipsis'); if (addFirstLast) { numbers.push(pages-1); } } else if ( page >= pages - 1 - half ) { numbers = _range(pages-(buttons-before), pages); numbers.unshift('ellipsis'); if (addFirstLast) { numbers.unshift(0); } } else { numbers = _range(page-half+before, page+half-after); numbers.push('ellipsis'); numbers.unshift('ellipsis'); if (addFirstLast) { numbers.push(pages-1); numbers.unshift(0); } } return numbers; } var __lengthCounter = 0; // opts // - menu // - text DataTable.feature.register( 'pageLength', function ( settings, opts ) { var features = settings.oFeatures; // For compatibility with the legacy `pageLength` top level option if (! features.bPaginate || ! features.bLengthChange) { return null; } opts = $.extend({ menu: settings.aLengthMenu, text: settings.oLanguage.sLengthMenu }, opts); var classes = settings.oClasses.length, tableId = settings.sTableId, menu = opts.menu, lengths = [], language = [], i; // Options can be given in a number of ways if (Array.isArray( menu[0] )) { // Old 1.x style - 2D array lengths = menu[0]; language = menu[1]; } else { for ( i=0 ; i<menu.length ; i++ ) { // An object with different label and value if ($.isPlainObject(menu[i])) { lengths.push(menu[i].value); language.push(menu[i].label); } else { // Or just a number to display and use lengths.push(menu[i]); language.push(menu[i]); } } } // We can put the <select> outside of the label if it is at the start or // end which helps improve accessability (not all screen readers like // implicit for elements). var end = opts.text.match(/_MENU_$/); var start = opts.text.match(/^_MENU_/); var removed = opts.text.replace(/_MENU_/, ''); var str = '<label>' + opts.text + '</label>'; if (start) { str = '_MENU_<label>' + removed + '</label>'; } else if (end) { str = '<label>' + removed + '</label>_MENU_'; } // Wrapper element - use a span as a holder for where the select will go var div = $('<div/>') .addClass( classes.container ) .append( str.replace( '_MENU_', '<span></span>' ) ); // Save text node content for macro updating var textNodes = []; div.find('label')[0].childNodes.forEach(function (el) { if (el.nodeType === Node.TEXT_NODE) { textNodes.push({ el: el, text: el.textContent }); } }) // Update the label text in case it has an entries value var updateEntries = function (len) { textNodes.forEach(function (node) { node.el.textContent = _fnMacros(settings, node.text, len); }); } // Next, the select itself, along with the options var select = $('<select/>', { 'name': tableId+'_length', 'aria-controls': tableId, 'class': classes.select } ); for ( i=0 ; i<lengths.length ; i++ ) { select[0][ i ] = new Option( typeof language[i] === 'number' ? settings.fnFormatNumber( language[i] ) : language[i], lengths[i] ); } // add for and id to label and input div.find('label').attr('for', 'dt-length-' + __lengthCounter); select.attr('id', 'dt-length-' + __lengthCounter); __lengthCounter++; // Swap in the select list div.find('span').replaceWith(select); // Can't use `select` variable as user might provide their own and the // reference is broken by the use of outerHTML $('select', div) .val( settings._iDisplayLength ) .on( 'change.DT', function() { _fnLengthChange( settings, $(this).val() ); _fnDraw( settings ); } ); // Update node value whenever anything changes the table's length $(settings.nTable).on( 'length.dt.DT', function (e, s, len) { if ( settings === s ) { $('select', div).val( len ); // Resolve plurals in the text for the new length updateEntries(len); } } ); updateEntries(settings._iDisplayLength); return div; }, 'l' ); // jQuery access $.fn.dataTable = DataTable; // Provide access to the host jQuery object (circular reference) DataTable.$ = $; // Legacy aliases $.fn.dataTableSettings = DataTable.settings; $.fn.dataTableExt = DataTable.ext; // With a capital `D` we return a DataTables API instance rather than a // jQuery object $.fn.DataTable = function ( opts ) { return $(this).dataTable( opts ).api(); }; // All properties that are available to $.fn.dataTable should also be // available on $.fn.DataTable $.each( DataTable, function ( prop, val ) { $.fn.DataTable[ prop ] = val; } ); return DataTable; })); Readme.md000064400000004503151676725350006306 0ustar00# DataTables for jQuery with styling for [DataTables](https://datatables.net/) This package contains distribution files required to style [DataTables library](https://datatables.net) for [jQuery](http://jquery.com/) with styling for [DataTables](https://datatables.net/). DataTables is a table enhancing library which adds features such as paging, ordering, search, scrolling and many more to a static HTML page. A comprehensive API is also available that can be used to manipulate the table. Please refer to the [DataTables web-site](//datatables.net) for a full range of documentation and examples. ## Installation ### Browser For inclusion of this library using a standard `<script>` tag, rather than using this package, it is recommended that you use the [DataTables download builder](//datatables.net/download) which can create CDN or locally hosted packages for you, will all dependencies satisfied. ### npm ``` npm install datatables.net-dt ``` ES3 Syntax ``` var $ = require( 'jquery' ); require( 'datatables.net-dt' )( window, $ ); ``` ES6 Syntax ``` import 'datatables.net-dt' ``` ### bower ``` bower install --save datatables.net-dt ``` ## Documentation Full documentation of the DataTables options, API and plug-in interface are available on the [website](https://datatables.net/reference/index). The site also contains information on the wide variety of plug-ins that are available for DataTables, which can be used to enhance and customise your table even further. ## Bug / Support Support for DataTables is available through the [DataTables forums](//datatables.net/forums) and [commercial support options](//datatables.net/support) are available. ### Contributing If you are thinking of contributing code to DataTables, first of all, thank you! All fixes, patches and enhancements to DataTables are very warmly welcomed. This repository is a distribution repo, so patches and issues sent to this repo will not be accepted. Instead, please direct pull requests to the [DataTables/DataTablesSrc](http://github.com/DataTables/DataTablesSrc). For issues / bugs, please direct your questions to the [DataTables forums](//datatables.net/forums). ## License This software is released under the [MIT license](//datatables.net/license). You are free to use, modify and distribute this software, but all copyright information must remain. License.txt000064400000002140151676725360006706 0ustar00The MIT License (MIT) Copyright SpryMedia Limited and other contributors http://datatables.net Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. 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. types/dataTables.dataTables.d.ts000064400000000144151676725360012632 0ustar00 import DataTable from 'datatables.net'; export default DataTable; export * from 'datatables.net'; js/dataTables.dataTables.js000064400000002157151676725360011654 0ustar00/*! DataTables styling integration * © SpryMedia Ltd - datatables.net/license */ (function( factory ){ if ( typeof define === 'function' && define.amd ) { // AMD define( ['jquery', 'datatables.net'], function ( $ ) { return factory( $, window, document ); } ); } else if ( typeof exports === 'object' ) { // CommonJS var jq = require('jquery'); var cjsRequires = function (root, $) { if ( ! $.fn.dataTable ) { require('datatables.net')(root, $); } }; if (typeof window === 'undefined') { module.exports = function (root, $) { if ( ! root ) { // CommonJS environments without a window global must pass a // root. This will give an error otherwise root = window; } if ( ! $ ) { $ = jq( root ); } cjsRequires( root, $ ); return factory( $, root, root.document ); }; } else { cjsRequires( window, jq ); module.exports = factory( jq, window, window.document ); } } else { // Browser factory( jQuery, window, document ); } }(function( $, window, document ) { 'use strict'; var DataTable = $.fn.dataTable; return DataTable; })); js/dataTables.dataTables.min.js000064400000001103151676725360012424 0ustar00/*! DataTables styling integration * © SpryMedia Ltd - datatables.net/license */ !function(t){var o,d;"function"==typeof define&&define.amd?define(["jquery","datatables.net"],function(e){return t(e,window,document)}):"object"==typeof exports?(o=require("jquery"),d=function(e,n){n.fn.dataTable||require("datatables.net")(e,n)},"undefined"==typeof window?module.exports=function(e,n){return e=e||window,n=n||o(e),d(e,n),t(n,0,e.document)}:(d(window,o),module.exports=t(o,window,window.document))):t(jQuery,window,document)}(function(e,n,t){"use strict";return e.fn.dataTable});js/dataTables.dataTables.mjs000064400000000361151676725360012024 0ustar00/*! DataTables styling integration * © SpryMedia Ltd - datatables.net/license */ import jQuery from 'jquery'; import DataTable from 'datatables.net'; // Allow reassignment of the $ variable let $ = jQuery; export default DataTable; js/dataTables.dataTables.min.mjs000064400000000273151676725360012610 0ustar00/*! DataTables styling integration * © SpryMedia Ltd - datatables.net/license */ import jQuery from"jquery";import DataTable from"datatables.net";let $=jQuery;export default DataTable;css/dataTables.dataTables.css000064400000071424151676725360012207 0ustar00@charset "UTF-8"; :root { --dt-row-selected: 13, 110, 253; --dt-row-selected-text: 255, 255, 255; --dt-row-selected-link: 9, 10, 11; --dt-row-stripe: 0, 0, 0; --dt-row-hover: 0, 0, 0; --dt-column-ordering: 0, 0, 0; --dt-html-background: white; } :root.dark { --dt-html-background: rgb(33, 37, 41); } table.dataTable td.dt-control { text-align: center; cursor: pointer; } table.dataTable td.dt-control:before { display: inline-block; box-sizing: border-box; content: ""; border-top: 5px solid transparent; border-left: 10px solid rgba(0, 0, 0, 0.5); border-bottom: 5px solid transparent; border-right: 0px solid transparent; } table.dataTable tr.dt-hasChild td.dt-control:before { border-top: 10px solid rgba(0, 0, 0, 0.5); border-left: 5px solid transparent; border-bottom: 0px solid transparent; border-right: 5px solid transparent; } html.dark table.dataTable td.dt-control:before, :root[data-bs-theme=dark] table.dataTable td.dt-control:before { border-left-color: rgba(255, 255, 255, 0.5); } html.dark table.dataTable tr.dt-hasChild td.dt-control:before, :root[data-bs-theme=dark] table.dataTable tr.dt-hasChild td.dt-control:before { border-top-color: rgba(255, 255, 255, 0.5); border-left-color: transparent; } div.dt-scroll-body thead tr, div.dt-scroll-body tfoot tr { height: 0; } div.dt-scroll-body thead tr th, div.dt-scroll-body thead tr td, div.dt-scroll-body tfoot tr th, div.dt-scroll-body tfoot tr td { height: 0 !important; padding-top: 0px !important; padding-bottom: 0px !important; border-top-width: 0px !important; border-bottom-width: 0px !important; } div.dt-scroll-body thead tr th div.dt-scroll-sizing, div.dt-scroll-body thead tr td div.dt-scroll-sizing, div.dt-scroll-body tfoot tr th div.dt-scroll-sizing, div.dt-scroll-body tfoot tr td div.dt-scroll-sizing { height: 0 !important; overflow: hidden !important; } table.dataTable thead > tr > th:active, table.dataTable thead > tr > td:active { outline: none; } table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order:before, table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:before { position: absolute; display: block; bottom: 50%; content: "▲"; content: "▲"/""; } table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order:after, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:after, table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order:after, table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:after { position: absolute; display: block; top: 50%; content: "▼"; content: "▼"/""; } table.dataTable thead > tr > th.dt-orderable-asc, table.dataTable thead > tr > th.dt-orderable-desc, table.dataTable thead > tr > th.dt-ordering-asc, table.dataTable thead > tr > th.dt-ordering-desc, table.dataTable thead > tr > td.dt-orderable-asc, table.dataTable thead > tr > td.dt-orderable-desc, table.dataTable thead > tr > td.dt-ordering-asc, table.dataTable thead > tr > td.dt-ordering-desc { position: relative; padding-right: 30px; } table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order, table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order, table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order, table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order, table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order, table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order { position: absolute; right: 12px; top: 0; bottom: 0; width: 12px; } table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order:after, table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order:before, table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order:after, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:after, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:after, table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order:before, table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order:after, table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order:before, table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order:after, table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:after, table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:before, table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:after { left: 0; opacity: 0.125; line-height: 9px; font-size: 0.8em; } table.dataTable thead > tr > th.dt-orderable-asc, table.dataTable thead > tr > th.dt-orderable-desc, table.dataTable thead > tr > td.dt-orderable-asc, table.dataTable thead > tr > td.dt-orderable-desc { cursor: pointer; } table.dataTable thead > tr > th.dt-orderable-asc:hover, table.dataTable thead > tr > th.dt-orderable-desc:hover, table.dataTable thead > tr > td.dt-orderable-asc:hover, table.dataTable thead > tr > td.dt-orderable-desc:hover { outline: 2px solid rgba(0, 0, 0, 0.05); outline-offset: -2px; } table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:after, table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:after { opacity: 0.6; } table.dataTable thead > tr > th.sorting_desc_disabled span.dt-column-order:after, table.dataTable thead > tr > th.sorting_asc_disabled span.dt-column-order:before, table.dataTable thead > tr > td.sorting_desc_disabled span.dt-column-order:after, table.dataTable thead > tr > td.sorting_asc_disabled span.dt-column-order:before { display: none; } table.dataTable thead > tr > th:active, table.dataTable thead > tr > td:active { outline: none; } div.dt-scroll-body > table.dataTable > thead > tr > th, div.dt-scroll-body > table.dataTable > thead > tr > td { overflow: hidden; } :root.dark table.dataTable thead > tr > th.dt-orderable-asc:hover, :root.dark table.dataTable thead > tr > th.dt-orderable-desc:hover, :root.dark table.dataTable thead > tr > td.dt-orderable-asc:hover, :root.dark table.dataTable thead > tr > td.dt-orderable-desc:hover, :root[data-bs-theme=dark] table.dataTable thead > tr > th.dt-orderable-asc:hover, :root[data-bs-theme=dark] table.dataTable thead > tr > th.dt-orderable-desc:hover, :root[data-bs-theme=dark] table.dataTable thead > tr > td.dt-orderable-asc:hover, :root[data-bs-theme=dark] table.dataTable thead > tr > td.dt-orderable-desc:hover { outline: 2px solid rgba(255, 255, 255, 0.05); } div.dt-processing { position: absolute; top: 50%; left: 50%; width: 200px; margin-left: -100px; margin-top: -22px; text-align: center; padding: 2px; z-index: 10; } div.dt-processing > div:last-child { position: relative; width: 80px; height: 15px; margin: 1em auto; } div.dt-processing > div:last-child > div { position: absolute; top: 0; width: 13px; height: 13px; border-radius: 50%; background: rgb(13, 110, 253); background: rgb(var(--dt-row-selected)); animation-timing-function: cubic-bezier(0, 1, 1, 0); } div.dt-processing > div:last-child > div:nth-child(1) { left: 8px; animation: datatables-loader-1 0.6s infinite; } div.dt-processing > div:last-child > div:nth-child(2) { left: 8px; animation: datatables-loader-2 0.6s infinite; } div.dt-processing > div:last-child > div:nth-child(3) { left: 32px; animation: datatables-loader-2 0.6s infinite; } div.dt-processing > div:last-child > div:nth-child(4) { left: 56px; animation: datatables-loader-3 0.6s infinite; } @keyframes datatables-loader-1 { 0% { transform: scale(0); } 100% { transform: scale(1); } } @keyframes datatables-loader-3 { 0% { transform: scale(1); } 100% { transform: scale(0); } } @keyframes datatables-loader-2 { 0% { transform: translate(0, 0); } 100% { transform: translate(24px, 0); } } table.dataTable.nowrap th, table.dataTable.nowrap td { white-space: nowrap; } table.dataTable th, table.dataTable td { box-sizing: border-box; } table.dataTable th.dt-left, table.dataTable td.dt-left { text-align: left; } table.dataTable th.dt-center, table.dataTable td.dt-center { text-align: center; } table.dataTable th.dt-right, table.dataTable td.dt-right { text-align: right; } table.dataTable th.dt-justify, table.dataTable td.dt-justify { text-align: justify; } table.dataTable th.dt-nowrap, table.dataTable td.dt-nowrap { white-space: nowrap; } table.dataTable th.dt-empty, table.dataTable td.dt-empty { text-align: center; vertical-align: top; } table.dataTable th.dt-type-numeric, table.dataTable th.dt-type-date, table.dataTable td.dt-type-numeric, table.dataTable td.dt-type-date { text-align: right; } table.dataTable thead th, table.dataTable thead td, table.dataTable tfoot th, table.dataTable tfoot td { text-align: left; } table.dataTable thead th.dt-head-left, table.dataTable thead td.dt-head-left, table.dataTable tfoot th.dt-head-left, table.dataTable tfoot td.dt-head-left { text-align: left; } table.dataTable thead th.dt-head-center, table.dataTable thead td.dt-head-center, table.dataTable tfoot th.dt-head-center, table.dataTable tfoot td.dt-head-center { text-align: center; } table.dataTable thead th.dt-head-right, table.dataTable thead td.dt-head-right, table.dataTable tfoot th.dt-head-right, table.dataTable tfoot td.dt-head-right { text-align: right; } table.dataTable thead th.dt-head-justify, table.dataTable thead td.dt-head-justify, table.dataTable tfoot th.dt-head-justify, table.dataTable tfoot td.dt-head-justify { text-align: justify; } table.dataTable thead th.dt-head-nowrap, table.dataTable thead td.dt-head-nowrap, table.dataTable tfoot th.dt-head-nowrap, table.dataTable tfoot td.dt-head-nowrap { white-space: nowrap; } table.dataTable tbody th.dt-body-left, table.dataTable tbody td.dt-body-left { text-align: left; } table.dataTable tbody th.dt-body-center, table.dataTable tbody td.dt-body-center { text-align: center; } table.dataTable tbody th.dt-body-right, table.dataTable tbody td.dt-body-right { text-align: right; } table.dataTable tbody th.dt-body-justify, table.dataTable tbody td.dt-body-justify { text-align: justify; } table.dataTable tbody th.dt-body-nowrap, table.dataTable tbody td.dt-body-nowrap { white-space: nowrap; } /* * Table styles */ table.dataTable { width: 100%; margin: 0 auto; border-spacing: 0; /* * Header and footer styles */ /* * Body styles */ } table.dataTable thead th, table.dataTable tfoot th { font-weight: bold; } table.dataTable > thead > tr > th, table.dataTable > thead > tr > td { padding: 10px; border-bottom: 1px solid rgba(0, 0, 0, 0.3); } table.dataTable > thead > tr > th:active, table.dataTable > thead > tr > td:active { outline: none; } table.dataTable > tfoot > tr > th, table.dataTable > tfoot > tr > td { border-top: 1px solid rgba(0, 0, 0, 0.3); padding: 10px 10px 6px 10px; } table.dataTable > tbody > tr { background-color: transparent; } table.dataTable > tbody > tr:first-child > * { border-top: none; } table.dataTable > tbody > tr:last-child > * { border-bottom: none; } table.dataTable > tbody > tr.selected > * { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.9); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.9); color: rgb(255, 255, 255); color: rgb(var(--dt-row-selected-text)); } table.dataTable > tbody > tr.selected a { color: rgb(9, 10, 11); color: rgb(var(--dt-row-selected-link)); } table.dataTable > tbody > tr > th, table.dataTable > tbody > tr > td { padding: 8px 10px; } table.dataTable.row-border > tbody > tr > *, table.dataTable.display > tbody > tr > * { border-top: 1px solid rgba(0, 0, 0, 0.15); } table.dataTable.row-border > tbody > tr:first-child > *, table.dataTable.display > tbody > tr:first-child > * { border-top: none; } table.dataTable.row-border > tbody > tr.selected + tr.selected > td, table.dataTable.display > tbody > tr.selected + tr.selected > td { border-top-color: rgba(13, 110, 253, 0.65); border-top-color: rgba(var(--dt-row-selected), 0.65); } table.dataTable.cell-border > tbody > tr > * { border-top: 1px solid rgba(0, 0, 0, 0.15); border-right: 1px solid rgba(0, 0, 0, 0.15); } table.dataTable.cell-border > tbody > tr > *:first-child { border-left: 1px solid rgba(0, 0, 0, 0.15); } table.dataTable.cell-border > tbody > tr:first-child > * { border-top: 1px solid rgba(0, 0, 0, 0.3); } table.dataTable.stripe > tbody > tr:nth-child(odd) > *, table.dataTable.display > tbody > tr:nth-child(odd) > * { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.023); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-stripe), 0.023); } table.dataTable.stripe > tbody > tr:nth-child(odd).selected > *, table.dataTable.display > tbody > tr:nth-child(odd).selected > * { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.923); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.923); } table.dataTable.hover > tbody > tr:hover > *, table.dataTable.display > tbody > tr:hover > * { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.035); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.035); } table.dataTable.hover > tbody > tr.selected:hover > *, table.dataTable.display > tbody > tr.selected:hover > * { box-shadow: inset 0 0 0 9999px #0d6efd !important; box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 1) !important; } table.dataTable.order-column > tbody tr > .sorting_1, table.dataTable.order-column > tbody tr > .sorting_2, table.dataTable.order-column > tbody tr > .sorting_3, table.dataTable.display > tbody tr > .sorting_1, table.dataTable.display > tbody tr > .sorting_2, table.dataTable.display > tbody tr > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.019); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.019); } table.dataTable.order-column > tbody tr.selected > .sorting_1, table.dataTable.order-column > tbody tr.selected > .sorting_2, table.dataTable.order-column > tbody tr.selected > .sorting_3, table.dataTable.display > tbody tr.selected > .sorting_1, table.dataTable.display > tbody tr.selected > .sorting_2, table.dataTable.display > tbody tr.selected > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.919); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.919); } table.dataTable.display > tbody > tr:nth-child(odd) > .sorting_1, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd) > .sorting_1 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.054); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.054); } table.dataTable.display > tbody > tr:nth-child(odd) > .sorting_2, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd) > .sorting_2 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.047); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.047); } table.dataTable.display > tbody > tr:nth-child(odd) > .sorting_3, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd) > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.039); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.039); } table.dataTable.display > tbody > tr:nth-child(odd).selected > .sorting_1, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd).selected > .sorting_1 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.954); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.954); } table.dataTable.display > tbody > tr:nth-child(odd).selected > .sorting_2, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd).selected > .sorting_2 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.947); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.947); } table.dataTable.display > tbody > tr:nth-child(odd).selected > .sorting_3, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd).selected > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.939); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.939); } table.dataTable.display > tbody > tr.even > .sorting_1, table.dataTable.order-column.stripe > tbody > tr.even > .sorting_1 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.019); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.019); } table.dataTable.display > tbody > tr.even > .sorting_2, table.dataTable.order-column.stripe > tbody > tr.even > .sorting_2 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.011); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.011); } table.dataTable.display > tbody > tr.even > .sorting_3, table.dataTable.order-column.stripe > tbody > tr.even > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.003); box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.003); } table.dataTable.display > tbody > tr.even.selected > .sorting_1, table.dataTable.order-column.stripe > tbody > tr.even.selected > .sorting_1 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.919); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.919); } table.dataTable.display > tbody > tr.even.selected > .sorting_2, table.dataTable.order-column.stripe > tbody > tr.even.selected > .sorting_2 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.911); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.911); } table.dataTable.display > tbody > tr.even.selected > .sorting_3, table.dataTable.order-column.stripe > tbody > tr.even.selected > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.903); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.903); } table.dataTable.display tbody tr:hover > .sorting_1, table.dataTable.order-column.hover tbody tr:hover > .sorting_1 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.082); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.082); } table.dataTable.display tbody tr:hover > .sorting_2, table.dataTable.order-column.hover tbody tr:hover > .sorting_2 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.074); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.074); } table.dataTable.display tbody tr:hover > .sorting_3, table.dataTable.order-column.hover tbody tr:hover > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.062); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.062); } table.dataTable.display tbody tr:hover.selected > .sorting_1, table.dataTable.order-column.hover tbody tr:hover.selected > .sorting_1 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.982); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.982); } table.dataTable.display tbody tr:hover.selected > .sorting_2, table.dataTable.order-column.hover tbody tr:hover.selected > .sorting_2 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.974); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.974); } table.dataTable.display tbody tr:hover.selected > .sorting_3, table.dataTable.order-column.hover tbody tr:hover.selected > .sorting_3 { box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.962); box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.962); } table.dataTable.compact thead th, table.dataTable.compact thead td, table.dataTable.compact tfoot th, table.dataTable.compact tfoot td, table.dataTable.compact tbody th, table.dataTable.compact tbody td { padding: 4px; } /* * Control feature layout */ div.dt-container { position: relative; clear: both; } div.dt-container div.dt-layout-row { display: table; clear: both; width: 100%; } div.dt-container div.dt-layout-row.dt-layout-table { display: block; } div.dt-container div.dt-layout-row.dt-layout-table div.dt-layout-cell { display: block; } div.dt-container div.dt-layout-cell { display: table-cell; vertical-align: middle; padding: 5px 0; } div.dt-container div.dt-layout-cell.dt-full { text-align: center; } div.dt-container div.dt-layout-cell.dt-start { text-align: left; } div.dt-container div.dt-layout-cell.dt-end { text-align: right; } div.dt-container div.dt-layout-cell:empty { display: none; } div.dt-container .dt-search input { border: 1px solid #aaa; border-radius: 3px; padding: 5px; background-color: transparent; color: inherit; margin-left: 3px; } div.dt-container .dt-input { border: 1px solid #aaa; border-radius: 3px; padding: 5px; background-color: transparent; color: inherit; } div.dt-container select.dt-input { padding: 4px; } div.dt-container .dt-paging .dt-paging-button { box-sizing: border-box; display: inline-block; min-width: 1.5em; padding: 0.5em 1em; margin-left: 2px; text-align: center; text-decoration: none !important; cursor: pointer; color: inherit !important; border: 1px solid transparent; border-radius: 2px; background: transparent; } div.dt-container .dt-paging .dt-paging-button.current, div.dt-container .dt-paging .dt-paging-button.current:hover { color: inherit !important; border: 1px solid rgba(0, 0, 0, 0.3); background-color: rgba(0, 0, 0, 0.05); background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(230, 230, 230, 0.05)), color-stop(100%, rgba(0, 0, 0, 0.05))); /* Chrome,Safari4+ */ background: -webkit-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* Chrome10+,Safari5.1+ */ background: -moz-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* FF3.6+ */ background: -ms-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* IE10+ */ background: -o-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* Opera 11.10+ */ background: linear-gradient(to bottom, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* W3C */ } div.dt-container .dt-paging .dt-paging-button.disabled, div.dt-container .dt-paging .dt-paging-button.disabled:hover, div.dt-container .dt-paging .dt-paging-button.disabled:active { cursor: default; color: rgba(0, 0, 0, 0.5) !important; border: 1px solid transparent; background: transparent; box-shadow: none; } div.dt-container .dt-paging .dt-paging-button:hover { color: white !important; border: 1px solid #111; background-color: #111; background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #585858), color-stop(100%, #111)); /* Chrome,Safari4+ */ background: -webkit-linear-gradient(top, #585858 0%, #111 100%); /* Chrome10+,Safari5.1+ */ background: -moz-linear-gradient(top, #585858 0%, #111 100%); /* FF3.6+ */ background: -ms-linear-gradient(top, #585858 0%, #111 100%); /* IE10+ */ background: -o-linear-gradient(top, #585858 0%, #111 100%); /* Opera 11.10+ */ background: linear-gradient(to bottom, #585858 0%, #111 100%); /* W3C */ } div.dt-container .dt-paging .dt-paging-button:active { outline: none; background-color: #0c0c0c; background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #2b2b2b), color-stop(100%, #0c0c0c)); /* Chrome,Safari4+ */ background: -webkit-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* Chrome10+,Safari5.1+ */ background: -moz-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* FF3.6+ */ background: -ms-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* IE10+ */ background: -o-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* Opera 11.10+ */ background: linear-gradient(to bottom, #2b2b2b 0%, #0c0c0c 100%); /* W3C */ box-shadow: inset 0 0 3px #111; } div.dt-container .dt-paging .ellipsis { padding: 0 1em; } div.dt-container .dt-length, div.dt-container .dt-search, div.dt-container .dt-info, div.dt-container .dt-processing, div.dt-container .dt-paging { color: inherit; } div.dt-container .dataTables_scroll { clear: both; } div.dt-container .dataTables_scroll div.dt-scroll-body { -webkit-overflow-scrolling: touch; } div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > th, div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > td, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > th, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > td { vertical-align: middle; } div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > th > div.dataTables_sizing, div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > td > div.dataTables_sizing, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > th > div.dataTables_sizing, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > td > div.dataTables_sizing { height: 0; overflow: hidden; margin: 0 !important; padding: 0 !important; } div.dt-container.dt-empty-footer tbody > tr:last-child > * { border-bottom: 1px solid rgba(0, 0, 0, 0.3); } div.dt-container.dt-empty-footer .dt-scroll-body { border-bottom: 1px solid rgba(0, 0, 0, 0.3); } div.dt-container.dt-empty-footer .dt-scroll-body tbody > tr:last-child > * { border-bottom: none; } @media screen and (max-width: 767px) { div.dt-container div.dt-layout-row { display: block; } div.dt-container div.dt-layout-cell { display: block; } div.dt-container div.dt-layout-cell.dt-full, div.dt-container div.dt-layout-cell.dt-start, div.dt-container div.dt-layout-cell.dt-end { text-align: center; } } @media screen and (max-width: 640px) { .dt-container .dt-length, .dt-container .dt-search { float: none; text-align: center; } .dt-container .dt-search { margin-top: 0.5em; } } html.dark { --dt-row-hover: 255, 255, 255; --dt-row-stripe: 255, 255, 255; --dt-column-ordering: 255, 255, 255; } html.dark table.dataTable > thead > tr > th, html.dark table.dataTable > thead > tr > td { border-bottom: 1px solid rgb(89, 91, 94); } html.dark table.dataTable > thead > tr > th:active, html.dark table.dataTable > thead > tr > td:active { outline: none; } html.dark table.dataTable > tfoot > tr > th, html.dark table.dataTable > tfoot > tr > td { border-top: 1px solid rgb(89, 91, 94); } html.dark table.dataTable.row-border > tbody > tr > *, html.dark table.dataTable.display > tbody > tr > * { border-top: 1px solid rgb(64, 67, 70); } html.dark table.dataTable.row-border > tbody > tr:first-child > *, html.dark table.dataTable.display > tbody > tr:first-child > * { border-top: none; } html.dark table.dataTable.row-border > tbody > tr.selected + tr.selected > td, html.dark table.dataTable.display > tbody > tr.selected + tr.selected > td { border-top-color: rgba(13, 110, 253, 0.65); border-top-color: rgba(var(--dt-row-selected), 0.65); } html.dark table.dataTable.cell-border > tbody > tr > th, html.dark table.dataTable.cell-border > tbody > tr > td { border-top: 1px solid rgb(64, 67, 70); border-right: 1px solid rgb(64, 67, 70); } html.dark table.dataTable.cell-border > tbody > tr > th:first-child, html.dark table.dataTable.cell-border > tbody > tr > td:first-child { border-left: 1px solid rgb(64, 67, 70); } html.dark .dt-container.dt-empty-footer table.dataTable { border-bottom: 1px solid rgb(89, 91, 94); } html.dark .dt-container .dt-search input, html.dark .dt-container .dt-length select { border: 1px solid rgba(255, 255, 255, 0.2); background-color: var(--dt-html-background); } html.dark .dt-container .dt-paging .dt-paging-button.current, html.dark .dt-container .dt-paging .dt-paging-button.current:hover { border: 1px solid rgb(89, 91, 94); background: rgba(255, 255, 255, 0.15); } html.dark .dt-container .dt-paging .dt-paging-button.disabled, html.dark .dt-container .dt-paging .dt-paging-button.disabled:hover, html.dark .dt-container .dt-paging .dt-paging-button.disabled:active { color: #666 !important; } html.dark .dt-container .dt-paging .dt-paging-button:hover { border: 1px solid rgb(53, 53, 53); background: rgb(53, 53, 53); } html.dark .dt-container .dt-paging .dt-paging-button:active { background: #3a3a3a; } /* * Overrides for RTL support */ *[dir=rtl] table.dataTable thead th, *[dir=rtl] table.dataTable thead td, *[dir=rtl] table.dataTable tfoot th, *[dir=rtl] table.dataTable tfoot td { text-align: right; } *[dir=rtl] table.dataTable th.dt-type-numeric, *[dir=rtl] table.dataTable th.dt-type-date, *[dir=rtl] table.dataTable td.dt-type-numeric, *[dir=rtl] table.dataTable td.dt-type-date { text-align: left; } *[dir=rtl] div.dt-container div.dt-layout-cell.dt-start { text-align: right; } *[dir=rtl] div.dt-container div.dt-layout-cell.dt-end { text-align: left; } *[dir=rtl] div.dt-container div.dt-search input { margin: 0 3px 0 0; } css/dataTables.dataTables.min.css000064400000062575151676725360013000 0ustar00:root{--dt-row-selected: 13, 110, 253;--dt-row-selected-text: 255, 255, 255;--dt-row-selected-link: 9, 10, 11;--dt-row-stripe: 0, 0, 0;--dt-row-hover: 0, 0, 0;--dt-column-ordering: 0, 0, 0;--dt-html-background: white}:root.dark{--dt-html-background: rgb(33, 37, 41)}table.dataTable td.dt-control{text-align:center;cursor:pointer}table.dataTable td.dt-control:before{display:inline-block;box-sizing:border-box;content:"";border-top:5px solid transparent;border-left:10px solid rgba(0, 0, 0, 0.5);border-bottom:5px solid transparent;border-right:0px solid transparent}table.dataTable tr.dt-hasChild td.dt-control:before{border-top:10px solid rgba(0, 0, 0, 0.5);border-left:5px solid transparent;border-bottom:0px solid transparent;border-right:5px solid transparent}html.dark table.dataTable td.dt-control:before,:root[data-bs-theme=dark] table.dataTable td.dt-control:before{border-left-color:rgba(255, 255, 255, 0.5)}html.dark table.dataTable tr.dt-hasChild td.dt-control:before,:root[data-bs-theme=dark] table.dataTable tr.dt-hasChild td.dt-control:before{border-top-color:rgba(255, 255, 255, 0.5);border-left-color:transparent}div.dt-scroll-body thead tr,div.dt-scroll-body tfoot tr{height:0}div.dt-scroll-body thead tr th,div.dt-scroll-body thead tr td,div.dt-scroll-body tfoot tr th,div.dt-scroll-body tfoot tr td{height:0 !important;padding-top:0px !important;padding-bottom:0px !important;border-top-width:0px !important;border-bottom-width:0px !important}div.dt-scroll-body thead tr th div.dt-scroll-sizing,div.dt-scroll-body thead tr td div.dt-scroll-sizing,div.dt-scroll-body tfoot tr th div.dt-scroll-sizing,div.dt-scroll-body tfoot tr td div.dt-scroll-sizing{height:0 !important;overflow:hidden !important}table.dataTable thead>tr>th:active,table.dataTable thead>tr>td:active{outline:none}table.dataTable thead>tr>th.dt-orderable-asc span.dt-column-order:before,table.dataTable thead>tr>th.dt-ordering-asc span.dt-column-order:before,table.dataTable thead>tr>td.dt-orderable-asc span.dt-column-order:before,table.dataTable thead>tr>td.dt-ordering-asc span.dt-column-order:before{position:absolute;display:block;bottom:50%;content:"▲";content:"▲"/""}table.dataTable thead>tr>th.dt-orderable-desc span.dt-column-order:after,table.dataTable thead>tr>th.dt-ordering-desc span.dt-column-order:after,table.dataTable thead>tr>td.dt-orderable-desc span.dt-column-order:after,table.dataTable thead>tr>td.dt-ordering-desc span.dt-column-order:after{position:absolute;display:block;top:50%;content:"▼";content:"▼"/""}table.dataTable thead>tr>th.dt-orderable-asc,table.dataTable thead>tr>th.dt-orderable-desc,table.dataTable thead>tr>th.dt-ordering-asc,table.dataTable thead>tr>th.dt-ordering-desc,table.dataTable thead>tr>td.dt-orderable-asc,table.dataTable thead>tr>td.dt-orderable-desc,table.dataTable thead>tr>td.dt-ordering-asc,table.dataTable thead>tr>td.dt-ordering-desc{position:relative;padding-right:30px}table.dataTable thead>tr>th.dt-orderable-asc span.dt-column-order,table.dataTable thead>tr>th.dt-orderable-desc span.dt-column-order,table.dataTable thead>tr>th.dt-ordering-asc span.dt-column-order,table.dataTable thead>tr>th.dt-ordering-desc span.dt-column-order,table.dataTable thead>tr>td.dt-orderable-asc span.dt-column-order,table.dataTable thead>tr>td.dt-orderable-desc span.dt-column-order,table.dataTable thead>tr>td.dt-ordering-asc span.dt-column-order,table.dataTable thead>tr>td.dt-ordering-desc span.dt-column-order{position:absolute;right:12px;top:0;bottom:0;width:12px}table.dataTable thead>tr>th.dt-orderable-asc span.dt-column-order:before,table.dataTable thead>tr>th.dt-orderable-asc span.dt-column-order:after,table.dataTable thead>tr>th.dt-orderable-desc span.dt-column-order:before,table.dataTable thead>tr>th.dt-orderable-desc span.dt-column-order:after,table.dataTable thead>tr>th.dt-ordering-asc span.dt-column-order:before,table.dataTable thead>tr>th.dt-ordering-asc span.dt-column-order:after,table.dataTable thead>tr>th.dt-ordering-desc span.dt-column-order:before,table.dataTable thead>tr>th.dt-ordering-desc span.dt-column-order:after,table.dataTable thead>tr>td.dt-orderable-asc span.dt-column-order:before,table.dataTable thead>tr>td.dt-orderable-asc span.dt-column-order:after,table.dataTable thead>tr>td.dt-orderable-desc span.dt-column-order:before,table.dataTable thead>tr>td.dt-orderable-desc span.dt-column-order:after,table.dataTable thead>tr>td.dt-ordering-asc span.dt-column-order:before,table.dataTable thead>tr>td.dt-ordering-asc span.dt-column-order:after,table.dataTable thead>tr>td.dt-ordering-desc span.dt-column-order:before,table.dataTable thead>tr>td.dt-ordering-desc span.dt-column-order:after{left:0;opacity:.125;line-height:9px;font-size:.8em}table.dataTable thead>tr>th.dt-orderable-asc,table.dataTable thead>tr>th.dt-orderable-desc,table.dataTable thead>tr>td.dt-orderable-asc,table.dataTable thead>tr>td.dt-orderable-desc{cursor:pointer}table.dataTable thead>tr>th.dt-orderable-asc:hover,table.dataTable thead>tr>th.dt-orderable-desc:hover,table.dataTable thead>tr>td.dt-orderable-asc:hover,table.dataTable thead>tr>td.dt-orderable-desc:hover{outline:2px solid rgba(0, 0, 0, 0.05);outline-offset:-2px}table.dataTable thead>tr>th.dt-ordering-asc span.dt-column-order:before,table.dataTable thead>tr>th.dt-ordering-desc span.dt-column-order:after,table.dataTable thead>tr>td.dt-ordering-asc span.dt-column-order:before,table.dataTable thead>tr>td.dt-ordering-desc span.dt-column-order:after{opacity:.6}table.dataTable thead>tr>th.sorting_desc_disabled span.dt-column-order:after,table.dataTable thead>tr>th.sorting_asc_disabled span.dt-column-order:before,table.dataTable thead>tr>td.sorting_desc_disabled span.dt-column-order:after,table.dataTable thead>tr>td.sorting_asc_disabled span.dt-column-order:before{display:none}table.dataTable thead>tr>th:active,table.dataTable thead>tr>td:active{outline:none}div.dt-scroll-body>table.dataTable>thead>tr>th,div.dt-scroll-body>table.dataTable>thead>tr>td{overflow:hidden}:root.dark table.dataTable thead>tr>th.dt-orderable-asc:hover,:root.dark table.dataTable thead>tr>th.dt-orderable-desc:hover,:root.dark table.dataTable thead>tr>td.dt-orderable-asc:hover,:root.dark table.dataTable thead>tr>td.dt-orderable-desc:hover,:root[data-bs-theme=dark] table.dataTable thead>tr>th.dt-orderable-asc:hover,:root[data-bs-theme=dark] table.dataTable thead>tr>th.dt-orderable-desc:hover,:root[data-bs-theme=dark] table.dataTable thead>tr>td.dt-orderable-asc:hover,:root[data-bs-theme=dark] table.dataTable thead>tr>td.dt-orderable-desc:hover{outline:2px solid rgba(255, 255, 255, 0.05)}div.dt-processing{position:absolute;top:50%;left:50%;width:200px;margin-left:-100px;margin-top:-22px;text-align:center;padding:2px;z-index:10}div.dt-processing>div:last-child{position:relative;width:80px;height:15px;margin:1em auto}div.dt-processing>div:last-child>div{position:absolute;top:0;width:13px;height:13px;border-radius:50%;background:rgb(13, 110, 253);background:rgb(var(--dt-row-selected));animation-timing-function:cubic-bezier(0, 1, 1, 0)}div.dt-processing>div:last-child>div:nth-child(1){left:8px;animation:datatables-loader-1 .6s infinite}div.dt-processing>div:last-child>div:nth-child(2){left:8px;animation:datatables-loader-2 .6s infinite}div.dt-processing>div:last-child>div:nth-child(3){left:32px;animation:datatables-loader-2 .6s infinite}div.dt-processing>div:last-child>div:nth-child(4){left:56px;animation:datatables-loader-3 .6s infinite}@keyframes datatables-loader-1{0%{transform:scale(0)}100%{transform:scale(1)}}@keyframes datatables-loader-3{0%{transform:scale(1)}100%{transform:scale(0)}}@keyframes datatables-loader-2{0%{transform:translate(0, 0)}100%{transform:translate(24px, 0)}}table.dataTable.nowrap th,table.dataTable.nowrap td{white-space:nowrap}table.dataTable th,table.dataTable td{box-sizing:border-box}table.dataTable th.dt-left,table.dataTable td.dt-left{text-align:left}table.dataTable th.dt-center,table.dataTable td.dt-center{text-align:center}table.dataTable th.dt-right,table.dataTable td.dt-right{text-align:right}table.dataTable th.dt-justify,table.dataTable td.dt-justify{text-align:justify}table.dataTable th.dt-nowrap,table.dataTable td.dt-nowrap{white-space:nowrap}table.dataTable th.dt-empty,table.dataTable td.dt-empty{text-align:center;vertical-align:top}table.dataTable th.dt-type-numeric,table.dataTable th.dt-type-date,table.dataTable td.dt-type-numeric,table.dataTable td.dt-type-date{text-align:right}table.dataTable thead th,table.dataTable thead td,table.dataTable tfoot th,table.dataTable tfoot td{text-align:left}table.dataTable thead th.dt-head-left,table.dataTable thead td.dt-head-left,table.dataTable tfoot th.dt-head-left,table.dataTable tfoot td.dt-head-left{text-align:left}table.dataTable thead th.dt-head-center,table.dataTable thead td.dt-head-center,table.dataTable tfoot th.dt-head-center,table.dataTable tfoot td.dt-head-center{text-align:center}table.dataTable thead th.dt-head-right,table.dataTable thead td.dt-head-right,table.dataTable tfoot th.dt-head-right,table.dataTable tfoot td.dt-head-right{text-align:right}table.dataTable thead th.dt-head-justify,table.dataTable thead td.dt-head-justify,table.dataTable tfoot th.dt-head-justify,table.dataTable tfoot td.dt-head-justify{text-align:justify}table.dataTable thead th.dt-head-nowrap,table.dataTable thead td.dt-head-nowrap,table.dataTable tfoot th.dt-head-nowrap,table.dataTable tfoot td.dt-head-nowrap{white-space:nowrap}table.dataTable tbody th.dt-body-left,table.dataTable tbody td.dt-body-left{text-align:left}table.dataTable tbody th.dt-body-center,table.dataTable tbody td.dt-body-center{text-align:center}table.dataTable tbody th.dt-body-right,table.dataTable tbody td.dt-body-right{text-align:right}table.dataTable tbody th.dt-body-justify,table.dataTable tbody td.dt-body-justify{text-align:justify}table.dataTable tbody th.dt-body-nowrap,table.dataTable tbody td.dt-body-nowrap{white-space:nowrap}table.dataTable{width:100%;margin:0 auto;border-spacing:0}table.dataTable thead th,table.dataTable tfoot th{font-weight:bold}table.dataTable>thead>tr>th,table.dataTable>thead>tr>td{padding:10px;border-bottom:1px solid rgba(0, 0, 0, 0.3)}table.dataTable>thead>tr>th:active,table.dataTable>thead>tr>td:active{outline:none}table.dataTable>tfoot>tr>th,table.dataTable>tfoot>tr>td{border-top:1px solid rgba(0, 0, 0, 0.3);padding:10px 10px 6px 10px}table.dataTable>tbody>tr{background-color:transparent}table.dataTable>tbody>tr:first-child>*{border-top:none}table.dataTable>tbody>tr:last-child>*{border-bottom:none}table.dataTable>tbody>tr.selected>*{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.9);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.9);color:rgb(255, 255, 255);color:rgb(var(--dt-row-selected-text))}table.dataTable>tbody>tr.selected a{color:rgb(9, 10, 11);color:rgb(var(--dt-row-selected-link))}table.dataTable>tbody>tr>th,table.dataTable>tbody>tr>td{padding:8px 10px}table.dataTable.row-border>tbody>tr>*,table.dataTable.display>tbody>tr>*{border-top:1px solid rgba(0, 0, 0, 0.15)}table.dataTable.row-border>tbody>tr:first-child>*,table.dataTable.display>tbody>tr:first-child>*{border-top:none}table.dataTable.row-border>tbody>tr.selected+tr.selected>td,table.dataTable.display>tbody>tr.selected+tr.selected>td{border-top-color:rgba(13, 110, 253, 0.65);border-top-color:rgba(var(--dt-row-selected), 0.65)}table.dataTable.cell-border>tbody>tr>*{border-top:1px solid rgba(0, 0, 0, 0.15);border-right:1px solid rgba(0, 0, 0, 0.15)}table.dataTable.cell-border>tbody>tr>*:first-child{border-left:1px solid rgba(0, 0, 0, 0.15)}table.dataTable.cell-border>tbody>tr:first-child>*{border-top:1px solid rgba(0, 0, 0, 0.3)}table.dataTable.stripe>tbody>tr:nth-child(odd)>*,table.dataTable.display>tbody>tr:nth-child(odd)>*{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.023);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-stripe), 0.023)}table.dataTable.stripe>tbody>tr:nth-child(odd).selected>*,table.dataTable.display>tbody>tr:nth-child(odd).selected>*{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.923);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.923)}table.dataTable.hover>tbody>tr:hover>*,table.dataTable.display>tbody>tr:hover>*{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.035);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.035)}table.dataTable.hover>tbody>tr.selected:hover>*,table.dataTable.display>tbody>tr.selected:hover>*{box-shadow:inset 0 0 0 9999px #0d6efd !important;box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 1) !important}table.dataTable.order-column>tbody tr>.sorting_1,table.dataTable.order-column>tbody tr>.sorting_2,table.dataTable.order-column>tbody tr>.sorting_3,table.dataTable.display>tbody tr>.sorting_1,table.dataTable.display>tbody tr>.sorting_2,table.dataTable.display>tbody tr>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.019);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.019)}table.dataTable.order-column>tbody tr.selected>.sorting_1,table.dataTable.order-column>tbody tr.selected>.sorting_2,table.dataTable.order-column>tbody tr.selected>.sorting_3,table.dataTable.display>tbody tr.selected>.sorting_1,table.dataTable.display>tbody tr.selected>.sorting_2,table.dataTable.display>tbody tr.selected>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.919);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.919)}table.dataTable.display>tbody>tr:nth-child(odd)>.sorting_1,table.dataTable.order-column.stripe>tbody>tr:nth-child(odd)>.sorting_1{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.054);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.054)}table.dataTable.display>tbody>tr:nth-child(odd)>.sorting_2,table.dataTable.order-column.stripe>tbody>tr:nth-child(odd)>.sorting_2{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.047);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.047)}table.dataTable.display>tbody>tr:nth-child(odd)>.sorting_3,table.dataTable.order-column.stripe>tbody>tr:nth-child(odd)>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.039);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.039)}table.dataTable.display>tbody>tr:nth-child(odd).selected>.sorting_1,table.dataTable.order-column.stripe>tbody>tr:nth-child(odd).selected>.sorting_1{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.954);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.954)}table.dataTable.display>tbody>tr:nth-child(odd).selected>.sorting_2,table.dataTable.order-column.stripe>tbody>tr:nth-child(odd).selected>.sorting_2{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.947);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.947)}table.dataTable.display>tbody>tr:nth-child(odd).selected>.sorting_3,table.dataTable.order-column.stripe>tbody>tr:nth-child(odd).selected>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.939);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.939)}table.dataTable.display>tbody>tr.even>.sorting_1,table.dataTable.order-column.stripe>tbody>tr.even>.sorting_1{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.019);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.019)}table.dataTable.display>tbody>tr.even>.sorting_2,table.dataTable.order-column.stripe>tbody>tr.even>.sorting_2{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.011);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.011)}table.dataTable.display>tbody>tr.even>.sorting_3,table.dataTable.order-column.stripe>tbody>tr.even>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.003);box-shadow:inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.003)}table.dataTable.display>tbody>tr.even.selected>.sorting_1,table.dataTable.order-column.stripe>tbody>tr.even.selected>.sorting_1{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.919);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.919)}table.dataTable.display>tbody>tr.even.selected>.sorting_2,table.dataTable.order-column.stripe>tbody>tr.even.selected>.sorting_2{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.911);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.911)}table.dataTable.display>tbody>tr.even.selected>.sorting_3,table.dataTable.order-column.stripe>tbody>tr.even.selected>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.903);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.903)}table.dataTable.display tbody tr:hover>.sorting_1,table.dataTable.order-column.hover tbody tr:hover>.sorting_1{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.082);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.082)}table.dataTable.display tbody tr:hover>.sorting_2,table.dataTable.order-column.hover tbody tr:hover>.sorting_2{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.074);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.074)}table.dataTable.display tbody tr:hover>.sorting_3,table.dataTable.order-column.hover tbody tr:hover>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(0, 0, 0, 0.062);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.062)}table.dataTable.display tbody tr:hover.selected>.sorting_1,table.dataTable.order-column.hover tbody tr:hover.selected>.sorting_1{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.982);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.982)}table.dataTable.display tbody tr:hover.selected>.sorting_2,table.dataTable.order-column.hover tbody tr:hover.selected>.sorting_2{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.974);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.974)}table.dataTable.display tbody tr:hover.selected>.sorting_3,table.dataTable.order-column.hover tbody tr:hover.selected>.sorting_3{box-shadow:inset 0 0 0 9999px rgba(13, 110, 253, 0.962);box-shadow:inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.962)}table.dataTable.compact thead th,table.dataTable.compact thead td,table.dataTable.compact tfoot th,table.dataTable.compact tfoot td,table.dataTable.compact tbody th,table.dataTable.compact tbody td{padding:4px}div.dt-container{position:relative;clear:both}div.dt-container div.dt-layout-row{display:table;clear:both;width:100%}div.dt-container div.dt-layout-row.dt-layout-table{display:block}div.dt-container div.dt-layout-row.dt-layout-table div.dt-layout-cell{display:block}div.dt-container div.dt-layout-cell{display:table-cell;vertical-align:middle;padding:5px 0}div.dt-container div.dt-layout-cell.dt-full{text-align:center}div.dt-container div.dt-layout-cell.dt-start{text-align:left}div.dt-container div.dt-layout-cell.dt-end{text-align:right}div.dt-container div.dt-layout-cell:empty{display:none}div.dt-container .dt-search input{border:1px solid #aaa;border-radius:3px;padding:5px;background-color:transparent;color:inherit;margin-left:3px}div.dt-container .dt-input{border:1px solid #aaa;border-radius:3px;padding:5px;background-color:transparent;color:inherit}div.dt-container select.dt-input{padding:4px}div.dt-container .dt-paging .dt-paging-button{box-sizing:border-box;display:inline-block;min-width:1.5em;padding:.5em 1em;margin-left:2px;text-align:center;text-decoration:none !important;cursor:pointer;color:inherit !important;border:1px solid transparent;border-radius:2px;background:transparent}div.dt-container .dt-paging .dt-paging-button.current,div.dt-container .dt-paging .dt-paging-button.current:hover{color:inherit !important;border:1px solid rgba(0, 0, 0, 0.3);background-color:rgba(0, 0, 0, 0.05);background:-webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(230, 230, 230, 0.05)), color-stop(100%, rgba(0, 0, 0, 0.05)));background:-webkit-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%);background:-moz-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%);background:-ms-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%);background:-o-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%);background:linear-gradient(to bottom, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%)}div.dt-container .dt-paging .dt-paging-button.disabled,div.dt-container .dt-paging .dt-paging-button.disabled:hover,div.dt-container .dt-paging .dt-paging-button.disabled:active{cursor:default;color:rgba(0, 0, 0, 0.5) !important;border:1px solid transparent;background:transparent;box-shadow:none}div.dt-container .dt-paging .dt-paging-button:hover{color:white !important;border:1px solid #111;background-color:#111;background:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #585858), color-stop(100%, #111));background:-webkit-linear-gradient(top, #585858 0%, #111 100%);background:-moz-linear-gradient(top, #585858 0%, #111 100%);background:-ms-linear-gradient(top, #585858 0%, #111 100%);background:-o-linear-gradient(top, #585858 0%, #111 100%);background:linear-gradient(to bottom, #585858 0%, #111 100%)}div.dt-container .dt-paging .dt-paging-button:active{outline:none;background-color:#0c0c0c;background:-webkit-gradient(linear, left top, left bottom, color-stop(0%, #2b2b2b), color-stop(100%, #0c0c0c));background:-webkit-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%);background:-moz-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%);background:-ms-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%);background:-o-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%);background:linear-gradient(to bottom, #2b2b2b 0%, #0c0c0c 100%);box-shadow:inset 0 0 3px #111}div.dt-container .dt-paging .ellipsis{padding:0 1em}div.dt-container .dt-length,div.dt-container .dt-search,div.dt-container .dt-info,div.dt-container .dt-processing,div.dt-container .dt-paging{color:inherit}div.dt-container .dataTables_scroll{clear:both}div.dt-container .dataTables_scroll div.dt-scroll-body{-webkit-overflow-scrolling:touch}div.dt-container .dataTables_scroll div.dt-scroll-body>table>thead>tr>th,div.dt-container .dataTables_scroll div.dt-scroll-body>table>thead>tr>td,div.dt-container .dataTables_scroll div.dt-scroll-body>table>tbody>tr>th,div.dt-container .dataTables_scroll div.dt-scroll-body>table>tbody>tr>td{vertical-align:middle}div.dt-container .dataTables_scroll div.dt-scroll-body>table>thead>tr>th>div.dataTables_sizing,div.dt-container .dataTables_scroll div.dt-scroll-body>table>thead>tr>td>div.dataTables_sizing,div.dt-container .dataTables_scroll div.dt-scroll-body>table>tbody>tr>th>div.dataTables_sizing,div.dt-container .dataTables_scroll div.dt-scroll-body>table>tbody>tr>td>div.dataTables_sizing{height:0;overflow:hidden;margin:0 !important;padding:0 !important}div.dt-container.dt-empty-footer tbody>tr:last-child>*{border-bottom:1px solid rgba(0, 0, 0, 0.3)}div.dt-container.dt-empty-footer .dt-scroll-body{border-bottom:1px solid rgba(0, 0, 0, 0.3)}div.dt-container.dt-empty-footer .dt-scroll-body tbody>tr:last-child>*{border-bottom:none}@media screen and (max-width: 767px){div.dt-container div.dt-layout-row{display:block}div.dt-container div.dt-layout-cell{display:block}div.dt-container div.dt-layout-cell.dt-full,div.dt-container div.dt-layout-cell.dt-start,div.dt-container div.dt-layout-cell.dt-end{text-align:center}}@media screen and (max-width: 640px){.dt-container .dt-length,.dt-container .dt-search{float:none;text-align:center}.dt-container .dt-search{margin-top:.5em}}html.dark{--dt-row-hover: 255, 255, 255;--dt-row-stripe: 255, 255, 255;--dt-column-ordering: 255, 255, 255}html.dark table.dataTable>thead>tr>th,html.dark table.dataTable>thead>tr>td{border-bottom:1px solid rgb(89, 91, 94)}html.dark table.dataTable>thead>tr>th:active,html.dark table.dataTable>thead>tr>td:active{outline:none}html.dark table.dataTable>tfoot>tr>th,html.dark table.dataTable>tfoot>tr>td{border-top:1px solid rgb(89, 91, 94)}html.dark table.dataTable.row-border>tbody>tr>*,html.dark table.dataTable.display>tbody>tr>*{border-top:1px solid rgb(64, 67, 70)}html.dark table.dataTable.row-border>tbody>tr:first-child>*,html.dark table.dataTable.display>tbody>tr:first-child>*{border-top:none}html.dark table.dataTable.row-border>tbody>tr.selected+tr.selected>td,html.dark table.dataTable.display>tbody>tr.selected+tr.selected>td{border-top-color:rgba(13, 110, 253, 0.65);border-top-color:rgba(var(--dt-row-selected), 0.65)}html.dark table.dataTable.cell-border>tbody>tr>th,html.dark table.dataTable.cell-border>tbody>tr>td{border-top:1px solid rgb(64, 67, 70);border-right:1px solid rgb(64, 67, 70)}html.dark table.dataTable.cell-border>tbody>tr>th:first-child,html.dark table.dataTable.cell-border>tbody>tr>td:first-child{border-left:1px solid rgb(64, 67, 70)}html.dark .dt-container.dt-empty-footer table.dataTable{border-bottom:1px solid rgb(89, 91, 94)}html.dark .dt-container .dt-search input,html.dark .dt-container .dt-length select{border:1px solid rgba(255, 255, 255, 0.2);background-color:var(--dt-html-background)}html.dark .dt-container .dt-paging .dt-paging-button.current,html.dark .dt-container .dt-paging .dt-paging-button.current:hover{border:1px solid rgb(89, 91, 94);background:rgba(255, 255, 255, 0.15)}html.dark .dt-container .dt-paging .dt-paging-button.disabled,html.dark .dt-container .dt-paging .dt-paging-button.disabled:hover,html.dark .dt-container .dt-paging .dt-paging-button.disabled:active{color:#666 !important}html.dark .dt-container .dt-paging .dt-paging-button:hover{border:1px solid rgb(53, 53, 53);background:rgb(53, 53, 53)}html.dark .dt-container .dt-paging .dt-paging-button:active{background:#3a3a3a}*[dir=rtl] table.dataTable thead th,*[dir=rtl] table.dataTable thead td,*[dir=rtl] table.dataTable tfoot th,*[dir=rtl] table.dataTable tfoot td{text-align:right}*[dir=rtl] table.dataTable th.dt-type-numeric,*[dir=rtl] table.dataTable th.dt-type-date,*[dir=rtl] table.dataTable td.dt-type-numeric,*[dir=rtl] table.dataTable td.dt-type-date{text-align:left}*[dir=rtl] div.dt-container div.dt-layout-cell.dt-start{text-align:right}*[dir=rtl] div.dt-container div.dt-layout-cell.dt-end{text-align:left}*[dir=rtl] div.dt-container div.dt-search input{margin:0 3px 0 0}
/home/emeraadmin/.htpasswds/../public_html/Trustmark/../node_modules/../4d695/datatables.net-dt.tar