2023-04-29 10:49:06 -07:00
|
|
|
import { IgniteHtml, IgniteObject, IgniteProperty, IgniteRendering } from "../ignite-html/ignite-html.js";
|
2023-04-20 06:36:23 -07:00
|
|
|
import { IgniteElement } from "../ignite-html/ignite-element.js";
|
2023-04-24 13:40:57 -07:00
|
|
|
import { IgniteTemplate, div, table, thead, tbody, tr, th, td, list, pagination, pager, population, nav, ul, li, button, span, input, select, option, i } from "../ignite-html/ignite-template.js";
|
2023-04-20 06:36:23 -07:00
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
/**
|
|
|
|
* The outline of a column apart of a DataTable.
|
|
|
|
*/
|
|
|
|
class DataColumn extends IgniteObject {
|
2023-07-08 09:49:33 -07:00
|
|
|
/**
|
|
|
|
* The data table this row belongs to.
|
|
|
|
* @type {DataTable}
|
|
|
|
*/
|
|
|
|
table;
|
|
|
|
|
2023-04-20 06:36:23 -07:00
|
|
|
/**
|
|
|
|
* The name of the column, this can contain html.
|
|
|
|
* @type {String|Any}
|
|
|
|
* */
|
2023-04-24 13:40:57 -07:00
|
|
|
name = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A selector to extract the value from the data for this column.
|
|
|
|
* @type {IgniteProperty|Function}
|
|
|
|
*/
|
|
|
|
selector = null;
|
|
|
|
|
|
|
|
/**
|
2023-04-29 10:49:06 -07:00
|
|
|
* A function to convert the value of the column to a component.
|
2023-04-24 13:40:57 -07:00
|
|
|
* @type {IgniteProperty|Function}
|
|
|
|
*/
|
|
|
|
converter = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A custom comparer used for sorted, default is null.
|
|
|
|
* @type {IgniteProperty|Function}
|
|
|
|
*/
|
|
|
|
comparer = null;
|
|
|
|
|
|
|
|
/**
|
2023-07-10 09:47:30 -07:00
|
|
|
* A custom function(value, searchStr, regex) used for searching, default is null.
|
2023-04-24 13:40:57 -07:00
|
|
|
* @type {IgniteProperty|Function}
|
|
|
|
*/
|
|
|
|
searcher = null;
|
2023-04-20 06:36:23 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not this column can be searched.
|
|
|
|
* @type {Boolean}
|
|
|
|
*/
|
2023-04-24 13:40:57 -07:00
|
|
|
searchable = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not this column can be sorted.
|
|
|
|
* @type {Boolean}
|
|
|
|
*/
|
|
|
|
sortable = false;
|
2023-04-20 06:36:23 -07:00
|
|
|
|
|
|
|
/**
|
2023-04-24 13:40:57 -07:00
|
|
|
* The current sorting of this column if any, default is null. Possible values are asc, desc, null.
|
|
|
|
* @type {String}
|
|
|
|
*/
|
|
|
|
sort = null;
|
|
|
|
|
2023-07-08 09:49:33 -07:00
|
|
|
/**
|
|
|
|
* A callback function that is invoked when the column is clicked.
|
|
|
|
* @type {Function} The callback function to be called when this column is clicked. The instance of the column is passed as the first parameter.
|
|
|
|
*/
|
|
|
|
onClick = null;
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
/**
|
|
|
|
* Creates a new DataColumn with a name and options.
|
2023-04-20 06:36:23 -07:00
|
|
|
* @param {String} name The name of the column to display
|
2023-04-24 13:40:57 -07:00
|
|
|
* @param {Object} options The options for this data column.
|
2023-07-10 09:47:30 -07:00
|
|
|
* @param {Boolean|Function} options.searchable Whether or not this column can be searched. Default is false.
|
2023-04-24 13:40:57 -07:00
|
|
|
* @param {String|Function} options.sortable Whether or not this column can be sorted. Default is false.
|
|
|
|
* @param {String} options.sort The current sorting of this column. Possible values: asc, desc. Default is null.
|
2023-04-29 10:49:06 -07:00
|
|
|
* @param {IgniteProperty|Function} options.selector A function to extract the column value from given data. Default is null.
|
2023-04-24 13:40:57 -07:00
|
|
|
* @param {IgniteProperty|Function} options.converter A function to convert the column value to a custom component. Default is null.
|
|
|
|
* @param {IgniteProperty|Function} options.comparer A function to compare two columns for sorting. Default is null.
|
2023-07-10 09:47:30 -07:00
|
|
|
* @param {IgniteProperty|Function} options.searcher A function(value, searchStr, regex) to search a column. Default is null.
|
2023-04-20 06:36:23 -07:00
|
|
|
*/
|
2023-04-24 13:40:57 -07:00
|
|
|
constructor(name, options = null) {
|
|
|
|
super();
|
|
|
|
|
2023-04-20 06:36:23 -07:00
|
|
|
this.name = name;
|
2023-04-24 13:40:57 -07:00
|
|
|
|
|
|
|
if (options) {
|
|
|
|
this.searchable = options.searchable ?? this.searchable;
|
|
|
|
this.sortable = options.sortable ?? this.sortable;
|
|
|
|
this.sort = options.sort ?? this.sort;
|
|
|
|
this.selector = options.selector ?? this.selector;
|
|
|
|
this.converter = options.converter ?? this.converter;
|
|
|
|
this.comparer = options.comparer ?? this.comparer;
|
|
|
|
this.searcher = options.searcher ?? this.searcher;
|
2023-07-08 09:49:33 -07:00
|
|
|
this.onClick = options.onClick ?? this.onClick;
|
2023-04-24 13:40:57 -07:00
|
|
|
|
|
|
|
if (this.sort == "asc" || this.sort == "desc") {
|
|
|
|
this.sortable = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The outline of a row apart of the DataTable.
|
|
|
|
*/
|
|
|
|
class DataRow {
|
2023-04-29 10:49:06 -07:00
|
|
|
/**
|
|
|
|
* The data table this row belongs to.
|
|
|
|
* @type {DataTable}
|
|
|
|
*/
|
|
|
|
table;
|
|
|
|
|
2023-07-08 09:49:33 -07:00
|
|
|
/**
|
|
|
|
* The raw data of this row that is then
|
|
|
|
* broken down into values and columns.
|
|
|
|
* @type {Any}
|
|
|
|
*/
|
|
|
|
data;
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
/**
|
|
|
|
* The raw values of this row.
|
|
|
|
* @type {Array<Any>}
|
|
|
|
*/
|
|
|
|
values;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The columns of this row that will be rendered.
|
|
|
|
* @type {Array<Any>}
|
|
|
|
*/
|
|
|
|
columns;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new DataRow from data a set of columns.
|
2023-04-29 10:49:06 -07:00
|
|
|
* @param {DataTable} table
|
2023-04-24 13:40:57 -07:00
|
|
|
* @param {Any} data
|
|
|
|
* @param {Array<DataColumn} columns
|
|
|
|
*/
|
2023-04-29 10:49:06 -07:00
|
|
|
constructor(table, data, columns) {
|
|
|
|
this.table = table;
|
|
|
|
|
2023-07-08 09:49:33 -07:00
|
|
|
this.data = data;
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
this.values = [];
|
|
|
|
|
|
|
|
this.columns = [];
|
|
|
|
|
2023-04-29 10:49:06 -07:00
|
|
|
for (let i = 0; i < columns.length; i++) {
|
|
|
|
var column = columns[i];
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
var value = data;
|
|
|
|
|
2023-04-29 10:49:06 -07:00
|
|
|
//We want to enter a rendering context for the selector so we have a chance to see if we are returned a property.
|
|
|
|
IgniteRendering.enter();
|
|
|
|
if (column.selector && column.selector instanceof IgniteProperty && column.selector.value instanceof Function) {
|
2023-04-24 13:40:57 -07:00
|
|
|
value = column.selector.value(data);
|
|
|
|
} else if (column.selector && column.selector instanceof Function) {
|
|
|
|
value = column.selector(data);
|
|
|
|
}
|
2023-04-29 10:49:06 -07:00
|
|
|
IgniteRendering.leave();
|
2023-04-24 13:40:57 -07:00
|
|
|
|
2023-04-29 10:49:06 -07:00
|
|
|
//Store this value for this column
|
|
|
|
this.values.push(value);
|
2023-04-24 13:40:57 -07:00
|
|
|
|
2023-04-29 10:49:06 -07:00
|
|
|
//If the value is a property, listen for changes and request reflitering of the table.
|
|
|
|
if (value instanceof IgniteProperty) {
|
|
|
|
value.attachOnChange(() => this.table.requestFilter());
|
2023-04-24 13:40:57 -07:00
|
|
|
}
|
|
|
|
|
2023-04-29 10:49:06 -07:00
|
|
|
//Invoke the column converter if there is one.
|
|
|
|
if (column.converter instanceof Function) {
|
2023-05-19 10:30:27 -07:00
|
|
|
var converter = column.converter;
|
|
|
|
|
|
|
|
IgniteRendering.enter();
|
|
|
|
this.columns.push(converter(value));
|
|
|
|
IgniteRendering.leave();
|
2023-04-29 10:49:06 -07:00
|
|
|
} else {
|
|
|
|
this.columns.push(value);
|
|
|
|
}
|
|
|
|
}
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
/**
|
|
|
|
* A data table component that can render rows and columns and handle
|
|
|
|
* searching and sorting along with pagination of a data set.
|
|
|
|
*/
|
2023-04-20 06:36:23 -07:00
|
|
|
class DataTable extends IgniteElement {
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
2023-04-22 20:21:45 -07:00
|
|
|
get styles() {
|
|
|
|
return /*css*/`
|
2023-04-24 13:40:57 -07:00
|
|
|
/* Prevent selecting of column names */
|
|
|
|
bt-data-table table thead th {
|
|
|
|
user-select: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Round the corners of the table */
|
|
|
|
bt-data-table table thead tr:last-child th:first-child {
|
2023-04-22 20:21:45 -07:00
|
|
|
border-top-left-radius: var(--bs-border-radius);
|
|
|
|
}
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
bt-data-table table thead tr:last-child th:last-child {
|
2023-04-22 20:21:45 -07:00
|
|
|
border-top-right-radius: var(--bs-border-radius);
|
|
|
|
}
|
|
|
|
|
|
|
|
bt-data-table table tbody tr:last-child td:first-child {
|
|
|
|
border-bottom-left-radius: var(--bs-border-radius);
|
|
|
|
}
|
|
|
|
|
|
|
|
bt-data-table table tbody tr:last-child td:last-child {
|
|
|
|
border-bottom-right-radius: var(--bs-border-radius);
|
|
|
|
}
|
|
|
|
`;
|
|
|
|
}
|
|
|
|
|
2023-04-20 06:36:23 -07:00
|
|
|
get properties() {
|
|
|
|
return {
|
|
|
|
columns: [],
|
2023-04-24 13:40:57 -07:00
|
|
|
data: new IgniteProperty([], {
|
2023-04-29 10:49:06 -07:00
|
|
|
onChange: (oldValue, newValue) => this.rows = (newValue ? newValue.map(row => new DataRow(this, row, this.columns)) : []),
|
2023-04-20 06:36:23 -07:00
|
|
|
onPop: () => this.rows.pop(),
|
2023-04-29 10:49:06 -07:00
|
|
|
onPush: (array, data) => this.rows.push(...data.map(row => new DataRow(this, row, this.columns))),
|
2023-04-20 06:36:23 -07:00
|
|
|
onShift: () => this.rows.shift(),
|
2023-04-29 10:49:06 -07:00
|
|
|
onSplice: (array, start, deleteCount, data) => this.rows.splice(start, deleteCount, ...data.map(row => new DataRow(this, row, this.columns))),
|
|
|
|
onUnshift: (array, data) => this.rows.unshift(...data.map(row => new DataRow(this, row, this.columns)))
|
2023-04-20 06:36:23 -07:00
|
|
|
}),
|
|
|
|
rows: new IgniteProperty([], {
|
2023-04-24 13:40:57 -07:00
|
|
|
onChange: () => this.filter(),
|
2023-04-20 06:36:23 -07:00
|
|
|
onPop: () => this.filter(),
|
|
|
|
onPush: () => this.filter(),
|
|
|
|
onShift: () => this.filter(),
|
|
|
|
onSplice: () => this.filter(),
|
|
|
|
onUnshift: () => this.filter()
|
|
|
|
}),
|
2023-04-24 13:40:57 -07:00
|
|
|
filtered: new IgniteProperty([], {
|
|
|
|
onChange: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
|
|
|
|
onPop: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
|
|
|
|
onPush: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
|
|
|
|
onShift: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
|
|
|
|
onSplice: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
|
|
|
|
onUnshift: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0
|
2023-04-20 06:36:23 -07:00
|
|
|
}),
|
|
|
|
filterSearch: null,
|
|
|
|
pageSizeOptions: [15, 25, 50, 100, 500],
|
2023-04-24 13:40:57 -07:00
|
|
|
pageSize: new IgniteProperty(15, {
|
|
|
|
onChange: (oldValue, newValue) => {
|
|
|
|
//Calculate the new page count based on the page size and data.
|
|
|
|
if (!this.data || newValue < 1) {
|
|
|
|
this.pageCount = 0;
|
|
|
|
} else {
|
|
|
|
this.pageCount = Math.ceil(this.data.length / newValue);
|
|
|
|
}
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
2023-04-24 13:40:57 -07:00
|
|
|
}),
|
|
|
|
pageCount: new IgniteProperty(0, {
|
|
|
|
onChange: (oldValue, newValue) => {
|
|
|
|
//If the currentPage is greater than the new page count, set it to the last page.
|
|
|
|
if (this.currentPage > newValue - 1) {
|
|
|
|
this.currentPage = newValue - 1;
|
|
|
|
}
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
2023-04-24 13:40:57 -07:00
|
|
|
}),
|
2023-04-20 06:36:23 -07:00
|
|
|
currentPage: 0,
|
2023-05-24 16:02:51 -07:00
|
|
|
headClass: "table-dark",
|
|
|
|
bodyClass: null,
|
2023-04-20 06:36:23 -07:00
|
|
|
pendingSearch: null,
|
|
|
|
showPageSize: true,
|
2024-05-13 09:01:03 -07:00
|
|
|
pageSizeClass: "form-select w-auto text-dark border-0",
|
|
|
|
pageSizeSpanClass: "input-group-text bg-white text-dark border-0",
|
2023-04-20 06:36:23 -07:00
|
|
|
showSearchBox: true,
|
2024-05-13 09:01:03 -07:00
|
|
|
searchBoxClass: "form-control bg-white border-0",
|
|
|
|
searchBoxSpanClass: "input-group-text bg-white text-dark border-0",
|
2023-04-20 06:36:23 -07:00
|
|
|
showRowCount: true,
|
2023-07-10 16:38:24 -07:00
|
|
|
rowCountClass: "bg-white p-2 rounded-2 flex-grow-1 flex-sm-grow-0 text-dark text-nowrap",
|
2023-04-20 16:14:35 -07:00
|
|
|
showPages: true,
|
2023-04-29 10:49:06 -07:00
|
|
|
showPageJumpTo: true,
|
2023-05-24 16:02:51 -07:00
|
|
|
pageJumpToClass: "form-select border-0 w-auto text-dark",
|
|
|
|
showRefreshButton: true,
|
2024-05-13 09:01:03 -07:00
|
|
|
refreshButtonClass: "btn bg-white text-dark",
|
|
|
|
previousPageButtonClass: "btn bg-dark text-white flex-grow-1",
|
|
|
|
nextPageButtonClass: "btn bg-dark text-white flex-grow-1",
|
|
|
|
primaryPageButtonClass: "btn text-center flex-grow-1 btn-primary",
|
|
|
|
secondaryPageButtonClass: "btn text-center flex-grow-1 btn-secondary",
|
2023-04-29 10:49:06 -07:00
|
|
|
refresh: null,
|
|
|
|
pendingFilter: null,
|
|
|
|
loading: false
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return this.template.child(
|
2023-07-10 16:38:24 -07:00
|
|
|
new div().show([this.showPageSize, this.showSearchBox, this.showRowCount, this.showRefreshButton], (pageSize, searchBox, rowCount, refreshButton) => pageSize || searchBox || rowCount || refreshButton).class("d-flex flex-column flex-sm-row gap-3 justify-content-between flex-wrap mb-3").child(
|
|
|
|
new div().class("d-flex flex-row justify-content-start gap-3 flex-nowrap flex-fill").child(
|
2023-04-20 11:32:08 -07:00
|
|
|
//Page size selector
|
2023-04-20 16:14:35 -07:00
|
|
|
new div().show(this.showPageSize).class("input-group d-flex flex-nowrap w-auto flex-grow-1 flex-sm-grow-0").child(
|
2023-05-24 16:02:51 -07:00
|
|
|
new span().class(this.pageSizeSpanClass).child(`<i class="fa-solid fa-list-ul"></i>`),
|
2023-04-20 06:36:23 -07:00
|
|
|
|
2023-05-24 16:02:51 -07:00
|
|
|
new select().class(this.pageSizeClass).child(
|
2023-04-20 11:32:08 -07:00
|
|
|
new list(this.pageSizeOptions, size => new option().child(size))
|
|
|
|
).value(this.pageSize, true)
|
2023-04-20 06:36:23 -07:00
|
|
|
),
|
|
|
|
|
2023-04-20 16:14:35 -07:00
|
|
|
//Row count
|
2023-05-24 16:02:51 -07:00
|
|
|
new span().show(this.showRowCount).class(this.rowCountClass).innerHTML(this.filtered, filtered => filtered.length == 0 ? "No rows" : `${filtered.length} ${(filtered.length < 2 ? "row" : "rows")}`),
|
2023-04-29 10:49:06 -07:00
|
|
|
|
|
|
|
//Refresh button
|
2023-05-24 16:02:51 -07:00
|
|
|
new button().show(this.showRefreshButton).class(this.refreshButtonClass).child(new i().class("fa-solid fa-arrows-rotate")).onClick(async () => {
|
2023-04-29 10:49:06 -07:00
|
|
|
if (this.refresh instanceof Function) {
|
|
|
|
this.loading = true;
|
|
|
|
|
|
|
|
var refreshStart = performance.now();
|
|
|
|
|
|
|
|
await this.refresh();
|
|
|
|
|
|
|
|
//Add an artifial delay of 250ms at least unless the operation took longer.
|
|
|
|
await new Promise(r => setTimeout(r, Math.max(0, 250 - (performance.now() - refreshStart))));
|
|
|
|
|
|
|
|
this.loading = false;
|
|
|
|
}
|
|
|
|
}),
|
2023-04-20 16:14:35 -07:00
|
|
|
),
|
|
|
|
|
|
|
|
//Search box
|
2023-07-10 16:38:24 -07:00
|
|
|
new div().show(this.showSearchBox).class("d-flex flex-row justify-content-end gap-3 flex-wrap flex-fill").style("max-width", "576px").child(
|
2023-04-20 16:14:35 -07:00
|
|
|
new div().class("input-group flex-nowrap w-auto flex-grow-1").child(
|
2023-05-24 16:02:51 -07:00
|
|
|
new span().class(this.searchBoxSpanClass).child(`<i class="fa-solid fa-magnifying-glass"></i>`),
|
2023-04-20 11:32:08 -07:00
|
|
|
|
2023-05-24 16:02:51 -07:00
|
|
|
new input().class(this.searchBoxClass)
|
2023-04-20 11:32:08 -07:00
|
|
|
.value(this.filterSearch, true)
|
|
|
|
.onEnter(() => this.filter())
|
|
|
|
.onChange(() => this.filter())
|
|
|
|
.on("keydown", (event) => {
|
|
|
|
if (event.key != 'Shift' && event.key != 'Capslock' && event.key != 'Space' && event.key != 'Control') {
|
2023-04-29 10:49:06 -07:00
|
|
|
this.requestSearch();
|
2023-04-20 11:32:08 -07:00
|
|
|
}
|
|
|
|
})
|
2023-04-20 10:22:33 -07:00
|
|
|
)
|
2023-04-20 06:36:23 -07:00
|
|
|
)
|
|
|
|
),
|
2023-04-24 13:40:57 -07:00
|
|
|
|
2023-04-21 08:33:46 -07:00
|
|
|
//Rows
|
2023-04-22 20:21:45 -07:00
|
|
|
new div().class("mb-3").child(
|
2023-04-26 07:27:39 -07:00
|
|
|
new table().class("table table-light align-middle mb-0 table-borderless").child(
|
2023-04-20 06:36:23 -07:00
|
|
|
//Table columns
|
2023-05-24 16:02:51 -07:00
|
|
|
new thead().class(this.headClass).child(
|
2023-04-20 06:36:23 -07:00
|
|
|
new tr().child(
|
2023-04-24 13:40:57 -07:00
|
|
|
new list(this.columns, column => {
|
2023-07-08 09:49:33 -07:00
|
|
|
//Ensure the column table instance is set.
|
|
|
|
column.table = this;
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
return new th().class("text-nowrap").class(column.sortable, sortable => sortable ? "cursor-pointer" : null).attribute("scope", "col").child(
|
2023-07-14 14:10:02 -07:00
|
|
|
new i().show([column.sortable, column.sort], (sortable, sort) => sortable || sort).class("me-2").class(column.sort, sort => sort ? (sort == "asc" ? "fa-solid fa-sort-up" : "fa-solid fa-sort-down") : "fa-solid fa-sort"),
|
2023-04-24 13:40:57 -07:00
|
|
|
column.name
|
|
|
|
).onClick(() => {
|
|
|
|
if (column.sortable) {
|
|
|
|
if (column.sort == "asc") {
|
|
|
|
column.sort = "desc";
|
|
|
|
this.filter();
|
|
|
|
} else if (column.sort == "desc") {
|
|
|
|
column.sort = null;
|
|
|
|
this.filter();
|
|
|
|
} else if (!column.sort) {
|
|
|
|
column.sort = "asc";
|
|
|
|
this.filter();
|
|
|
|
}
|
|
|
|
}
|
2023-07-08 09:49:33 -07:00
|
|
|
|
|
|
|
if (column.onClick) {
|
|
|
|
column.onClick(column);
|
|
|
|
}
|
2023-04-24 13:40:57 -07:00
|
|
|
});
|
2023-04-20 06:36:23 -07:00
|
|
|
})
|
|
|
|
)
|
|
|
|
),
|
|
|
|
|
|
|
|
//Table rows
|
2023-05-24 16:02:51 -07:00
|
|
|
new tbody().class(this.bodyClass).hide(this.loading).child(
|
2023-04-24 13:40:57 -07:00
|
|
|
new pagination(this.filtered, this.pageSize, this.currentPage, row => new tr().child(new list(row.columns, column => new td().child(column))))
|
2023-04-20 06:36:23 -07:00
|
|
|
)
|
2023-04-29 10:49:06 -07:00
|
|
|
),
|
|
|
|
|
|
|
|
//Loading spinner
|
|
|
|
new div().show(this.loading).class("d-flex justify-content-center mt-3").child(
|
|
|
|
new div().class("spinner-border")
|
2023-04-20 06:36:23 -07:00
|
|
|
)
|
|
|
|
),
|
|
|
|
|
2023-04-20 16:14:35 -07:00
|
|
|
new div().class("row g-3").child(
|
|
|
|
//Pagination pages
|
|
|
|
new div().show(this.showPages).class("col-12 col-sm-9 col-lg-8 d-flex flex-row justify-content-start").child(
|
2023-04-20 16:45:47 -07:00
|
|
|
new div().class("btn-group flex-wrap justify-content-center d-flex flex-grow-1 flex-sm-grow-0").child(
|
2023-04-20 06:36:23 -07:00
|
|
|
//Previous page button
|
|
|
|
new button()
|
2024-05-13 09:01:03 -07:00
|
|
|
.class(this.previousPageButtonClass)
|
2023-04-20 06:36:23 -07:00
|
|
|
.child(`<i class="fa-solid fa-chevron-left"></i>`)
|
|
|
|
.onClick(() => {
|
|
|
|
if (this.currentPage > 0) {
|
|
|
|
this.currentPage = this.currentPage - 1;
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
|
|
|
|
new pager(
|
2023-04-24 13:40:57 -07:00
|
|
|
this.filtered,
|
2023-04-20 06:36:23 -07:00
|
|
|
this.pageSize,
|
|
|
|
this.currentPage,
|
|
|
|
2,
|
|
|
|
(index, current, filler) => {
|
|
|
|
if (filler) {
|
|
|
|
return null;
|
|
|
|
}
|
2023-04-24 13:40:57 -07:00
|
|
|
|
2023-04-20 06:36:23 -07:00
|
|
|
return new button()
|
2023-07-10 17:19:27 -07:00
|
|
|
.class(current, current => current ? this.primaryPageButtonClass : this.secondaryPageButtonClass)
|
2023-04-20 06:36:23 -07:00
|
|
|
.innerText(filler ? "..." : index + 1)
|
|
|
|
.onClick(() => this.currentPage = index)
|
|
|
|
}),
|
|
|
|
|
|
|
|
//Next page button
|
|
|
|
new button()
|
2024-05-13 09:01:03 -07:00
|
|
|
.class(this.nextPageButtonClass)
|
2023-04-20 06:36:23 -07:00
|
|
|
.child(`<i class="fa-solid fa-chevron-right"></i>`)
|
|
|
|
.onClick(() => {
|
|
|
|
if (this.currentPage < this.pageCount) {
|
|
|
|
this.currentPage = this.currentPage + 1;
|
|
|
|
}
|
|
|
|
}),
|
2023-04-20 16:14:35 -07:00
|
|
|
)
|
|
|
|
),
|
2023-04-20 06:36:23 -07:00
|
|
|
|
2023-04-20 16:14:35 -07:00
|
|
|
//Page jump to select
|
|
|
|
new div().show(this.showPageJumpTo).class("col-12 col-sm-3 col-lg-4 d-flex flex-row justify-content-end").child(
|
|
|
|
new div().class("input-group flex-nowrap w-auto flex-grow-1 flex-sm-grow-0").child(
|
2023-05-24 16:02:51 -07:00
|
|
|
new select().class(this.pageJumpToClass).child(
|
2023-04-20 06:36:23 -07:00
|
|
|
new population(
|
2023-04-24 13:40:57 -07:00
|
|
|
this.pageCount,
|
2023-04-20 10:22:33 -07:00
|
|
|
index => new option().attribute("selected", this.currentPage, currentPage => currentPage == index ? "selected" : null).value(index).innerText(`Page ${index + 1}`)
|
2023-04-20 06:36:23 -07:00
|
|
|
)
|
|
|
|
).value(this.currentPage, true)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
filter() {
|
2023-04-24 13:40:57 -07:00
|
|
|
var filtered = [];
|
|
|
|
|
|
|
|
filtered = filtered.concat(this.rows);
|
2023-04-20 06:36:23 -07:00
|
|
|
|
|
|
|
//First filter the results by the search query if we have one.
|
|
|
|
if (this.filterSearch && this.filterSearch != "" && this.filterSearch != " ") {
|
|
|
|
var regex = new RegExp(this.filterSearch, 'i');
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
filtered = filtered.filter((/** @type {DataRow} **/ row) => {
|
2023-04-20 06:36:23 -07:00
|
|
|
for (var i = 0; i < this.columns.length; i++) {
|
2023-04-29 10:49:06 -07:00
|
|
|
var value = row.values[i] instanceof IgniteProperty ? row.values[i].value : row.values[i];
|
|
|
|
|
2023-07-10 09:47:30 -07:00
|
|
|
if (this.columns[i].searchable) {
|
|
|
|
if (this.columns[i].searcher && this.columns[i].searcher(value, this.filterSearch, regex)) {
|
|
|
|
return true;
|
|
|
|
} else if (value && typeof value == 'string' && value.match(regex)) {
|
|
|
|
return true;
|
|
|
|
}
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-29 10:49:06 -07:00
|
|
|
return false;
|
2023-04-20 06:36:23 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-04-24 13:40:57 -07:00
|
|
|
//Apply any sorting if needed
|
|
|
|
filtered.sort((/** @type {DataRow} **/ a, /** @type {DataRow} **/ b) => {
|
|
|
|
for (var i = 0; i < this.columns.length; i++) {
|
|
|
|
if (this.columns[i].sortable) {
|
2023-04-29 10:49:06 -07:00
|
|
|
var aValue = a.values[i] instanceof IgniteProperty ? a.values[i].value : a.values[i];
|
|
|
|
var bValue = b.values[i] instanceof IgniteProperty ? b.values[i].value : b.values[i];
|
|
|
|
|
|
|
|
//Sort if we are sorting by ascending or descending
|
2023-04-24 13:40:57 -07:00
|
|
|
if (this.columns[i].sort == "asc") {
|
2023-04-29 10:49:06 -07:00
|
|
|
if (!aValue && bValue) {
|
|
|
|
return -1;
|
|
|
|
} else if (!bValue && aValue) {
|
|
|
|
return 1;
|
|
|
|
} else if (aValue && bValue && aValue > bValue) {
|
2023-04-24 13:40:57 -07:00
|
|
|
return 1;
|
2023-04-29 10:49:06 -07:00
|
|
|
} else if (aValue && bValue && aValue < bValue) {
|
2023-04-24 13:40:57 -07:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (this.columns[i].sort == "desc") {
|
2023-04-29 10:49:06 -07:00
|
|
|
if (!aValue && bValue) {
|
|
|
|
return 1;
|
|
|
|
} else if (!bValue && aValue) {
|
|
|
|
return -1;
|
|
|
|
} else if (aValue && bValue && aValue > bValue) {
|
2023-04-24 13:40:57 -07:00
|
|
|
return -1;
|
2023-04-29 10:49:06 -07:00
|
|
|
} else if (aValue && bValue && aValue < bValue) {
|
|
|
|
return 1;
|
2023-04-24 13:40:57 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
});
|
|
|
|
|
|
|
|
this.filtered = filtered;
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
2023-04-29 10:49:06 -07:00
|
|
|
|
|
|
|
requestSearch() {
|
|
|
|
if (this.pendingSearch) {
|
|
|
|
clearTimeout(this.pendingSearch);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.pendingSearch = setTimeout(() => {
|
|
|
|
this.pendingSearch = null;
|
|
|
|
this.filter();
|
|
|
|
}, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
requestFilter() {
|
|
|
|
if (this.pendingFilter) {
|
|
|
|
clearTimeout(this.pendingFilter);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.pendingFilter = setTimeout(() => {
|
|
|
|
this.pendingFilter = null;
|
|
|
|
this.filter();
|
|
|
|
}, 10);
|
|
|
|
}
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
class Template extends IgniteTemplate {
|
|
|
|
/**
|
2023-04-24 13:40:57 -07:00
|
|
|
* Creates a new data table with the columns and data to display.
|
|
|
|
* @param {Array<DataColumn>} columns The columns of this table
|
|
|
|
* @param {Array<any>} data The data of this table
|
2023-04-20 06:36:23 -07:00
|
|
|
*/
|
2023-04-24 13:40:57 -07:00
|
|
|
constructor(columns, data, converter) {
|
2023-04-20 06:36:23 -07:00
|
|
|
super("bt-data-table", null);
|
|
|
|
|
|
|
|
this.property("columns", columns);
|
2023-04-24 13:40:57 -07:00
|
|
|
this.property("data", data);
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IgniteHtml.register("bt-data-table", DataTable);
|
|
|
|
|
|
|
|
export {
|
|
|
|
Template as DataTable,
|
2023-07-08 09:49:33 -07:00
|
|
|
DataColumn,
|
|
|
|
DataRow
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|