2023-04-20 06:36:23 -07:00
|
|
|
import { IgniteHtml, IgniteProperty } from "../ignite-html/ignite-html.js";
|
|
|
|
import { IgniteElement } from "../ignite-html/ignite-element.js";
|
|
|
|
import { IgniteTemplate, div, table, thead, tbody, tr, th, td, list, pagination, pager, population, nav, ul, li, button, span, input, select, option } from "../ignite-html/ignite-template.js";
|
|
|
|
|
|
|
|
class DataColumn {
|
|
|
|
/**
|
|
|
|
* The name of the column, this can contain html.
|
|
|
|
* @type {String|Any}
|
|
|
|
* */
|
|
|
|
name;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not this column can be searched.
|
|
|
|
* @type {Boolean}
|
|
|
|
*/
|
|
|
|
search;
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {String} name The name of the column to display
|
|
|
|
* @param {Boolean} search Whether or not this column is searchable.
|
|
|
|
*/
|
|
|
|
constructor(name, search) {
|
|
|
|
this.name = name;
|
|
|
|
this.search = search;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class DataTable extends IgniteElement {
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
get properties() {
|
|
|
|
return {
|
|
|
|
columns: [],
|
|
|
|
items: new IgniteProperty([], {
|
|
|
|
onChange: (oldValue, newValue) => {
|
|
|
|
if (newValue) {
|
|
|
|
this.rows = newValue.map(row => this.converter instanceof IgniteProperty ? this.converter.value(row) : this.converter(row));
|
|
|
|
} else {
|
|
|
|
this.rows = [];
|
|
|
|
}
|
|
|
|
},
|
|
|
|
onPop: () => this.rows.pop(),
|
|
|
|
onPush: (array, items) => this.rows.push(...items.map(row => this.converter instanceof IgniteProperty ? this.converter.value(row) : this.converter(row))),
|
|
|
|
onShift: () => this.rows.shift(),
|
|
|
|
onSplice: (array, start, deleteCount, items) => this.rows.splice(start, deleteCount, ...items.map(row => this.converter instanceof IgniteProperty ? this.converter.value(row) : this.converter(row))),
|
|
|
|
onUnshift: (array, items) => this.rows.unshift(...items.map(row => this.converter instanceof IgniteProperty ? this.converter.value(row) : this.converter(row)))
|
|
|
|
}),
|
|
|
|
rows: new IgniteProperty([], {
|
|
|
|
onChange: (oldValue, newValue) => this.filter(),
|
|
|
|
onPop: () => this.filter(),
|
|
|
|
onPush: () => this.filter(),
|
|
|
|
onShift: () => this.filter(),
|
|
|
|
onSplice: () => this.filter(),
|
|
|
|
onUnshift: () => this.filter()
|
|
|
|
}),
|
|
|
|
results: new IgniteProperty([], {
|
|
|
|
onChange: (oldValue, newValue) => this.pageCount = newValue && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0,
|
|
|
|
onPop: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0,
|
|
|
|
onPush: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0,
|
|
|
|
onShift: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0,
|
|
|
|
onSplice: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0,
|
|
|
|
onUnshift: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0
|
|
|
|
}),
|
|
|
|
filterSearch: null,
|
|
|
|
converter: null,
|
|
|
|
pageSizeOptions: [15, 25, 50, 100, 500],
|
|
|
|
pageSize: new IgniteProperty(15, { onChange: (oldValue, newValue) => {
|
|
|
|
//Calculate the new page count based on the page size and items.
|
|
|
|
if (!this.items || newValue < 1) {
|
|
|
|
this.pageCount = 0;
|
|
|
|
} else {
|
|
|
|
this.pageCount = Math.ceil(this.items.length / newValue);
|
|
|
|
}
|
|
|
|
}}),
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}}),
|
|
|
|
currentPage: 0,
|
|
|
|
searching: false,
|
|
|
|
pendingSearch: null,
|
|
|
|
showPageSize: true,
|
|
|
|
showSearchBox: true,
|
|
|
|
showRowCount: true,
|
2023-04-20 16:14:35 -07:00
|
|
|
showPages: true,
|
2023-04-20 06:36:23 -07:00
|
|
|
showPageJumpTo: true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return this.template.child(
|
2023-04-20 16:14:35 -07:00
|
|
|
new div().show([this.showPageSize, this.showSearchBox, this.showRowCount], (pageSize, searchBox, rowCount) => pageSize || searchBox || rowCount).class("row g-3 mb-3").child(
|
|
|
|
new div().class("col-12 col-sm-8 col-lg-8 d-flex flex-row justify-content-start gap-3 flex-nowrap").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-04-20 11:32:08 -07:00
|
|
|
new span().class("input-group-text border-0 bg-white").child(`<i class="fa-solid fa-list-ul"></i>`),
|
2023-04-20 06:36:23 -07:00
|
|
|
|
2023-04-20 11:32:08 -07:00
|
|
|
new select().class("form-select border-0 w-auto").child(
|
|
|
|
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
|
|
|
|
new span().show(this.showRowCount).class("bg-white p-2 rounded-2 flex-grow-1 flex-sm-grow-0").innerHTML(this.results, results => results.length == 0 ? "No rows" : `${results.length} ${(results.length < 2 ? "row" : "rows")}`)
|
|
|
|
),
|
|
|
|
|
|
|
|
//Search box
|
|
|
|
new div().show(this.showSearchBox).class("col-12 col-sm-4 col-lg-4 d-flex flex-row justify-content-end gap-3 flex-wrap").child(
|
|
|
|
new div().class("input-group flex-nowrap w-auto flex-grow-1").child(
|
2023-04-20 11:32:08 -07:00
|
|
|
new span().class("input-group-text border-0 bg-white").child(`<i class="fa-solid fa-magnifying-glass"></i>`),
|
|
|
|
|
|
|
|
new input().class("form-control")
|
|
|
|
.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') {
|
|
|
|
this.search();
|
|
|
|
}
|
|
|
|
})
|
2023-04-20 10:22:33 -07:00
|
|
|
)
|
2023-04-20 06:36:23 -07:00
|
|
|
)
|
|
|
|
),
|
|
|
|
|
2023-04-20 10:22:33 -07:00
|
|
|
new div().class("table-responsive rounded-2 mb-3").child(
|
2023-04-20 06:36:23 -07:00
|
|
|
new table().class("table table-striped align-middle mb-0").child(
|
|
|
|
//Table columns
|
|
|
|
new thead().class("table-dark").child(
|
|
|
|
new tr().child(
|
|
|
|
new list(this.columns, column => {
|
|
|
|
if (column instanceof DataColumn) {
|
2023-04-20 10:22:33 -07:00
|
|
|
return new th().class("text-nowrap").attribute("scope", "col").child(column.name);
|
2023-04-20 06:36:23 -07:00
|
|
|
} else {
|
2023-04-20 10:22:33 -07:00
|
|
|
return new th().class("text-nowrap").attribute("scope", "col").child(column);
|
2023-04-20 06:36:23 -07:00
|
|
|
}
|
|
|
|
})
|
|
|
|
)
|
|
|
|
),
|
|
|
|
|
|
|
|
//Table rows
|
|
|
|
new tbody().child(
|
|
|
|
new pagination(this.results, this.pageSize, this.currentPage, row => new tr().child(new list(row, column => new td().child(column))))
|
|
|
|
)
|
|
|
|
)
|
|
|
|
),
|
|
|
|
|
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()
|
2023-04-20 16:14:35 -07:00
|
|
|
.class("btn btn-secondary flex-grow-1")
|
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(
|
|
|
|
this.results,
|
|
|
|
this.pageSize,
|
|
|
|
this.currentPage,
|
|
|
|
2,
|
|
|
|
(index, current, filler) => {
|
|
|
|
if (filler) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return new button()
|
2023-04-20 16:14:35 -07:00
|
|
|
.class("btn text-center flex-grow-1")
|
2023-04-20 06:36:23 -07:00
|
|
|
.class(current, current => current ? "btn-primary" : "btn-secondary")
|
|
|
|
.innerText(filler ? "..." : index + 1)
|
|
|
|
.onClick(() => this.currentPage = index)
|
|
|
|
}),
|
|
|
|
|
|
|
|
//Next page button
|
|
|
|
new button()
|
2023-04-20 16:14:35 -07:00
|
|
|
.class("btn btn-secondary flex-grow-1")
|
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-04-20 06:36:23 -07:00
|
|
|
new select().class("form-select border-0 w-auto").child(
|
|
|
|
new population(
|
|
|
|
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)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
search() {
|
|
|
|
this.searching = true;
|
|
|
|
|
|
|
|
if (!this.pendingSearch) {
|
|
|
|
this.pendingSearch = setTimeout(async () => {
|
|
|
|
this.pendingSearch = null;
|
|
|
|
|
|
|
|
this.filter();
|
|
|
|
|
|
|
|
this.searching = false;
|
|
|
|
}, 300);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
filter() {
|
|
|
|
var results = this.rows;
|
|
|
|
|
|
|
|
//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');
|
|
|
|
|
|
|
|
results = results.filter(result => {
|
|
|
|
var found = false;
|
|
|
|
|
|
|
|
for (var i = 0; i < this.columns.length; i++) {
|
|
|
|
if (this.columns[i] instanceof DataColumn && this.columns[i].search && result[i].match(regex)) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
//Set the results
|
|
|
|
this.results = results;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Template extends IgniteTemplate {
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string|Array<DataColumn>} columns The columns of this table
|
|
|
|
* @param {Array<any>} items The items of this table
|
|
|
|
* @param {Function|IgniteProperty} converter A converter that converts a row into a series of columns.
|
|
|
|
*/
|
|
|
|
constructor(columns, items, converter) {
|
|
|
|
super("bt-data-table", null);
|
|
|
|
|
|
|
|
this.property("columns", columns);
|
|
|
|
this.property("items", items);
|
|
|
|
this.property("converter", converter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IgniteHtml.register("bt-data-table", DataTable);
|
|
|
|
|
|
|
|
export {
|
|
|
|
Template as DataTable,
|
|
|
|
DataColumn
|
|
|
|
}
|