DataTable can now sort columns and allows the user to click on columns and change their sorting. Improved documentation.

This commit is contained in:
MattMo 2023-04-24 13:40:57 -07:00
parent dde2e5293a
commit a100f77b8b

View File

@ -1,31 +1,147 @@
import { IgniteHtml, IgniteProperty } from "../ignite-html/ignite-html.js"; import { IgniteHtml, IgniteObject, IgniteProperty } from "../ignite-html/ignite-html.js";
import { IgniteElement } from "../ignite-html/ignite-element.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"; 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";
class DataColumn { /**
* The outline of a column apart of a DataTable.
*/
class DataColumn extends IgniteObject {
/** /**
* The name of the column, this can contain html. * The name of the column, this can contain html.
* @type {String|Any} * @type {String|Any}
* */ * */
name; name = null;
/**
* A selector to extract the value from the data for this column.
* @type {IgniteProperty|Function}
*/
selector = null;
/**
* A converter to convert the value of this column to a custom component if needed.
* @type {IgniteProperty|Function}
*/
converter = null;
/**
* A custom comparer used for sorted, default is null.
* @type {IgniteProperty|Function}
*/
comparer = null;
/**
* A custom function used for searching, default is null.
* @type {IgniteProperty|Function}
*/
searcher = null;
/** /**
* Whether or not this column can be searched. * Whether or not this column can be searched.
* @type {Boolean} * @type {Boolean}
*/ */
search; searchable = false;
/** /**
* * Whether or not this column can be sorted.
* @param {String} name The name of the column to display * @type {Boolean}
* @param {Boolean} search Whether or not this column is searchable.
*/ */
constructor(name, search) { sortable = false;
/**
* The current sorting of this column if any, default is null. Possible values are asc, desc, null.
* @type {String}
*/
sort = null;
/**
* Creates a new DataColumn with a name and options.
* @param {String} name The name of the column to display
* @param {Object} options The options for this data column.
* @param {Boolean|Function} options.searchable Whether or not this column can be searched. Default is false.
* @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.
* @param {IgniteProperty|Function} options.selector A function to extract the column value from a given data. Default is null.
* @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.
* @param {IgniteProperty|Function} options.searcher A function to seatch a column. Default is null.
*/
constructor(name, options = null) {
super();
this.name = name; this.name = name;
this.search = search;
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;
if (this.sort == "asc" || this.sort == "desc") {
this.sortable = true;
}
}
this.update();
} }
} }
/**
* The outline of a row apart of the DataTable.
*/
class DataRow {
/**
* 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.
* @param {Any} data
* @param {Array<DataColumn} columns
*/
constructor(data, columns) {
this.values = [];
this.columns = [];
columns.forEach(column => {
var value = data;
if (column.selector && column.selector instanceof IgniteProperty) {
value = column.selector.value(data);
} else if (column.selector && column.selector instanceof Function) {
value = column.selector(data);
}
var converted = value;
if (column.converter && column.converter instanceof IgniteProperty) {
converted = column.converter.value(value);
} else if (column.converter && column.converter instanceof Function) {
converted = column.converter(value);
}
this.values.push(value);
this.columns.push(converted);
});
}
}
/**
* A data table component that can render rows and columns and handle
* searching and sorting along with pagination of a data set.
*/
class DataTable extends IgniteElement { class DataTable extends IgniteElement {
constructor() { constructor() {
super(); super();
@ -33,11 +149,17 @@ class DataTable extends IgniteElement {
get styles() { get styles() {
return /*css*/` return /*css*/`
bt-data-table table thead th:first-child { /* 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 {
border-top-left-radius: var(--bs-border-radius); border-top-left-radius: var(--bs-border-radius);
} }
bt-data-table table thead th:last-child { bt-data-table table thead tr:last-child th:last-child {
border-top-right-radius: var(--bs-border-radius); border-top-right-radius: var(--bs-border-radius);
} }
@ -54,53 +176,50 @@ class DataTable extends IgniteElement {
get properties() { get properties() {
return { return {
columns: [], columns: [],
items: new IgniteProperty([], { data: new IgniteProperty([], {
onChange: (oldValue, newValue) => { onChange: (oldValue, newValue) => this.rows = (newValue ? newValue.map(row => new DataRow(row, this.columns)) : []),
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(), onPop: () => this.rows.pop(),
onPush: (array, items) => this.rows.push(...items.map(row => this.converter instanceof IgniteProperty ? this.converter.value(row) : this.converter(row))), onPush: (array, data) => this.rows.push(...data.map(row => new DataRow(row, this.columns))),
onShift: () => this.rows.shift(), 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))), onSplice: (array, start, deleteCount, data) => this.rows.splice(start, deleteCount, ...data.map(row => new DataRow(row, this.columns))),
onUnshift: (array, items) => this.rows.unshift(...items.map(row => this.converter instanceof IgniteProperty ? this.converter.value(row) : this.converter(row))) onUnshift: (array, data) => this.rows.unshift(...data.map(row => new DataRow(row, this.columns)))
}), }),
rows: new IgniteProperty([], { rows: new IgniteProperty([], {
onChange: (oldValue, newValue) => this.filter(), onChange: () => this.filter(),
onPop: () => this.filter(), onPop: () => this.filter(),
onPush: () => this.filter(), onPush: () => this.filter(),
onShift: () => this.filter(), onShift: () => this.filter(),
onSplice: () => this.filter(), onSplice: () => this.filter(),
onUnshift: () => this.filter() onUnshift: () => this.filter()
}), }),
results: new IgniteProperty([], { filtered: new IgniteProperty([], {
onChange: (oldValue, newValue) => this.pageCount = newValue && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0, onChange: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
onPop: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0, onPop: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
onPush: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0, onPush: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
onShift: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0, onShift: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
onSplice: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0, onSplice: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0,
onUnshift: () => this.pageCount = this.results && this.pageSize > 0 ? Math.ceil(this.results.length / this.pageSize) : 0 onUnshift: () => this.pageCount = this.filtered && this.pageSize > 0 ? Math.ceil(this.filtered.length / this.pageSize) : 0
}), }),
filterSearch: null, filterSearch: null,
converter: null,
pageSizeOptions: [15, 25, 50, 100, 500], pageSizeOptions: [15, 25, 50, 100, 500],
pageSize: new IgniteProperty(15, { onChange: (oldValue, newValue) => { pageSize: new IgniteProperty(15, {
//Calculate the new page count based on the page size and items. onChange: (oldValue, newValue) => {
if (!this.items || newValue < 1) { //Calculate the new page count based on the page size and data.
if (!this.data || newValue < 1) {
this.pageCount = 0; this.pageCount = 0;
} else { } else {
this.pageCount = Math.ceil(this.items.length / newValue); this.pageCount = Math.ceil(this.data.length / newValue);
} }
}}), }
pageCount: new IgniteProperty(0, { onChange: (oldValue, 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 the currentPage is greater than the new page count, set it to the last page.
if (this.currentPage > newValue - 1) { if (this.currentPage > newValue - 1) {
this.currentPage = newValue - 1; this.currentPage = newValue - 1;
} }
}}), }
}),
currentPage: 0, currentPage: 0,
searching: false, searching: false,
pendingSearch: null, pendingSearch: null,
@ -126,7 +245,7 @@ class DataTable extends IgniteElement {
), ),
//Row count //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")}`) new span().show(this.showRowCount).class("bg-white p-2 rounded-2 flex-grow-1 flex-sm-grow-0").innerHTML(this.filtered, filtered => filtered.length == 0 ? "No rows" : `${filtered.length} ${(filtered.length < 2 ? "row" : "rows")}`)
), ),
//Search box //Search box
@ -154,18 +273,30 @@ class DataTable extends IgniteElement {
new thead().class("table-dark").child( new thead().class("table-dark").child(
new tr().child( new tr().child(
new list(this.columns, column => { new list(this.columns, column => {
if (column instanceof DataColumn) { return new th().class("text-nowrap").class(column.sortable, sortable => sortable ? "cursor-pointer" : null).attribute("scope", "col").child(
return new th().class("text-nowrap").attribute("scope", "col").child(column.name); new i().show([column.sortable, column.sort], (sortable, sort) => sortable && sort).class("me-2").class(column.sort, sort => sort ? (sort == "asc" ? "fa-solid fa-angle-up" : "fa-solid fa-angle-down") : null),
} else { column.name
return new th().class("text-nowrap").attribute("scope", "col").child(column); ).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();
} }
}
});
}) })
) )
), ),
//Table rows //Table rows
new tbody().child( new tbody().child(
new pagination(this.results, this.pageSize, this.currentPage, row => new tr().child(new list(row, column => new td().child(column)))) new pagination(this.filtered, this.pageSize, this.currentPage, row => new tr().child(new list(row.columns, column => new td().child(column))))
) )
) )
), ),
@ -185,7 +316,7 @@ class DataTable extends IgniteElement {
}), }),
new pager( new pager(
this.results, this.filtered,
this.pageSize, this.pageSize,
this.currentPage, this.currentPage,
2, 2,
@ -193,6 +324,7 @@ class DataTable extends IgniteElement {
if (filler) { if (filler) {
return null; return null;
} }
return new button() return new button()
.class("btn text-center flex-grow-1") .class("btn text-center flex-grow-1")
.class(current, current => current ? "btn-primary" : "btn-secondary") .class(current, current => current ? "btn-primary" : "btn-secondary")
@ -242,17 +374,18 @@ class DataTable extends IgniteElement {
} }
filter() { filter() {
var results = this.rows; var filtered = [];
filtered = filtered.concat(this.rows);
//First filter the results by the search query if we have one. //First filter the results by the search query if we have one.
if (this.filterSearch && this.filterSearch != "" && this.filterSearch != " ") { if (this.filterSearch && this.filterSearch != "" && this.filterSearch != " ") {
var regex = new RegExp(this.filterSearch, 'i'); var regex = new RegExp(this.filterSearch, 'i');
results = results.filter(result => { filtered = filtered.filter((/** @type {DataRow} **/ row) => {
var found = false; var found = false;
for (var i = 0; i < this.columns.length; i++) { for (var i = 0; i < this.columns.length; i++) {
if (this.columns[i] instanceof DataColumn && this.columns[i].search && result[i].match(regex)) { if (this.columns[i].searchable && row.values[i].match(regex)) {
found = true; found = true;
break; break;
} }
@ -262,24 +395,44 @@ class DataTable extends IgniteElement {
}); });
} }
//Set the results //Apply any sorting if needed
this.results = results; filtered.sort((/** @type {DataRow} **/ a, /** @type {DataRow} **/ b) => {
for (var i = 0; i < this.columns.length; i++) {
if (this.columns[i].sortable) {
if (this.columns[i].sort == "asc") {
if (a.values[i] > b.values[i]) {
return 1;
} else if (a.values[i] < b.values[i]) {
return -1;
}
} else if (this.columns[i].sort == "desc") {
if (a.values[i] > b.values[i]) {
return -1;
} else if (a.values[i] < b.values[i]) {
return 11;
}
}
}
}
return 0;
});
this.filtered = filtered;
} }
} }
class Template extends IgniteTemplate { class Template extends IgniteTemplate {
/** /**
* * Creates a new data table with the columns and data to display.
* @param {string|Array<DataColumn>} columns The columns of this table * @param {Array<DataColumn>} columns The columns of this table
* @param {Array<any>} items The items of this table * @param {Array<any>} data The data of this table
* @param {Function|IgniteProperty} converter A converter that converts a row into a series of columns.
*/ */
constructor(columns, items, converter) { constructor(columns, data, converter) {
super("bt-data-table", null); super("bt-data-table", null);
this.property("columns", columns); this.property("columns", columns);
this.property("items", items); this.property("data", data);
this.property("converter", converter);
} }
} }