The OptimizationResultView is now completely ported to the new GUI system.

This commit is contained in:
Daan Vanden Bosch 2019-09-14 11:59:50 +02:00
parent f7010d75e0
commit c8ff72726e
6 changed files with 139 additions and 23 deletions

View File

@ -61,12 +61,14 @@ export class Menu<T> extends Widget {
} }
hover_next(): void { hover_next(): void {
this.visible.val = true;
this.hover_item( this.hover_item(
this.hovered_index != undefined ? (this.hovered_index + 1) % this.items.val.length : 0, this.hovered_index != undefined ? (this.hovered_index + 1) % this.items.val.length : 0,
); );
} }
hover_prev(): void { hover_prev(): void {
this.visible.val = true;
this.hover_item(this.hovered_index ? this.hovered_index - 1 : this.items.val.length - 1); this.hover_item(this.hovered_index ? this.hovered_index - 1 : this.items.val.length - 1);
} }
@ -78,9 +80,12 @@ export class Menu<T> extends Widget {
protected set_visible(visible: boolean): void { protected set_visible(visible: boolean): void {
super.set_visible(visible); super.set_visible(visible);
if (this.visible.val != visible) {
this.hover_item(); this.hover_item();
this.inner_element.scrollTo(0, 0); this.inner_element.scrollTo(0, 0);
} }
}
protected set_selected(): void { protected set_selected(): void {
// Noop // Noop

View File

@ -52,7 +52,12 @@
white-space: nowrap; white-space: nowrap;
} }
.core_Table tbody th { .core_Table tbody th,
.core_Table tfoot th {
text-align: left;
}
.core_Table th.fixed {
position: sticky; position: sticky;
text-align: left; text-align: left;
} }

View File

@ -13,12 +13,16 @@ const logger = Logger.get("core/gui/Table");
export type Column<T> = { export type Column<T> = {
title: string; title: string;
sticky?: boolean; fixed?: boolean;
width: number; width: number;
input?: boolean; input?: boolean;
text_align?: string; text_align?: string;
tooltip?: (value: T) => string; tooltip?: (value: T) => string;
render_cell(value: T, disposer: Disposer): string | HTMLElement; render_cell(value: T, disposer: Disposer): string | HTMLElement;
footer?: {
render_cell(): string;
tooltip?(): string;
};
}; };
export type TableOptions<T> = WidgetOptions & { export type TableOptions<T> = WidgetOptions & {
@ -29,6 +33,7 @@ export type TableOptions<T> = WidgetOptions & {
export class Table<T> extends Widget<HTMLTableElement> { export class Table<T> extends Widget<HTMLTableElement> {
private readonly table_disposer = this.disposable(new Disposer()); private readonly table_disposer = this.disposable(new Disposer());
private readonly tbody_element = el.tbody(); private readonly tbody_element = el.tbody();
private readonly footer_row_element?: HTMLTableRowElement;
private readonly values: ListProperty<T>; private readonly values: ListProperty<T>;
private readonly columns: Column<T>[]; private readonly columns: Column<T>[];
@ -42,12 +47,13 @@ export class Table<T> extends Widget<HTMLTableElement> {
const header_tr_element = el.tr(); const header_tr_element = el.tr();
let left = 0; let left = 0;
let has_footer = false;
header_tr_element.append( header_tr_element.append(
...this.columns.map(column => { ...this.columns.map(column => {
const th = el.th({}, el.span({ text: column.title })); const th = el.th({}, el.span({ text: column.title }));
if (column.sticky) { if (column.fixed) {
th.style.position = "sticky"; th.style.position = "sticky";
th.style.left = `${left}px`; th.style.left = `${left}px`;
left += column.width; left += column.width;
@ -55,6 +61,10 @@ export class Table<T> extends Widget<HTMLTableElement> {
th.style.width = `${column.width}px`; th.style.width = `${column.width}px`;
if (column.footer) {
has_footer = true;
}
return th; return th;
}), }),
); );
@ -63,6 +73,12 @@ export class Table<T> extends Widget<HTMLTableElement> {
this.tbody_element = el.tbody(); this.tbody_element = el.tbody();
this.element.append(thead_element, this.tbody_element); this.element.append(thead_element, this.tbody_element);
if (has_footer) {
this.footer_row_element = el.tr();
this.element.append(el.tfoot({}, this.footer_row_element));
this.create_footer();
}
this.disposables(this.values.observe_list(this.update_table)); this.disposables(this.values.observe_list(this.update_table));
this.splice_rows(0, this.values.length.val, this.values.val); this.splice_rows(0, this.values.length.val, this.values.val);
@ -71,6 +87,7 @@ export class Table<T> extends Widget<HTMLTableElement> {
private update_table = (change: ListPropertyChangeEvent<T>): void => { private update_table = (change: ListPropertyChangeEvent<T>): void => {
if (change.type === ListChangeType.ListChange) { if (change.type === ListChangeType.ListChange) {
this.splice_rows(change.index, change.removed.length, change.inserted); this.splice_rows(change.index, change.removed.length, change.inserted);
this.update_footer();
} else if (change.type === ListChangeType.ValueChange) { } else if (change.type === ListChangeType.ValueChange) {
// TODO: update rows // TODO: update rows
} }
@ -104,7 +121,7 @@ export class Table<T> extends Widget<HTMLTableElement> {
return el.tr( return el.tr(
{}, {},
...this.columns.map((column, i) => { ...this.columns.map((column, i) => {
const cell = column.sticky ? el.th() : el.td(); const cell = column.fixed ? el.th() : el.td();
try { try {
const content = column.render_cell(value, disposer); const content = column.render_cell(value, disposer);
@ -113,12 +130,13 @@ export class Table<T> extends Widget<HTMLTableElement> {
if (column.input) cell.classList.add("input"); if (column.input) cell.classList.add("input");
if (column.sticky) { if (column.fixed) {
cell.classList.add("fixed");
cell.style.left = `${left}px`; cell.style.left = `${left}px`;
left += column.width || 0; left += column.width || 0;
} }
if (column.width != undefined) cell.style.width = `${column.width}px`; cell.style.width = `${column.width}px`;
if (column.text_align) cell.style.textAlign = column.text_align; if (column.text_align) cell.style.textAlign = column.text_align;
@ -131,4 +149,49 @@ export class Table<T> extends Widget<HTMLTableElement> {
}), }),
); );
}; };
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);
}
private update_footer(): void {
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() : "";
}
}
}
} }

View File

@ -61,6 +61,9 @@ export const el = {
tbody: (attributes?: {}, ...children: HTMLElement[]): HTMLTableSectionElement => tbody: (attributes?: {}, ...children: HTMLElement[]): HTMLTableSectionElement =>
create_element("tbody", attributes, ...children), create_element("tbody", attributes, ...children),
tfoot: (attributes?: {}, ...children: HTMLElement[]): HTMLTableSectionElement =>
create_element("tfoot", attributes, ...children),
tr: (attributes?: {}, ...children: HTMLElement[]): HTMLTableRowElement => tr: (attributes?: {}, ...children: HTMLElement[]): HTMLTableRowElement =>
create_element("tr", attributes, ...children), create_element("tr", attributes, ...children),

View File

@ -36,7 +36,7 @@ export class MethodsForEpisodeView extends ResizableWidget {
columns: [ columns: [
{ {
title: "Method", title: "Method",
sticky: true, fixed: true,
width: 250, width: 250,
render_cell(method: HuntMethodModel) { render_cell(method: HuntMethodModel) {
return method.name; return method.name;
@ -44,7 +44,7 @@ export class MethodsForEpisodeView extends ResizableWidget {
}, },
{ {
title: "Time", title: "Time",
sticky: true, fixed: true,
width: 60, width: 60,
input: true, input: true,
render_cell(method: HuntMethodModel, disposer: Disposer) { render_cell(method: HuntMethodModel, disposer: Disposer) {

View File

@ -8,6 +8,7 @@ import { OptimalMethodModel, OptimalResultModel } from "../model";
import { Difficulty } from "../../core/model"; import { Difficulty } from "../../core/model";
import { Episode } from "../../core/data_formats/parsing/quest/Episode"; import { Episode } from "../../core/data_formats/parsing/quest/Episode";
import "./OptimizationResultView.css"; import "./OptimizationResultView.css";
import { Duration } from "luxon";
export class OptimizationResultView extends Widget { export class OptimizationResultView extends Widget {
private results_observer?: Disposable; private results_observer?: Disposable;
@ -57,18 +58,33 @@ export class OptimizationResultView extends Widget {
this.table.dispose(); this.table.dispose();
} }
let total_runs = 0;
let total_time = Duration.fromMillis(0);
if (result) {
for (const method of result.optimal_methods) {
total_runs += method.runs;
total_time = total_time.plus(method.total_time);
}
}
const columns: Column<OptimalMethodModel>[] = [ const columns: Column<OptimalMethodModel>[] = [
{ {
title: "Difficulty", title: "Difficulty",
sticky: true, fixed: true,
width: 80, width: 80,
render_cell(value: OptimalMethodModel) { render_cell(value: OptimalMethodModel) {
return Difficulty[value.difficulty]; return Difficulty[value.difficulty];
}, },
footer: {
render_cell() {
return "Totals:";
},
},
}, },
{ {
title: "Method", title: "Method",
sticky: true, fixed: true,
width: 250, width: 250,
render_cell(value: OptimalMethodModel) { render_cell(value: OptimalMethodModel) {
return value.method_name; return value.method_name;
@ -76,7 +92,7 @@ export class OptimizationResultView extends Widget {
}, },
{ {
title: "Ep.", title: "Ep.",
sticky: true, fixed: true,
width: 40, width: 40,
render_cell(value: OptimalMethodModel) { render_cell(value: OptimalMethodModel) {
return Episode[value.method_episode]; return Episode[value.method_episode];
@ -84,7 +100,7 @@ export class OptimizationResultView extends Widget {
}, },
{ {
title: "Section ID", title: "Section ID",
sticky: true, fixed: true,
width: 90, width: 90,
render_cell(value: OptimalMethodModel) { render_cell(value: OptimalMethodModel) {
const element = el.span( const element = el.span(
@ -107,45 +123,69 @@ export class OptimizationResultView extends Widget {
title: "Runs", title: "Runs",
width: 60, width: 60,
text_align: "right", text_align: "right",
render_cell(value: OptimalMethodModel) {
return value.runs.toFixed(1);
},
tooltip(value: OptimalMethodModel) { tooltip(value: OptimalMethodModel) {
return value.runs.toString(); return value.runs.toString();
}, },
render_cell(value: OptimalMethodModel) { footer: {
return value.runs.toFixed(1); render_cell() {
return total_runs.toFixed(1);
},
tooltip() {
return total_runs.toString();
},
}, },
}, },
{ {
title: "Total Hours", title: "Total Hours",
width: 60, width: 60,
text_align: "right", text_align: "right",
render_cell(value: OptimalMethodModel) {
return value.total_time.as("hours").toFixed(1);
},
tooltip(value: OptimalMethodModel) { tooltip(value: OptimalMethodModel) {
return value.total_time.as("hours").toString(); return value.total_time.as("hours").toString();
}, },
render_cell(value: OptimalMethodModel) { footer: {
return value.total_time.as("hours").toFixed(1); render_cell() {
return total_time.as("hours").toFixed(1);
},
tooltip() {
return total_time.as("hours").toString();
},
}, },
}, },
]; ];
if (result) { if (result) {
for (const item of result.wanted_items) { for (const item of result.wanted_items) {
let totalCount = 0; let total_count = 0;
for (const method of result.optimal_methods) { for (const method of result.optimal_methods) {
totalCount += method.item_counts.get(item) || 0; total_count += method.item_counts.get(item) || 0;
} }
columns.push({ columns.push({
title: item.name, title: item.name,
width: 80, width: 80,
text_align: "right", text_align: "right",
render_cell(value: OptimalMethodModel) {
const count = value.item_counts.get(item);
return count ? count.toFixed(2) : "";
},
tooltip(value: OptimalMethodModel) { tooltip(value: OptimalMethodModel) {
const count = value.item_counts.get(item); const count = value.item_counts.get(item);
return count ? count.toString() : ""; return count ? count.toString() : "";
}, },
render_cell(value: OptimalMethodModel) { footer: {
const count = value.item_counts.get(item); render_cell() {
return count ? count.toFixed(2) : ""; return total_count.toFixed(2);
},
tooltip() {
return total_count.toString();
},
}, },
}); });
} }