2019-09-10 00:37:20 +08:00
|
|
|
import { Widget } from "../../core/gui/Widget";
|
2019-09-11 21:51:56 +08:00
|
|
|
import { el, section_id_icon } from "../../core/gui/dom";
|
|
|
|
import { Column, Table } from "../../core/gui/Table";
|
2019-09-10 00:37:20 +08:00
|
|
|
import { Disposable } from "../../core/observable/Disposable";
|
|
|
|
import { list_property } from "../../core/observable";
|
2019-09-11 21:51:56 +08:00
|
|
|
import { OptimalMethodModel, OptimalResultModel } from "../model";
|
2019-09-10 00:37:20 +08:00
|
|
|
import { Difficulty } from "../../core/model";
|
|
|
|
import { Episode } from "../../core/data_formats/parsing/quest/Episode";
|
2019-09-11 21:51:56 +08:00
|
|
|
import "./OptimizationResultView.css";
|
2019-09-14 17:59:50 +08:00
|
|
|
import { Duration } from "luxon";
|
2019-12-22 02:40:42 +08:00
|
|
|
import { ServerMap } from "../../core/stores/ServerMap";
|
|
|
|
import { HuntOptimizerStore } from "../stores/HuntOptimizerStore";
|
2019-09-10 00:37:20 +08:00
|
|
|
|
|
|
|
export class OptimizationResultView extends Widget {
|
2019-09-16 01:32:34 +08:00
|
|
|
readonly element = el.div(
|
|
|
|
{ class: "hunt_optimizer_OptimizationResultView" },
|
|
|
|
el.h2({ text: "Ideal Combination of Methods" }),
|
|
|
|
);
|
|
|
|
|
2019-09-10 00:37:20 +08:00
|
|
|
private results_observer?: Disposable;
|
2019-09-11 21:51:56 +08:00
|
|
|
private table?: Table<OptimalMethodModel>;
|
2019-09-10 00:37:20 +08:00
|
|
|
|
2019-12-22 02:40:42 +08:00
|
|
|
constructor(hunt_optimizer_stores: ServerMap<HuntOptimizerStore>) {
|
2019-09-16 01:32:34 +08:00
|
|
|
super();
|
2019-09-10 00:37:20 +08:00
|
|
|
|
|
|
|
this.disposable(
|
|
|
|
hunt_optimizer_stores.observe_current(
|
|
|
|
hunt_optimizer_store => {
|
|
|
|
if (this.results_observer) {
|
|
|
|
this.results_observer.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.results_observer = hunt_optimizer_store.result.observe(
|
2019-09-11 21:51:56 +08:00
|
|
|
({ value }) => this.update_table(value),
|
2019-09-10 00:37:20 +08:00
|
|
|
{
|
|
|
|
call_now: true,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
{ call_now: true },
|
|
|
|
),
|
|
|
|
);
|
2019-09-14 21:15:59 +08:00
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
this.finalize_construction();
|
2019-09-10 00:37:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): void {
|
|
|
|
super.dispose();
|
|
|
|
|
|
|
|
if (this.results_observer) {
|
|
|
|
this.results_observer.dispose();
|
|
|
|
}
|
2019-09-11 21:51:56 +08:00
|
|
|
|
|
|
|
if (this.table) {
|
|
|
|
this.table.dispose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private update_table(result?: OptimalResultModel): void {
|
|
|
|
if (this.table) {
|
|
|
|
this.table.dispose();
|
|
|
|
}
|
|
|
|
|
2019-09-14 17:59:50 +08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-11 21:51:56 +08:00
|
|
|
const columns: Column<OptimalMethodModel>[] = [
|
|
|
|
{
|
|
|
|
title: "Difficulty",
|
2019-09-14 17:59:50 +08:00
|
|
|
fixed: true,
|
2019-09-11 21:51:56 +08:00
|
|
|
width: 80,
|
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
return Difficulty[value.difficulty];
|
|
|
|
},
|
2019-09-14 17:59:50 +08:00
|
|
|
footer: {
|
|
|
|
render_cell() {
|
|
|
|
return "Totals:";
|
|
|
|
},
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Method",
|
2019-09-14 17:59:50 +08:00
|
|
|
fixed: true,
|
2019-09-11 21:51:56 +08:00
|
|
|
width: 250,
|
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
return value.method_name;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Ep.",
|
2019-09-14 17:59:50 +08:00
|
|
|
fixed: true,
|
2019-09-11 21:51:56 +08:00
|
|
|
width: 40,
|
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
return Episode[value.method_episode];
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Section ID",
|
2019-09-14 17:59:50 +08:00
|
|
|
fixed: true,
|
2019-09-11 21:51:56 +08:00
|
|
|
width: 90,
|
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
const element = el.span(
|
|
|
|
{},
|
|
|
|
...value.section_ids.map(sid => section_id_icon(sid, { size: 17 })),
|
|
|
|
);
|
|
|
|
element.style.display = "flex";
|
|
|
|
return element;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Time/Run",
|
|
|
|
width: 90,
|
|
|
|
text_align: "center",
|
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
return value.method_time.toFormat("hh:mm");
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Runs",
|
|
|
|
width: 60,
|
|
|
|
text_align: "right",
|
2019-09-14 17:59:50 +08:00
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
return value.runs.toFixed(1);
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
tooltip(value: OptimalMethodModel) {
|
|
|
|
return value.runs.toString();
|
|
|
|
},
|
2019-09-14 17:59:50 +08:00
|
|
|
footer: {
|
|
|
|
render_cell() {
|
|
|
|
return total_runs.toFixed(1);
|
|
|
|
},
|
|
|
|
tooltip() {
|
|
|
|
return total_runs.toString();
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "Total Hours",
|
|
|
|
width: 60,
|
|
|
|
text_align: "right",
|
2019-09-14 17:59:50 +08:00
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
return value.total_time.as("hours").toFixed(1);
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
tooltip(value: OptimalMethodModel) {
|
|
|
|
return value.total_time.as("hours").toString();
|
|
|
|
},
|
2019-09-14 17:59:50 +08:00
|
|
|
footer: {
|
|
|
|
render_cell() {
|
|
|
|
return total_time.as("hours").toFixed(1);
|
|
|
|
},
|
|
|
|
tooltip() {
|
|
|
|
return total_time.as("hours").toString();
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
for (const item of result.wanted_items) {
|
2019-09-14 17:59:50 +08:00
|
|
|
let total_count = 0;
|
2019-09-11 21:51:56 +08:00
|
|
|
|
|
|
|
for (const method of result.optimal_methods) {
|
2019-09-14 17:59:50 +08:00
|
|
|
total_count += method.item_counts.get(item) || 0;
|
2019-09-11 21:51:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
columns.push({
|
|
|
|
title: item.name,
|
|
|
|
width: 80,
|
|
|
|
text_align: "right",
|
2019-09-14 17:59:50 +08:00
|
|
|
render_cell(value: OptimalMethodModel) {
|
|
|
|
const count = value.item_counts.get(item);
|
|
|
|
return count ? count.toFixed(2) : "";
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
tooltip(value: OptimalMethodModel) {
|
|
|
|
const count = value.item_counts.get(item);
|
|
|
|
return count ? count.toString() : "";
|
|
|
|
},
|
2019-09-14 17:59:50 +08:00
|
|
|
footer: {
|
|
|
|
render_cell() {
|
|
|
|
return total_count.toFixed(2);
|
|
|
|
},
|
|
|
|
tooltip() {
|
|
|
|
return total_count.toString();
|
|
|
|
},
|
2019-09-11 21:51:56 +08:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.table = new Table({
|
|
|
|
class: "hunt_optimizer_OptimizationResultView_table",
|
|
|
|
values: result ? list_property(undefined, ...result.optimal_methods) : list_property(),
|
|
|
|
columns,
|
|
|
|
});
|
|
|
|
|
|
|
|
this.element.append(this.table.element);
|
2019-09-10 00:37:20 +08:00
|
|
|
}
|
|
|
|
}
|