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 {
this.visible.val = true;
this.hover_item(
this.hovered_index != undefined ? (this.hovered_index + 1) % this.items.val.length : 0,
);
}
hover_prev(): void {
this.visible.val = true;
this.hover_item(this.hovered_index ? this.hovered_index - 1 : this.items.val.length - 1);
}
@ -78,8 +80,11 @@ export class Menu<T> extends Widget {
protected set_visible(visible: boolean): void {
super.set_visible(visible);
this.hover_item();
this.inner_element.scrollTo(0, 0);
if (this.visible.val != visible) {
this.hover_item();
this.inner_element.scrollTo(0, 0);
}
}
protected set_selected(): void {

View File

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

View File

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

View File

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

View File

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