2019-09-07 23:49:21 +08:00
|
|
|
import { Widget, WidgetOptions } from "./Widget";
|
2019-09-17 23:26:25 +08:00
|
|
|
import { bind_children_to, el } from "./dom";
|
|
|
|
import { ListProperty } from "../observable/property/list/ListProperty";
|
2019-09-07 23:49:21 +08:00
|
|
|
import { Disposer } from "../observable/Disposer";
|
|
|
|
import "./Table.css";
|
2019-09-17 23:26:25 +08:00
|
|
|
import { Disposable } from "../observable/Disposable";
|
2019-10-02 00:30:26 +08:00
|
|
|
import Logger = require("js-logger");
|
2019-09-11 21:51:56 +08:00
|
|
|
|
|
|
|
const logger = Logger.get("core/gui/Table");
|
2019-09-07 23:49:21 +08:00
|
|
|
|
|
|
|
export type Column<T> = {
|
2019-09-14 22:11:45 +08:00
|
|
|
key?: string;
|
2019-09-07 23:49:21 +08:00
|
|
|
title: string;
|
2019-09-14 17:59:50 +08:00
|
|
|
fixed?: boolean;
|
2019-09-10 00:37:20 +08:00
|
|
|
width: number;
|
2019-09-11 21:51:56 +08:00
|
|
|
input?: boolean;
|
2019-09-08 02:43:53 +08:00
|
|
|
text_align?: string;
|
2019-09-11 21:51:56 +08:00
|
|
|
tooltip?: (value: T) => string;
|
2019-09-14 22:11:45 +08:00
|
|
|
sortable?: boolean;
|
2019-09-11 21:51:56 +08:00
|
|
|
render_cell(value: T, disposer: Disposer): string | HTMLElement;
|
2019-09-14 17:59:50 +08:00
|
|
|
footer?: {
|
|
|
|
render_cell(): string;
|
|
|
|
tooltip?(): string;
|
|
|
|
};
|
2019-09-07 23:49:21 +08:00
|
|
|
};
|
|
|
|
|
2019-09-14 22:11:45 +08:00
|
|
|
export enum SortDirection {
|
|
|
|
Asc,
|
|
|
|
Desc,
|
|
|
|
}
|
|
|
|
|
2019-09-07 23:49:21 +08:00
|
|
|
export type TableOptions<T> = WidgetOptions & {
|
|
|
|
values: ListProperty<T>;
|
|
|
|
columns: Column<T>[];
|
2019-09-14 22:11:45 +08:00
|
|
|
sort?(sort_columns: { column: Column<T>; direction: SortDirection }[]): void;
|
2019-09-07 23:49:21 +08:00
|
|
|
};
|
|
|
|
|
2019-09-16 01:32:34 +08:00
|
|
|
export class Table<T> extends Widget {
|
|
|
|
readonly element = el.table({ class: "core_Table" });
|
|
|
|
|
2019-09-07 23:49:21 +08:00
|
|
|
private readonly tbody_element = el.tbody();
|
2019-09-14 17:59:50 +08:00
|
|
|
private readonly footer_row_element?: HTMLTableRowElement;
|
2019-09-07 23:49:21 +08:00
|
|
|
private readonly values: ListProperty<T>;
|
|
|
|
private readonly columns: Column<T>[];
|
|
|
|
|
|
|
|
constructor(options: TableOptions<T>) {
|
2019-09-16 01:32:34 +08:00
|
|
|
super(options);
|
2019-09-07 23:49:21 +08:00
|
|
|
|
|
|
|
this.values = options.values;
|
|
|
|
this.columns = options.columns;
|
|
|
|
|
2019-10-02 00:30:26 +08:00
|
|
|
const sort_columns: { column: Column<T>; direction: SortDirection }[] = [];
|
2019-09-14 22:11:45 +08:00
|
|
|
|
2019-09-07 23:49:21 +08:00
|
|
|
const thead_element = el.thead();
|
|
|
|
const header_tr_element = el.tr();
|
|
|
|
|
|
|
|
let left = 0;
|
2019-09-14 17:59:50 +08:00
|
|
|
let has_footer = false;
|
2019-09-07 23:49:21 +08:00
|
|
|
|
|
|
|
header_tr_element.append(
|
2019-09-14 22:11:45 +08:00
|
|
|
...this.columns.map((column, index) => {
|
|
|
|
const th = el.th(
|
|
|
|
{ data: { index: index.toString() } },
|
|
|
|
el.span({ text: column.title }),
|
|
|
|
);
|
2019-09-07 23:49:21 +08:00
|
|
|
|
2019-09-14 17:59:50 +08:00
|
|
|
if (column.fixed) {
|
2019-09-07 23:49:21 +08:00
|
|
|
th.style.position = "sticky";
|
|
|
|
th.style.left = `${left}px`;
|
2019-09-10 00:37:20 +08:00
|
|
|
left += column.width;
|
2019-09-07 23:49:21 +08:00
|
|
|
}
|
|
|
|
|
2019-09-10 00:37:20 +08:00
|
|
|
th.style.width = `${column.width}px`;
|
2019-09-08 02:43:53 +08:00
|
|
|
|
2019-09-14 17:59:50 +08:00
|
|
|
if (column.footer) {
|
|
|
|
has_footer = true;
|
|
|
|
}
|
|
|
|
|
2019-09-07 23:49:21 +08:00
|
|
|
return th;
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
2019-09-14 22:11:45 +08:00
|
|
|
const sort = options.sort;
|
|
|
|
|
|
|
|
if (sort) {
|
|
|
|
header_tr_element.onmousedown = e => {
|
|
|
|
if (e.target instanceof HTMLElement) {
|
|
|
|
let element: HTMLElement = e.target;
|
|
|
|
|
|
|
|
for (let i = 0; i < 5; i++) {
|
|
|
|
if (element.dataset.index) {
|
|
|
|
break;
|
|
|
|
} else if (element.parentElement) {
|
|
|
|
element = element.parentElement;
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!element.dataset.index) return;
|
|
|
|
|
|
|
|
const index = parseInt(element.dataset.index, 10);
|
|
|
|
const column = this.columns[index];
|
|
|
|
if (!column.sortable) return;
|
|
|
|
|
|
|
|
const existing_index = sort_columns.findIndex(sc => sc.column === column);
|
|
|
|
|
|
|
|
if (existing_index === 0) {
|
|
|
|
const sc = sort_columns[0];
|
|
|
|
sc.direction =
|
|
|
|
sc.direction === SortDirection.Asc
|
|
|
|
? SortDirection.Desc
|
|
|
|
: SortDirection.Asc;
|
|
|
|
} else {
|
|
|
|
if (existing_index !== -1) {
|
|
|
|
sort_columns.splice(existing_index, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
sort_columns.unshift({ column, direction: SortDirection.Asc });
|
|
|
|
}
|
|
|
|
|
|
|
|
sort(sort_columns);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-09-07 23:49:21 +08:00
|
|
|
thead_element.append(header_tr_element);
|
|
|
|
this.tbody_element = el.tbody();
|
|
|
|
this.element.append(thead_element, this.tbody_element);
|
|
|
|
|
2019-09-14 17:59:50 +08:00
|
|
|
if (has_footer) {
|
|
|
|
this.footer_row_element = el.tr();
|
|
|
|
this.element.append(el.tfoot({}, this.footer_row_element));
|
|
|
|
this.create_footer();
|
|
|
|
}
|
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
this.disposables(
|
|
|
|
bind_children_to(this.tbody_element, this.values, this.create_row),
|
|
|
|
this.values.observe(this.update_footer),
|
|
|
|
);
|
2019-09-14 21:15:59 +08:00
|
|
|
|
|
|
|
this.finalize_construction(Table.prototype);
|
2019-09-07 23:49:21 +08:00
|
|
|
}
|
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
private create_row = (value: T, index: number): [HTMLTableRowElement, Disposable] => {
|
|
|
|
const disposer = new Disposer();
|
2019-09-07 23:49:21 +08:00
|
|
|
let left = 0;
|
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
return [
|
|
|
|
el.tr(
|
|
|
|
{},
|
|
|
|
...this.columns.map((column, i) => {
|
|
|
|
const cell = column.fixed ? el.th() : el.td();
|
2019-09-07 23:49:21 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
try {
|
|
|
|
const content = column.render_cell(value, disposer);
|
2019-09-07 23:49:21 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
cell.append(content);
|
2019-09-08 02:43:53 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
if (column.input) cell.classList.add("input");
|
2019-09-11 21:51:56 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
if (column.fixed) {
|
|
|
|
cell.classList.add("fixed");
|
|
|
|
cell.style.left = `${left}px`;
|
|
|
|
left += column.width || 0;
|
|
|
|
}
|
2019-09-11 21:51:56 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
cell.style.width = `${column.width}px`;
|
2019-09-11 21:51:56 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
if (column.text_align) cell.style.textAlign = column.text_align;
|
2019-09-11 21:51:56 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
if (column.tooltip) cell.title = column.tooltip(value);
|
|
|
|
} catch (e) {
|
|
|
|
logger.warn(
|
|
|
|
`Error while rendering cell for index ${index}, column ${i}.`,
|
|
|
|
e,
|
|
|
|
);
|
|
|
|
}
|
2019-09-08 02:43:53 +08:00
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
return cell;
|
|
|
|
}),
|
|
|
|
),
|
|
|
|
disposer,
|
|
|
|
];
|
2019-09-07 23:49:21 +08:00
|
|
|
};
|
2019-09-14 17:59:50 +08:00
|
|
|
|
|
|
|
private create_footer(): void {
|
|
|
|
const footer_cells: HTMLTableHeaderCellElement[] = [];
|
|
|
|
let left = 0;
|
|
|
|
|
|
|
|
for (let i = 0; i < this.columns.length; i++) {
|
|
|
|
const column = this.columns[i];
|
|
|
|
const cell = el.th();
|
|
|
|
|
|
|
|
cell.style.width = `${column.width}px`;
|
|
|
|
|
|
|
|
if (column.fixed) {
|
|
|
|
cell.classList.add("fixed");
|
|
|
|
cell.style.left = `${left}px`;
|
|
|
|
left += column.width || 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (column.footer) {
|
|
|
|
cell.textContent = column.footer.render_cell();
|
|
|
|
cell.title = column.footer.tooltip ? column.footer.tooltip() : "";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (column.text_align) cell.style.textAlign = column.text_align;
|
|
|
|
|
|
|
|
footer_cells.push(cell);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.footer_row_element!.append(...footer_cells);
|
|
|
|
}
|
|
|
|
|
2019-09-17 23:26:25 +08:00
|
|
|
private update_footer = (): void => {
|
2019-09-14 17:59:50 +08:00
|
|
|
if (!this.footer_row_element) return;
|
|
|
|
|
|
|
|
const col_count = this.columns.length;
|
|
|
|
|
|
|
|
for (let i = 0; i < col_count; i++) {
|
|
|
|
const column = this.columns[i];
|
|
|
|
|
|
|
|
if (column.footer) {
|
|
|
|
const cell = this.footer_row_element.children[i] as HTMLTableHeaderCellElement;
|
|
|
|
cell.textContent = column.footer.render_cell();
|
|
|
|
cell.title = column.footer.tooltip ? column.footer.tooltip() : "";
|
|
|
|
}
|
|
|
|
}
|
2019-09-17 23:26:25 +08:00
|
|
|
};
|
2019-09-07 23:49:21 +08:00
|
|
|
}
|