import { gettext } from "django"; import { CSSResult, html, LitElement, property, TemplateResult } from "lit-element"; import { PBResponse } from "../../api/Client"; import { COMMON_STYLES } from "../../common/styles"; import "./TablePagination"; import "../EmptyState"; export class TableColumn { title: string; orderBy?: string; onClick?: () => void; constructor(title: string, orderBy?: string) { this.title = title; this.orderBy = orderBy; } headerClickHandler(table: Table): void { if (!this.orderBy) { return; } if (table.order === this.orderBy) { table.order = `-${this.orderBy}`; } else { table.order = this.orderBy; } table.fetch(); } private getSortIndicator(table: Table): string { switch (table.order) { case this.orderBy: return "fa-long-arrow-alt-down"; case `-${this.orderBy}`: return "fa-long-arrow-alt-up"; default: return "fa-arrows-alt-v"; } } renderSortable(table: Table): TemplateResult { return html` `; } render(table: Table): TemplateResult { return html` ${this.orderBy ? this.renderSortable(table) : html`${gettext(this.title)}`} `; } } export abstract class Table extends LitElement { abstract apiEndpoint(page: number): Promise>; abstract columns(): TableColumn[]; abstract row(item: T): TemplateResult[]; // eslint-disable-next-line @typescript-eslint/no-unused-vars renderExpanded(item: T): TemplateResult { if (this.expandable) { throw new Error("Expandable is enabled but renderExpanded is not overridden!"); } return html``; } @property({attribute: false}) data?: PBResponse; @property({type: Number}) page = 1; @property({type: String}) order?: string; @property({type: String}) search?: string; @property({type: Boolean}) expandable = false; @property({attribute: false}) expandedRows: boolean[] = []; static get styles(): CSSResult[] { return COMMON_STYLES; } constructor() { super(); this.addEventListener("ak-refresh", () => { this.fetch(); }); } public fetch(): void { this.data = undefined; this.apiEndpoint(this.page).then((r) => { this.data = r; this.page = r.pagination.current; }); } private renderLoading(): TemplateResult { return html`

${gettext("Loading")}

`; } renderEmpty(inner?: TemplateResult): TemplateResult { return html`
${inner ? inner : html``}
`; } private renderRows(): TemplateResult[] | undefined { if (!this.data) { return; } if (this.data.pagination.count === 0) { return [this.renderEmpty()]; } return this.data.results.map((item: T, idx: number) => { if ((this.expandedRows.length - 1) < idx) { this.expandedRows[idx] = false; } return html` ${this.expandable ? html` ` : html``} ${this.row(item).map((col) => { return html`${col}`; })} ${this.renderExpanded(item)} `; }); } renderToolbar(): TemplateResult { return html` `; } renderSearch(): TemplateResult { return html``; } renderTable(): TemplateResult { if (!this.data) { this.fetch(); } return html`
${this.renderSearch()} 
${this.renderToolbar()}
{this.page = page; }}>
${this.expandable ? html` ${this.data ? this.renderRows() : this.renderLoading()}
` : html``} ${this.columns().map((col) => col.render(this))}
{ this.page = page; }}>
`; } render(): TemplateResult { return this.renderTable(); } }