More data is now shown in the optimization result view.

This commit is contained in:
Daan Vanden Bosch 2019-09-11 15:51:56 +02:00
parent 0cfa20e30f
commit c9820ceccb
20 changed files with 287 additions and 171 deletions

View File

@ -1,7 +1,7 @@
.core_TabContainer_Bar { .core_TabContainer_Bar {
box-sizing: border-box; box-sizing: border-box;
padding: 3px 3px 0 3px; padding: 3px 3px 0 3px;
border-bottom: solid 1px var(--border-color); border-bottom: var(--border);
} }
.core_TabContainer_Tab { .core_TabContainer_Tab {
@ -10,7 +10,7 @@
align-items: center; align-items: center;
height: calc(100% + 1px); height: calc(100% + 1px);
padding: 0 10px; padding: 0 10px;
border: solid 1px var(--border-color); border: var(--border);
margin: 0 1px -1px 1px; margin: 0 1px -1px 1px;
background-color: hsl(0, 0%, 12%); background-color: hsl(0, 0%, 12%);
color: hsl(0, 0%, 75%); color: hsl(0, 0%, 75%);

View File

@ -1,4 +1,4 @@
import { Widget } from "./Widget"; import { Widget, WidgetOptions } from "./Widget";
import { create_element, el } from "./dom"; import { create_element, el } from "./dom";
import { LazyWidget } from "./LazyWidget"; import { LazyWidget } from "./LazyWidget";
import { Resizable } from "./Resizable"; import { Resizable } from "./Resizable";
@ -11,6 +11,10 @@ export type Tab = {
create_view: () => Promise<Widget & Resizable>; create_view: () => Promise<Widget & Resizable>;
}; };
export type TabContainerOptions = WidgetOptions & {
tabs: Tab[];
};
type TabInfo = Tab & { tab_element: HTMLSpanElement; lazy_view: LazyWidget }; type TabInfo = Tab & { tab_element: HTMLSpanElement; lazy_view: LazyWidget };
const BAR_HEIGHT = 28; const BAR_HEIGHT = 28;
@ -20,12 +24,12 @@ export class TabContainer extends ResizableWidget {
private bar_element = el.div({ class: "core_TabContainer_Bar" }); private bar_element = el.div({ class: "core_TabContainer_Bar" });
private panes_element = el.div({ class: "core_TabContainer_Panes" }); private panes_element = el.div({ class: "core_TabContainer_Panes" });
constructor(...tabs: Tab[]) { constructor(options: TabContainerOptions) {
super(el.div({ class: "core_TabContainer" })); super(el.div({ class: "core_TabContainer" }), options);
this.bar_element.onmousedown = this.bar_mousedown; this.bar_element.onmousedown = this.bar_mousedown;
for (const tab of tabs) { for (const tab of options.tabs) {
const tab_element = create_element("span", { const tab_element = create_element("span", {
class: "core_TabContainer_Tab", class: "core_TabContainer_Tab",
text: tab.title, text: tab.title,

View File

@ -24,14 +24,21 @@
top: 0; top: 0;
} }
.core_Table thead th {
display: flex;
flex-direction: column;
justify-content: center;
overflow: hidden;
}
.core_Table th, .core_Table th,
.core_Table td { .core_Table td {
box-sizing: border-box; box-sizing: border-box;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
padding: 3px 6px; padding: 3px 6px;
border-right: solid 1px var(--border-color); border-right: var(--border);
border-bottom: solid 1px var(--border-color); border-bottom: var(--border);
background-color: var(--bg-color); background-color: var(--bg-color);
} }
@ -46,6 +53,7 @@
} }
.core_Table tbody th { .core_Table tbody th {
position: sticky;
text-align: left; text-align: left;
} }

View File

@ -7,13 +7,18 @@ import {
} from "../observable/property/list/ListProperty"; } from "../observable/property/list/ListProperty";
import { Disposer } from "../observable/Disposer"; import { Disposer } from "../observable/Disposer";
import "./Table.css"; import "./Table.css";
import Logger = require("js-logger");
const logger = Logger.get("core/gui/Table");
export type Column<T> = { export type Column<T> = {
title: string; title: string;
sticky?: boolean; sticky?: boolean;
width: number; width: number;
input?: boolean;
text_align?: string; text_align?: string;
create_cell(value: T, disposer: Disposer): HTMLTableCellElement; tooltip?: (value: T) => string;
render_cell(value: T, disposer: Disposer): string | HTMLElement;
}; };
export type TableOptions<T> = WidgetOptions & { export type TableOptions<T> = WidgetOptions & {
@ -40,9 +45,7 @@ export class Table<T> extends Widget<HTMLTableElement> {
header_tr_element.append( header_tr_element.append(
...this.columns.map(column => { ...this.columns.map(column => {
const th = el.th({ const th = el.th({}, el.span({ text: column.title }));
text: column.title,
});
if (column.sticky) { if (column.sticky) {
th.style.position = "sticky"; th.style.position = "sticky";
@ -61,6 +64,8 @@ export class Table<T> extends Widget<HTMLTableElement> {
this.element.append(thead_element, this.tbody_element); this.element.append(thead_element, this.tbody_element);
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);
} }
private update_table = (change: ListPropertyChangeEvent<T>): void => { private update_table = (change: ListPropertyChangeEvent<T>): void => {
@ -98,11 +103,17 @@ export class Table<T> extends Widget<HTMLTableElement> {
return el.tr( return el.tr(
{}, {},
...this.columns.map(column => { ...this.columns.map((column, i) => {
const cell = column.create_cell(value, disposer); const cell = column.sticky ? el.th() : el.td();
try {
const content = column.render_cell(value, disposer);
cell.append(content);
if (column.input) cell.classList.add("input");
if (column.sticky) { if (column.sticky) {
cell.style.position = "sticky";
cell.style.left = `${left}px`; cell.style.left = `${left}px`;
left += column.width || 0; left += column.width || 0;
} }
@ -111,6 +122,11 @@ export class Table<T> extends Widget<HTMLTableElement> {
if (column.text_align) cell.style.textAlign = column.text_align; if (column.text_align) cell.style.textAlign = column.text_align;
if (column.tooltip) cell.title = column.tooltip(value);
} catch (e) {
logger.warn(`Error while rendering cell for index ${index}, column ${i}.`, e);
}
return cell; return cell;
}), }),
); );

View File

@ -3,7 +3,7 @@
display: flex; display: flex;
flex-direction: row; flex-direction: row;
align-items: center; align-items: center;
border-bottom: solid var(--border-color) 1px; border-bottom: var(--border);
} }
.core_ToolBar > * { .core_ToolBar > * {

View File

@ -1,6 +1,7 @@
import { Disposable } from "../observable/Disposable"; import { Disposable } from "../observable/Disposable";
import { Observable } from "../observable/Observable"; import { Observable } from "../observable/Observable";
import { is_property } from "../observable/property/Property"; import { is_property } from "../observable/property/Property";
import { SectionId } from "../model";
export const el = { export const el = {
div: ( div: (
@ -143,6 +144,20 @@ export function icon(icon: Icon): HTMLElement {
return el.span({ class: `fas ${icon_str}` }); return el.span({ class: `fas ${icon_str}` });
} }
export function section_id_icon(section_id: SectionId, options?: { size?: number }): HTMLElement {
const element = el.span();
const size = options && options.size;
element.style.display = "inline-block";
element.style.width = `${size}px`;
element.style.height = `${size}px`;
element.style.backgroundImage = `url(${process.env.PUBLIC_URL}/images/sectionids/${SectionId[section_id]}.png)`;
element.style.backgroundSize = `${size}px`;
element.title = SectionId[section_id];
return element;
}
export function disposable_listener( export function disposable_listener(
element: DocumentAndElementEventHandlers, element: DocumentAndElementEventHandlers,
event: string, event: string,

View File

@ -1,7 +1,7 @@
#root .lm_header { #root .lm_header {
box-sizing: border-box; box-sizing: border-box;
padding: 3px 0 0 0; padding: 3px 0 0 0;
border-bottom: solid 1px var(--border-color); border-bottom: var(--border);
} }
#root .lm_tabs { #root .lm_tabs {
@ -14,7 +14,7 @@
align-items: center; align-items: center;
height: 23px; height: 23px;
padding: 0 10px; padding: 0 10px;
border: solid 1px var(--border-color); border: var(--border);
margin: 0 1px -1px 1px; margin: 0 1px -1px 1px;
background-color: hsl(0, 0%, 12%); background-color: hsl(0, 0%, 12%);
color: hsl(0, 0%, 75%); color: hsl(0, 0%, 75%);
@ -38,13 +38,13 @@
} }
#root .lm_splitter.lm_vertical { #root .lm_splitter.lm_vertical {
border-top: solid 1px var(--border-color); border-top: var(--border);
border-bottom: solid 1px var(--border-color); border-bottom: var(--border);
} }
#root .lm_splitter.lm_horizontal { #root .lm_splitter.lm_horizontal {
border-left: solid 1px var(--border-color); border-left: var(--border);
border-right: solid 1px var(--border-color); border-right: var(--border);
} }
body .lm_dropTargetIndicator { body .lm_dropTargetIndicator {

View File

@ -5,7 +5,7 @@
--text-color: hsl(0, 0%, 80%); --text-color: hsl(0, 0%, 80%);
--text-color-disabled: hsl(0, 0%, 55%); --text-color-disabled: hsl(0, 0%, 55%);
--font-family: Verdana, Geneva, sans-serif; --font-family: Verdana, Geneva, sans-serif;
--border-color: hsl(0, 0%, 25%); --border: solid 1px hsl(0, 0%, 25%);
/* Scrollbars */ /* Scrollbars */
@ -69,7 +69,7 @@ body {
h2 { h2 {
font-size: 1.1em; font-size: 1.1em;
margin: 0.1em 0; margin: 0.5em 0;
} }
#root *[hidden] { #root *[hidden] {

View File

@ -2,7 +2,9 @@ import { TabContainer } from "../../core/gui/TabContainer";
export class HuntOptimizerView extends TabContainer { export class HuntOptimizerView extends TabContainer {
constructor() { constructor() {
super( super({
class: "hunt_optimizer_HuntOptimizerView",
tabs: [
{ {
title: "Optimize", title: "Optimize",
key: "optimize", key: "optimize",
@ -17,6 +19,7 @@ export class HuntOptimizerView extends TabContainer {
return new (await import("./MethodsView")).MethodsView(); return new (await import("./MethodsView")).MethodsView();
}, },
}, },
); ],
});
} }
} }

View File

@ -38,18 +38,16 @@ export class MethodsForEpisodeView extends ResizableWidget {
title: "Method", title: "Method",
sticky: true, sticky: true,
width: 250, width: 250,
create_cell(method: HuntMethodModel): HTMLTableDataCellElement { render_cell(method: HuntMethodModel) {
return el.th({ text: method.name }); return method.name;
}, },
}, },
{ {
title: "Time", title: "Time",
sticky: true, sticky: true,
width: 60, width: 60,
create_cell( input: true,
method: HuntMethodModel, render_cell(method: HuntMethodModel, disposer: Disposer) {
disposer: Disposer,
): HTMLTableDataCellElement {
const time_input = disposer.add(new DurationInput(method.time.val)); const time_input = disposer.add(new DurationInput(method.time.val));
disposer.add( disposer.add(
@ -58,7 +56,7 @@ export class MethodsForEpisodeView extends ResizableWidget {
), ),
); );
return el.th({ class: "input" }, time_input.element); return time_input.element;
}, },
}, },
...this.enemy_types.map(enemy_type => { ...this.enemy_types.map(enemy_type => {
@ -66,9 +64,9 @@ export class MethodsForEpisodeView extends ResizableWidget {
title: npc_data(enemy_type).simple_name, title: npc_data(enemy_type).simple_name,
width: 80, width: 80,
text_align: "right", text_align: "right",
create_cell(method: HuntMethodModel): HTMLTableDataCellElement { render_cell(method: HuntMethodModel) {
const count = method.enemy_counts.get(enemy_type); const count = method.enemy_counts.get(enemy_type);
return el.td({ text: count == undefined ? "" : count.toString() }); return count == undefined ? "" : count.toString();
}, },
}; };
}), }),

View File

@ -4,7 +4,9 @@ import { MethodsForEpisodeView } from "./MethodsForEpisodeView";
export class MethodsView extends TabContainer { export class MethodsView extends TabContainer {
constructor() { constructor() {
super( super({
class: "hunt_optimizer_MethodsView",
tabs: [
{ {
title: "Episode I", title: "Episode I",
key: "episode_1", key: "episode_1",
@ -26,6 +28,7 @@ export class MethodsView extends TabContainer {
return new MethodsForEpisodeView(Episode.IV); return new MethodsForEpisodeView(Episode.IV);
}, },
}, },
); ],
});
} }
} }

View File

@ -0,0 +1,10 @@
.hunt_optimizer_OptimizationResultView {
display: flex;
flex-direction: column;
}
.hunt_optimizer_OptimizationResultView_table {
flex: 1;
border-top: var(--border);
border-left: var(--border);
}

View File

@ -1,57 +1,26 @@
import { Widget } from "../../core/gui/Widget"; import { Widget } from "../../core/gui/Widget";
import { el } from "../../core/gui/dom"; import { el, section_id_icon } from "../../core/gui/dom";
import { Table } from "../../core/gui/Table"; import { Column, Table } from "../../core/gui/Table";
import { hunt_optimizer_stores } from "../stores/HuntOptimizerStore"; import { hunt_optimizer_stores } from "../stores/HuntOptimizerStore";
import { Disposable } from "../../core/observable/Disposable"; import { Disposable } from "../../core/observable/Disposable";
import { list_property } from "../../core/observable"; import { list_property } from "../../core/observable";
import { OptimalMethodModel } from "../model"; 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";
export class OptimizationResultView extends Widget { export class OptimizationResultView extends Widget {
private results_observer?: Disposable; private results_observer?: Disposable;
private table?: Table<OptimalMethodModel>;
constructor() { constructor() {
super( super(
el.div( el.div(
{ class: "hunt_optimizer_OptimizationResultView" }, { class: "hunt_optimizer_OptimizationResultView" },
el.h2({ text: "Optimization Result" }), el.h2({ text: "Ideal Combination of Methods" }),
), ),
); );
const optimal_methods = list_property<OptimalMethodModel>();
this.element.append(
this.disposable(
new Table({
values: optimal_methods,
columns: [
{
title: "Difficulty",
width: 80,
create_cell(value: OptimalMethodModel) {
return el.td({ text: Difficulty[value.difficulty] });
},
},
{
title: "Method",
width: 200,
create_cell(value: OptimalMethodModel) {
return el.td({ text: value.method_name });
},
},
{
title: "Ep.",
width: 50,
create_cell(value: OptimalMethodModel) {
return el.td({ text: Episode[value.method_episode] });
},
},
],
}),
).element,
);
this.disposable( this.disposable(
hunt_optimizer_stores.observe_current( hunt_optimizer_stores.observe_current(
hunt_optimizer_store => { hunt_optimizer_store => {
@ -60,13 +29,7 @@ export class OptimizationResultView extends Widget {
} }
this.results_observer = hunt_optimizer_store.result.observe( this.results_observer = hunt_optimizer_store.result.observe(
({ value: result }) => { ({ value }) => this.update_table(value),
if (result) {
optimal_methods.val = result.optimal_methods;
} else {
optimal_methods.val = [];
}
},
{ {
call_now: true, call_now: true,
}, },
@ -83,5 +46,117 @@ export class OptimizationResultView extends Widget {
if (this.results_observer) { if (this.results_observer) {
this.results_observer.dispose(); this.results_observer.dispose();
} }
if (this.table) {
this.table.dispose();
}
}
private update_table(result?: OptimalResultModel): void {
if (this.table) {
this.table.dispose();
}
const columns: Column<OptimalMethodModel>[] = [
{
title: "Difficulty",
sticky: true,
width: 80,
render_cell(value: OptimalMethodModel) {
return Difficulty[value.difficulty];
},
},
{
title: "Method",
sticky: true,
width: 250,
render_cell(value: OptimalMethodModel) {
return value.method_name;
},
},
{
title: "Ep.",
sticky: true,
width: 40,
render_cell(value: OptimalMethodModel) {
return Episode[value.method_episode];
},
},
{
title: "Section ID",
sticky: true,
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",
tooltip(value: OptimalMethodModel) {
return value.runs.toString();
},
render_cell(value: OptimalMethodModel) {
return value.runs.toFixed(1);
},
},
{
title: "Total Hours",
width: 60,
text_align: "right",
tooltip(value: OptimalMethodModel) {
return value.total_time.as("hours").toString();
},
render_cell(value: OptimalMethodModel) {
return value.total_time.as("hours").toFixed(1);
},
},
];
if (result) {
for (const item of result.wanted_items) {
let totalCount = 0;
for (const method of result.optimal_methods) {
totalCount += method.item_counts.get(item) || 0;
}
columns.push({
title: item.name,
width: 80,
text_align: "right",
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) : "";
},
});
}
}
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);
} }
} }

View File

@ -3,3 +3,8 @@
align-items: stretch; align-items: stretch;
overflow: hidden; overflow: hidden;
} }
.hunt_optimizer_OptimizerView div:nth-child(2) {
flex: 1;
overflow: hidden;
}

View File

@ -3,6 +3,8 @@
flex-direction: column; flex-direction: column;
align-items: stretch; align-items: stretch;
overflow: hidden; overflow: hidden;
padding-left: 6px;
min-width: 200px;
} }
.hunt_optimizer_WantedItemsView .hunt_optimizer_WantedItemsView_table_wrapper { .hunt_optimizer_WantedItemsView .hunt_optimizer_WantedItemsView_table_wrapper {
@ -11,7 +13,11 @@
overflow: auto; overflow: auto;
} }
.hunt_optimizer_WantedItemsView .hunt_optimizer_WantedItemsView_table_wrapper table { .hunt_optimizer_WantedItemsView .hunt_optimizer_WantedItemsView_table_wrapper table {
width: 100%; width: 100%;
border-collapse: collapse;
}
.hunt_optimizer_WantedItemsView .hunt_optimizer_WantedItemsView_table_wrapper td {
padding: 0 6px 3px 0;
} }

View File

@ -32,6 +32,8 @@ export class OptimalResultModel {
} }
export class OptimalMethodModel { export class OptimalMethodModel {
readonly total_time: Duration;
constructor( constructor(
readonly difficulty: Difficulty, readonly difficulty: Difficulty,
readonly section_ids: SectionId[], readonly section_ids: SectionId[],
@ -40,5 +42,7 @@ export class OptimalMethodModel {
readonly method_time: Duration, readonly method_time: Duration,
readonly runs: number, readonly runs: number,
readonly item_counts: Map<ItemType, number>, readonly item_counts: Map<ItemType, number>,
) {} ) {
this.total_time = Duration.fromMillis(runs * method_time.as("milliseconds"));
}
} }

View File

@ -1,25 +0,0 @@
import React from "react";
import { SectionId } from "../../../core/model";
export function SectionIdIcon({
section_id,
size = 28,
title,
}: {
section_id: SectionId;
size?: number;
title?: string;
}): JSX.Element {
return (
<div
title={title}
style={{
display: "inline-block",
width: size,
height: size,
backgroundImage: `url(${process.env.PUBLIC_URL}/images/sectionids/${SectionId[section_id]}.png)`,
backgroundSize: size,
}}
/>
);
}

View File

@ -1,9 +0,0 @@
/**
* @param hours can be fractional.
* @returns a string of the shape ##:##.
*/
export function hours_to_string(hours: number): string {
const h = Math.floor(hours);
const m = Math.round(60 * (hours - h));
return `${h.toString().padStart(2, "0")}:${m.toString().padStart(2, "0")}`;
}

View File

@ -2,7 +2,9 @@ import { TabContainer } from "../../core/gui/TabContainer";
export class ViewerView extends TabContainer { export class ViewerView extends TabContainer {
constructor() { constructor() {
super( super({
class: "viewer_ViewerView",
tabs: [
{ {
title: "Models", title: "Models",
key: "model", key: "model",
@ -17,6 +19,7 @@ export class ViewerView extends TabContainer {
return new (await import("./TextureView")).TextureView(); return new (await import("./TextureView")).TextureView();
}, },
}, },
); ],
});
} }
} }

View File

@ -6,8 +6,8 @@ import "./Model3DSelectListView.css";
export class Model3DSelectListView<T extends { name: string }> extends ResizableWidget { export class Model3DSelectListView<T extends { name: string }> extends ResizableWidget {
set borders(borders: boolean) { set borders(borders: boolean) {
if (borders) { if (borders) {
this.element.style.borderLeft = "solid 1px var(--border-color)"; this.element.style.borderLeft = "var(--border)";
this.element.style.borderRight = "solid 1px var(--border-color)"; this.element.style.borderRight = "var(--border)";
} else { } else {
this.element.style.borderLeft = "none"; this.element.style.borderLeft = "none";
this.element.style.borderRight = "none"; this.element.style.borderRight = "none";