From 87c6ae37e406e322b6009125a670a0341266b469 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sun, 15 Sep 2019 19:32:34 +0200 Subject: [PATCH] Widget's element property is now abstract and is overridden in every concrete subclass. This often simplifies widget code. --- src/application/gui/ApplicationView.ts | 10 ++- src/application/gui/MainContentView.ts | 4 +- src/application/gui/NavigationButton.ts | 4 +- src/application/gui/NavigationView.ts | 72 ++++++++++--------- src/core/gui/Button.ts | 9 ++- src/core/gui/CheckBox.ts | 6 +- src/core/gui/ComboBox.ts | 4 +- src/core/gui/Control.ts | 2 +- src/core/gui/DropDown.ts | 24 +++---- src/core/gui/FileButton.ts | 13 ++-- src/core/gui/Input.ts | 8 ++- src/core/gui/Label.ts | 6 +- src/core/gui/LabelledControl.ts | 6 +- src/core/gui/LazyWidget.ts | 4 +- src/core/gui/Menu.ts | 3 +- src/core/gui/RendererWidget.ts | 6 +- src/core/gui/ResizableWidget.ts | 3 +- src/core/gui/Select.ts | 22 +++--- src/core/gui/TabContainer.ts | 4 +- src/core/gui/Table.ts | 6 +- src/core/gui/TextArea.ts | 4 +- src/core/gui/ToolBar.ts | 3 +- src/core/gui/Widget.ts | 17 +++-- src/hunt_optimizer/gui/HelpView.ts | 37 +++++----- .../gui/MethodsForEpisodeView.ts | 4 +- .../gui/OptimizationResultView.ts | 12 ++-- src/hunt_optimizer/gui/OptimizerView.ts | 4 +- src/hunt_optimizer/gui/WantedItemsView.ts | 4 +- src/quest_editor/gui/AsmEditorView.ts | 4 +- src/quest_editor/gui/DisabledView.ts | 4 +- src/quest_editor/gui/EntityInfoView.ts | 4 +- src/quest_editor/gui/NpcCountsView.ts | 4 +- src/quest_editor/gui/QuestEditorToolBar.ts | 1 - src/quest_editor/gui/QuestEditorView.ts | 4 +- src/quest_editor/gui/QuestInfoView.ts | 4 +- src/quest_editor/gui/QuestRendererView.ts | 4 +- src/viewer/gui/TextureView.ts | 4 +- .../gui/model_3d/Model3DSelectListView.ts | 4 +- src/viewer/gui/model_3d/Model3DView.ts | 18 +++-- 39 files changed, 207 insertions(+), 149 deletions(-) diff --git a/src/application/gui/ApplicationView.ts b/src/application/gui/ApplicationView.ts index a1a73b47..6146cd4f 100644 --- a/src/application/gui/ApplicationView.ts +++ b/src/application/gui/ApplicationView.ts @@ -7,13 +7,17 @@ export class ApplicationView extends ResizableWidget { private menu_view = this.disposable(new NavigationView()); private main_content_view = this.disposable(new MainContentView()); + readonly element = el.div( + { class: "application_ApplicationView" }, + this.menu_view.element, + this.main_content_view.element, + ); + constructor() { - super(el.div({ class: "application_ApplicationView" })); + super(); this.element.id = "root"; - this.element.append(this.menu_view.element, this.main_content_view.element); - this.finalize_construction(ApplicationView.prototype); } diff --git a/src/application/gui/MainContentView.ts b/src/application/gui/MainContentView.ts index 4a208bec..9fa7ec6b 100644 --- a/src/application/gui/MainContentView.ts +++ b/src/application/gui/MainContentView.ts @@ -18,12 +18,14 @@ const TOOLS: [GuiTool, () => Promise][] = [ ]; export class MainContentView extends ResizableWidget { + readonly element = el.div({ class: "application_MainContentView" }); + private tool_views = new Map( TOOLS.map(([tool, create_view]) => [tool, this.disposable(new LazyWidget(create_view))]), ); constructor() { - super(el.div({ class: "application_MainContentView" })); + super(); for (const tool_view of this.tool_views.values()) { this.element.append(tool_view.element); diff --git a/src/application/gui/NavigationButton.ts b/src/application/gui/NavigationButton.ts index 98c8c6dc..b2240b1b 100644 --- a/src/application/gui/NavigationButton.ts +++ b/src/application/gui/NavigationButton.ts @@ -4,11 +4,13 @@ import { GuiTool } from "../../core/stores/GuiStore"; import "./NavigationButton.css"; export class NavigationButton extends Widget { + readonly element = el.span({ class: "application_NavigationButton" }); + private input: HTMLInputElement = create_element("input"); private label: HTMLLabelElement = create_element("label"); constructor(tool: GuiTool, text: string) { - super(el.span({ class: "application_NavigationButton" })); + super(); const tool_str = GuiTool[tool]; diff --git a/src/application/gui/NavigationView.ts b/src/application/gui/NavigationView.ts index f2741086..e686df8b 100644 --- a/src/application/gui/NavigationView.ts +++ b/src/application/gui/NavigationView.ts @@ -13,49 +13,51 @@ const TOOLS: [GuiTool, string][] = [ ]; export class NavigationView extends Widget { - readonly height = 30; - - private buttons = new Map( + private readonly buttons = new Map( TOOLS.map(([value, text]) => [value, this.disposable(new NavigationButton(value, text))]), ); + private readonly server_select = this.disposable( + new Select(property(["Ephinea"]), server => server, { + label: "Server:", + enabled: false, + selected: "Ephinea", + tooltip: "Only Ephinea is supported at the moment", + }), + ); + + readonly element = el.div( + { class: "application_NavigationView" }, + + ...[...this.buttons.values()].map(button => button.element), + + el.div({ class: "application_NavigationView_spacer" }), + + this.server_select.element, + + el.span( + { class: "application_NavigationView_server" }, + this.server_select.label!.element, + this.server_select.element, + ), + + el.a( + { + class: "application_NavigationView_github", + href: "https://github.com/DaanVandenBosch/phantasmal-world", + title: "GitHub", + }, + icon(Icon.GitHub), + ), + ); + + readonly height = 30; constructor() { - super(el.div({ class: "application_NavigationView" })); + super(); this.element.style.height = `${this.height}px`; this.element.onmousedown = this.mousedown; - for (const button of this.buttons.values()) { - this.element.append(button.element); - } - - this.element.append(el.div({ class: "application_NavigationView_spacer" })); - - const server_select = this.disposable( - new Select(property(["Ephinea"]), server => server, { - label: "Server:", - enabled: false, - selected: "Ephinea", - tooltip: "Only Ephinea is supported at the moment", - }), - ); - - this.element.append( - el.span( - { class: "application_NavigationView_server" }, - server_select.label!.element, - server_select.element, - ), - el.a( - { - class: "application_NavigationView_github", - href: "https://github.com/DaanVandenBosch/phantasmal-world", - title: "GitHub", - }, - icon(Icon.GitHub), - ), - ); - this.mark_tool_button(gui_store.tool.val); this.disposable(gui_store.tool.observe(({ value }) => this.mark_tool_button(value))); diff --git a/src/core/gui/Button.ts b/src/core/gui/Button.ts index 2774c587..b3f06868 100644 --- a/src/core/gui/Button.ts +++ b/src/core/gui/Button.ts @@ -14,7 +14,8 @@ export type ButtonOptions = WidgetOptions & { icon_right?: Icon; }; -export class Button extends Control { +export class Button extends Control { + readonly element = el.button({ class: "core_Button" }); readonly mousedown: Observable; readonly mouseup: Observable; readonly click: Observable; @@ -27,9 +28,9 @@ export class Button extends Control { private readonly center_element: HTMLSpanElement; constructor(text: string | Property, options?: ButtonOptions) { - const inner_element = el.span({ class: "core_Button_inner" }); + super(options); - super(el.button({ class: "core_Button" }, inner_element), options); + const inner_element = el.span({ class: "core_Button_inner" }); this.center_element = el.span({ class: "core_Button_center" }); @@ -64,6 +65,8 @@ export class Button extends Control { this.text.bind_to(text); } + this.element.append(inner_element); + this.finalize_construction(Button.prototype); } diff --git a/src/core/gui/CheckBox.ts b/src/core/gui/CheckBox.ts index 9e9dad08..eb0ed1ca 100644 --- a/src/core/gui/CheckBox.ts +++ b/src/core/gui/CheckBox.ts @@ -5,7 +5,9 @@ import { WidgetProperty } from "../observable/property/WidgetProperty"; export type CheckBoxOptions = LabelledControlOptions; -export class CheckBox extends LabelledControl { +export class CheckBox extends LabelledControl { + readonly element = create_element("input", { class: "core_CheckBox" }); + readonly preferred_label_position = "right"; readonly checked: WritableProperty; @@ -13,7 +15,7 @@ export class CheckBox extends LabelledControl { private readonly _checked: WidgetProperty; constructor(checked: boolean = false, options?: CheckBoxOptions) { - super(create_element("input", { class: "core_CheckBox" }), options); + super(options); this._checked = new WidgetProperty(this, checked, this.set_checked); this.checked = this._checked; diff --git a/src/core/gui/ComboBox.ts b/src/core/gui/ComboBox.ts index f43a23f1..12790a21 100644 --- a/src/core/gui/ComboBox.ts +++ b/src/core/gui/ComboBox.ts @@ -16,6 +16,8 @@ export type ComboBoxOptions = LabelledControlOptions & { }; export class ComboBox extends LabelledControl { + readonly element = el.span({ class: "core_ComboBox core_Input" }); + readonly preferred_label_position = "left"; readonly selected: WritableProperty; @@ -26,7 +28,7 @@ export class ComboBox extends LabelledControl { private readonly _selected: WidgetProperty; constructor(options: ComboBoxOptions) { - super(el.span({ class: "core_ComboBox core_Input" }), options); + super(options); this.to_label = options.to_label; diff --git a/src/core/gui/Control.ts b/src/core/gui/Control.ts index 8f509507..e680f13b 100644 --- a/src/core/gui/Control.ts +++ b/src/core/gui/Control.ts @@ -2,4 +2,4 @@ import { Widget, WidgetOptions } from "./Widget"; export type ControlOptions = WidgetOptions; -export abstract class Control extends Widget {} +export abstract class Control extends Widget {} diff --git a/src/core/gui/DropDown.ts b/src/core/gui/DropDown.ts index 0f092ea9..9b45f074 100644 --- a/src/core/gui/DropDown.ts +++ b/src/core/gui/DropDown.ts @@ -11,6 +11,8 @@ import { emitter } from "../observable"; export type DropDownOptions = ButtonOptions; export class DropDown extends Control { + readonly element = el.div({ class: "core_DropDown" }); + readonly chosen: Observable; private readonly button: Button; @@ -24,17 +26,15 @@ export class DropDown extends Control { to_label: (element: T) => string, options?: DropDownOptions, ) { - const element = el.div({ class: "core_DropDown" }); - const button = new Button(text, { - icon_left: options && options.icon_left, - icon_right: Icon.TriangleDown, - }); - const menu = new Menu(items, to_label, element); + super(options); - super(element, options); - - this.button = this.disposable(button); - this.menu = this.disposable(menu); + this.button = this.disposable( + new Button(text, { + icon_left: options && options.icon_left, + icon_right: Icon.TriangleDown, + }), + ); + this.menu = this.disposable(new Menu(items, to_label, this.element)); this.element.append(this.button.element, this.menu.element); this._chosen = emitter(); @@ -43,11 +43,11 @@ export class DropDown extends Control { this.just_opened = false; this.disposables( - disposable_listener(button.element, "mousedown", () => this.button_mousedown(), { + disposable_listener(this.button.element, "mousedown", () => this.button_mousedown(), { capture: true, }), - button.mouseup.observe(() => this.button_mouseup()), + this.button.mouseup.observe(() => this.button_mouseup()), this.menu.selected.observe(({ value }) => { if (value) { diff --git a/src/core/gui/FileButton.ts b/src/core/gui/FileButton.ts index 43266868..54dfdaec 100644 --- a/src/core/gui/FileButton.ts +++ b/src/core/gui/FileButton.ts @@ -11,7 +11,11 @@ export type FileButtonOptions = ControlOptions & { icon_left?: Icon; }; -export class FileButton extends Control { +export class FileButton extends Control { + readonly element = create_element("label", { + class: "core_FileButton core_Button", + }); + readonly files: Property; private input: HTMLInputElement = create_element("input", { @@ -21,12 +25,7 @@ export class FileButton extends Control { private readonly _files: WritableProperty = property([]); constructor(text: string, options?: FileButtonOptions) { - super( - create_element("label", { - class: "core_FileButton core_Button", - }), - options, - ); + super(options); this.files = this._files; diff --git a/src/core/gui/Input.ts b/src/core/gui/Input.ts index a0325836..4036602e 100644 --- a/src/core/gui/Input.ts +++ b/src/core/gui/Input.ts @@ -8,7 +8,9 @@ import { WidgetProperty } from "../observable/property/WidgetProperty"; export type InputOptions = LabelledControlOptions; -export abstract class Input extends LabelledControl { +export abstract class Input extends LabelledControl { + readonly element: HTMLElement; + readonly value: WritableProperty; protected readonly input_element: HTMLInputElement; @@ -22,7 +24,9 @@ export abstract class Input extends LabelledControl { input_class_name: string, options?: InputOptions, ) { - super(el.span({ class: `${class_name} core_Input` }), options); + super(options); + + this.element = el.span({ class: `${class_name} core_Input` }); this._value = new WidgetProperty(this, value, this.set_value); this.value = this._value; diff --git a/src/core/gui/Label.ts b/src/core/gui/Label.ts index efb6c204..8b7cc288 100644 --- a/src/core/gui/Label.ts +++ b/src/core/gui/Label.ts @@ -5,7 +5,9 @@ import "./Label.css"; import { Property } from "../observable/property/Property"; import { WidgetProperty } from "../observable/property/WidgetProperty"; -export class Label extends Widget { +export class Label extends Widget { + readonly element = create_element("label", { class: "core_Label" }); + set for(id: string) { this.element.htmlFor = id; } @@ -15,7 +17,7 @@ export class Label extends Widget { private readonly _text = new WidgetProperty(this, "", this.set_text); constructor(text: string | Property, options?: WidgetOptions) { - super(create_element("label", { class: "core_Label" }), options); + super(options); this.text = this._text; diff --git a/src/core/gui/LabelledControl.ts b/src/core/gui/LabelledControl.ts index 026e0e39..04bd44a2 100644 --- a/src/core/gui/LabelledControl.ts +++ b/src/core/gui/LabelledControl.ts @@ -8,7 +8,7 @@ export type LabelledControlOptions = WidgetOptions & { export type LabelPosition = "left" | "right" | "top" | "bottom"; -export abstract class LabelledControl extends Control { +export abstract class LabelledControl extends Control { abstract readonly preferred_label_position: LabelPosition; get label(): Label | undefined { @@ -28,8 +28,8 @@ export abstract class LabelledControl exten private readonly _label_text?: string; private _label?: Label; - protected constructor(element: E, options?: LabelledControlOptions) { - super(element, options); + protected constructor(options?: LabelledControlOptions) { + super(options); this._label_text = options && options.label; } diff --git a/src/core/gui/LazyWidget.ts b/src/core/gui/LazyWidget.ts index e326d4f0..6c443932 100644 --- a/src/core/gui/LazyWidget.ts +++ b/src/core/gui/LazyWidget.ts @@ -4,11 +4,13 @@ import { Resizable } from "./Resizable"; import { ResizableWidget } from "./ResizableWidget"; export class LazyWidget extends ResizableWidget { + readonly element = el.div({ class: "core_LazyView" }); + private initialized = false; private view: Widget & Resizable | undefined; constructor(private create_view: () => Promise) { - super(el.div({ class: "core_LazyView" })); + super(); this.visible.val = false; } diff --git a/src/core/gui/Menu.ts b/src/core/gui/Menu.ts index 61781866..47380279 100644 --- a/src/core/gui/Menu.ts +++ b/src/core/gui/Menu.ts @@ -7,6 +7,7 @@ import { WidgetProperty } from "../observable/property/WidgetProperty"; import "./Menu.css"; export class Menu extends Widget { + readonly element = el.div({ class: "core_Menu", tab_index: -1 }); readonly selected: WritableProperty; private readonly to_label: (element: T) => string; @@ -22,7 +23,7 @@ export class Menu extends Widget { to_label: (element: T) => string, related_element: HTMLElement, ) { - super(el.div({ class: "core_Menu", tab_index: -1 })); + super(); this.visible.val = false; diff --git a/src/core/gui/RendererWidget.ts b/src/core/gui/RendererWidget.ts index 47c41619..3382ffdb 100644 --- a/src/core/gui/RendererWidget.ts +++ b/src/core/gui/RendererWidget.ts @@ -1,10 +1,12 @@ import { ResizableWidget } from "./ResizableWidget"; -import { create_element } from "./dom"; +import { el } from "./dom"; import { Renderer } from "../rendering/Renderer"; export class RendererWidget extends ResizableWidget { + readonly element = el.div(); + constructor(private renderer: Renderer) { - super(create_element("div")); + super(); this.element.append(renderer.dom_element); diff --git a/src/core/gui/ResizableWidget.ts b/src/core/gui/ResizableWidget.ts index b9d596ec..1221a8e2 100644 --- a/src/core/gui/ResizableWidget.ts +++ b/src/core/gui/ResizableWidget.ts @@ -1,8 +1,7 @@ import { Widget } from "./Widget"; import { Resizable } from "./Resizable"; -export abstract class ResizableWidget extends Widget - implements Resizable { +export abstract class ResizableWidget extends Widget implements Resizable { protected width: number = 0; protected height: number = 0; diff --git a/src/core/gui/Select.ts b/src/core/gui/Select.ts index 089b1e65..ba8f19fb 100644 --- a/src/core/gui/Select.ts +++ b/src/core/gui/Select.ts @@ -12,6 +12,8 @@ export type SelectOptions = LabelledControlOptions & { }; export class Select extends LabelledControl { + readonly element = el.div({ class: "core_Select" }); + readonly preferred_label_position: LabelPosition; readonly selected: WritableProperty; @@ -27,19 +29,17 @@ export class Select extends LabelledControl { to_label: (element: T) => string, options?: SelectOptions, ) { - const element = el.div({ class: "core_Select" }); - const button = new Button(" ", { - icon_right: Icon.TriangleDown, - }); - const menu = new Menu(items, to_label, element); - - super(element, options); + super(options); this.preferred_label_position = "left"; this.to_label = to_label; - this.button = this.disposable(button); - this.menu = this.disposable(menu); + this.button = this.disposable( + new Button(" ", { + icon_right: Icon.TriangleDown, + }), + ); + this.menu = this.disposable(new Menu(items, to_label, this.element)); this.element.append(this.button.element, this.menu.element); this._selected = new WidgetProperty(this, undefined, this.set_selected); @@ -48,9 +48,9 @@ export class Select extends LabelledControl { this.just_opened = false; this.disposables( - disposable_listener(button.element, "mousedown", e => this.button_mousedown(e)), + disposable_listener(this.button.element, "mousedown", e => this.button_mousedown(e)), - button.mouseup.observe(() => this.button_mouseup()), + this.button.mouseup.observe(() => this.button_mouseup()), this.menu.selected.observe(({ value }) => this._selected.set_val(value, { silent: false }), diff --git a/src/core/gui/TabContainer.ts b/src/core/gui/TabContainer.ts index 9c666ade..c0dc8a17 100644 --- a/src/core/gui/TabContainer.ts +++ b/src/core/gui/TabContainer.ts @@ -20,12 +20,14 @@ type TabInfo = Tab & { tab_element: HTMLSpanElement; lazy_view: LazyWidget }; const BAR_HEIGHT = 28; export class TabContainer extends ResizableWidget { + readonly element = el.div({ class: "core_TabContainer" }); + private tabs: TabInfo[] = []; private bar_element = el.div({ class: "core_TabContainer_Bar" }); private panes_element = el.div({ class: "core_TabContainer_Panes" }); constructor(options: TabContainerOptions) { - super(el.div({ class: "core_TabContainer" }), options); + super(options); this.bar_element.onmousedown = this.bar_mousedown; diff --git a/src/core/gui/Table.ts b/src/core/gui/Table.ts index f4d42399..2c229d7f 100644 --- a/src/core/gui/Table.ts +++ b/src/core/gui/Table.ts @@ -38,7 +38,9 @@ export type TableOptions = WidgetOptions & { sort?(sort_columns: { column: Column; direction: SortDirection }[]): void; }; -export class Table extends Widget { +export class Table extends Widget { + readonly element = el.table({ class: "core_Table" }); + private readonly table_disposer = this.disposable(new Disposer()); private readonly tbody_element = el.tbody(); private readonly footer_row_element?: HTMLTableRowElement; @@ -46,7 +48,7 @@ export class Table extends Widget { private readonly columns: Column[]; constructor(options: TableOptions) { - super(el.table({ class: "core_Table" }), options); + super(options); this.values = options.values; this.columns = options.columns; diff --git a/src/core/gui/TextArea.ts b/src/core/gui/TextArea.ts index 7c51f2a0..b3ff0f89 100644 --- a/src/core/gui/TextArea.ts +++ b/src/core/gui/TextArea.ts @@ -12,6 +12,8 @@ export type TextAreaOptions = LabelledControlOptions & { }; export class TextArea extends LabelledControl { + readonly element = el.div({ class: "core_TextArea" }); + readonly preferred_label_position = "left"; readonly value: WritableProperty; @@ -23,7 +25,7 @@ export class TextArea extends LabelledControl { private readonly _value = new WidgetProperty(this, "", this.set_value); constructor(value = "", options?: TextAreaOptions) { - super(el.div({ class: "core_TextArea" }), options); + super(options); if (options) { if (options.max_length != undefined) this.text_element.maxLength = options.max_length; diff --git a/src/core/gui/ToolBar.ts b/src/core/gui/ToolBar.ts index f2f73cb0..55527ea0 100644 --- a/src/core/gui/ToolBar.ts +++ b/src/core/gui/ToolBar.ts @@ -8,10 +8,11 @@ export type ToolBarOptions = WidgetOptions & { }; export class ToolBar extends Widget { + readonly element = create_element("div", { class: "core_ToolBar" }); readonly height = 33; constructor(options?: ToolBarOptions) { - super(create_element("div", { class: "core_ToolBar" }), options); + super(options); this.element.style.height = `${this.height}px`; diff --git a/src/core/gui/Widget.ts b/src/core/gui/Widget.ts index 4eec6bcb..2cfb3c96 100644 --- a/src/core/gui/Widget.ts +++ b/src/core/gui/Widget.ts @@ -15,8 +15,8 @@ export type WidgetOptions = { tooltip?: string | Property; }; -export abstract class Widget implements Disposable { - readonly element: E; +export abstract class Widget implements Disposable { + abstract readonly element: HTMLElement; get id(): string { return this.element.id; @@ -51,18 +51,13 @@ export abstract class Widget implements Dis private readonly options: WidgetOptions; private construction_finalized = false; - protected constructor(element: E, options?: WidgetOptions) { - this.element = element; + protected constructor(options?: WidgetOptions) { this.visible = this._visible; this.enabled = this._enabled; this.tooltip = this._tooltip; this.options = options || {}; - if (this.options.class) { - this.element.classList.add(this.options.class); - } - setTimeout(() => { if (!this.construction_finalized) { logger.warn( @@ -87,7 +82,9 @@ export abstract class Widget implements Dis protected finalize_construction(proto: any): void { if (Object.getPrototypeOf(this) !== proto) return; - this.construction_finalized = true; + if (this.options.class) { + this.element.classList.add(this.options.class); + } if (typeof this.options.enabled === "boolean") { this.enabled.val = this.options.enabled; @@ -100,6 +97,8 @@ export abstract class Widget implements Dis } else if (this.options.tooltip) { this.tooltip.bind_to(this.options.tooltip); } + + this.construction_finalized = true; } protected set_visible(visible: boolean): void { diff --git a/src/hunt_optimizer/gui/HelpView.ts b/src/hunt_optimizer/gui/HelpView.ts index dc533150..05a40ce7 100644 --- a/src/hunt_optimizer/gui/HelpView.ts +++ b/src/hunt_optimizer/gui/HelpView.ts @@ -3,26 +3,25 @@ import { ResizableWidget } from "../../core/gui/ResizableWidget"; import "./HelpView.css"; export class HelpView extends ResizableWidget { - constructor() { - super( - el.div( - { class: "hunt_optimizer_HelpView" }, - el.p({ - text: - "Add some items with the combo box on the left to see the optimal combination of hunt methods on the right.", - }), - el.p({ - text: - 'At the moment a hunt method is simply a quest run-through. Partial quest run-throughs are coming. View the list of methods on the "Methods" tab. Each method takes a certain amount of time, which affects the optimization result. Make sure the times are correct for you.', - }), - el.p({ text: "Only enemy drops are considered. Box drops are coming." }), - el.p({ - text: - "The optimal result is calculated using linear optimization. The optimizer takes into account rare enemies and the fact that pan arms can be split in two.", - }), - ), - ); + readonly element = el.div( + { class: "hunt_optimizer_HelpView" }, + el.p({ + text: + "Add some items with the combo box on the left to see the optimal combination of hunt methods on the right.", + }), + el.p({ + text: + 'At the moment a hunt method is simply a quest run-through. Partial quest run-throughs are coming. View the list of methods on the "Methods" tab. Each method takes a certain amount of time, which affects the optimization result. Make sure the times are correct for you.', + }), + el.p({ text: "Only enemy drops are considered. Box drops are coming." }), + el.p({ + text: + "The optimal result is calculated using linear optimization. The optimizer takes into account rare enemies and the fact that pan arms can be split in two.", + }), + ); + constructor() { + super(); this.finalize_construction(HelpView.prototype); } } diff --git a/src/hunt_optimizer/gui/MethodsForEpisodeView.ts b/src/hunt_optimizer/gui/MethodsForEpisodeView.ts index 8826cfef..e1df71b3 100644 --- a/src/hunt_optimizer/gui/MethodsForEpisodeView.ts +++ b/src/hunt_optimizer/gui/MethodsForEpisodeView.ts @@ -16,12 +16,14 @@ import { SortDirection, Table } from "../../core/gui/Table"; import { list_property } from "../../core/observable"; export class MethodsForEpisodeView extends ResizableWidget { + readonly element = el.div({ class: "hunt_optimizer_MethodsForEpisodeView" }); + private readonly episode: Episode; private readonly enemy_types: NpcType[]; private hunt_methods_observer?: Disposable; constructor(episode: Episode) { - super(el.div({ class: "hunt_optimizer_MethodsForEpisodeView" })); + super(); this.episode = episode; diff --git a/src/hunt_optimizer/gui/OptimizationResultView.ts b/src/hunt_optimizer/gui/OptimizationResultView.ts index 35062e1b..013b6906 100644 --- a/src/hunt_optimizer/gui/OptimizationResultView.ts +++ b/src/hunt_optimizer/gui/OptimizationResultView.ts @@ -11,16 +11,16 @@ import "./OptimizationResultView.css"; import { Duration } from "luxon"; export class OptimizationResultView extends Widget { + readonly element = el.div( + { class: "hunt_optimizer_OptimizationResultView" }, + el.h2({ text: "Ideal Combination of Methods" }), + ); + private results_observer?: Disposable; private table?: Table; constructor() { - super( - el.div( - { class: "hunt_optimizer_OptimizationResultView" }, - el.h2({ text: "Ideal Combination of Methods" }), - ), - ); + super(); this.disposable( hunt_optimizer_stores.observe_current( diff --git a/src/hunt_optimizer/gui/OptimizerView.ts b/src/hunt_optimizer/gui/OptimizerView.ts index eeb17746..f02043e2 100644 --- a/src/hunt_optimizer/gui/OptimizerView.ts +++ b/src/hunt_optimizer/gui/OptimizerView.ts @@ -5,8 +5,10 @@ import "./OptimizerView.css"; import { OptimizationResultView } from "./OptimizationResultView"; export class OptimizerView extends ResizableWidget { + readonly element = el.div({ class: "hunt_optimizer_OptimizerView" }); + constructor() { - super(el.div({ class: "hunt_optimizer_OptimizerView" })); + super(); this.element.append( this.disposable(new WantedItemsView()).element, diff --git a/src/hunt_optimizer/gui/WantedItemsView.ts b/src/hunt_optimizer/gui/WantedItemsView.ts index 049e33e0..ee3614b0 100644 --- a/src/hunt_optimizer/gui/WantedItemsView.ts +++ b/src/hunt_optimizer/gui/WantedItemsView.ts @@ -15,12 +15,14 @@ import { list_property } from "../../core/observable"; import { ItemType } from "../../core/model/items"; export class WantedItemsView extends Widget { + readonly element = el.div({ class: "hunt_optimizer_WantedItemsView" }); + private readonly tbody_element = el.tbody(); private readonly table_disposer = this.disposable(new Disposer()); private readonly store_disposer = this.disposable(new Disposer()); constructor() { - super(el.div({ class: "hunt_optimizer_WantedItemsView" })); + super(); const huntable_items = list_property(); const filtered_huntable_items = list_property(); diff --git a/src/quest_editor/gui/AsmEditorView.ts b/src/quest_editor/gui/AsmEditorView.ts index d8301319..e1f2e394 100644 --- a/src/quest_editor/gui/AsmEditorView.ts +++ b/src/quest_editor/gui/AsmEditorView.ts @@ -26,10 +26,12 @@ editor.defineTheme("phantasmal-world", { const DUMMY_MODEL = editor.createModel("", "psoasm"); export class AsmEditorView extends ResizableWidget { + readonly element = el.div(); + private readonly editor: IStandaloneCodeEditor; constructor() { - super(el.div()); + super(); this.editor = this.disposable( editor.create(this.element, { diff --git a/src/quest_editor/gui/DisabledView.ts b/src/quest_editor/gui/DisabledView.ts index 3247093d..67f90c08 100644 --- a/src/quest_editor/gui/DisabledView.ts +++ b/src/quest_editor/gui/DisabledView.ts @@ -4,10 +4,12 @@ import { Label } from "../../core/gui/Label"; import "./DisabledView.css"; export class DisabledView extends Widget { + readonly element = el.div({ class: "quest_editor_DisabledView" }); + private readonly label: Label; constructor(text: string) { - super(el.div({ class: "quest_editor_DisabledView" })); + super(); this.label = this.disposable(new Label(text, { enabled: false })); diff --git a/src/quest_editor/gui/EntityInfoView.ts b/src/quest_editor/gui/EntityInfoView.ts index 46c28585..c1e8ea8f 100644 --- a/src/quest_editor/gui/EntityInfoView.ts +++ b/src/quest_editor/gui/EntityInfoView.ts @@ -12,6 +12,8 @@ import { Vec3 } from "../../core/data_formats/vector"; import { QuestEntityModel } from "../model/QuestEntityModel"; export class EntityInfoView extends ResizableWidget { + readonly element = el.div({ class: "quest_editor_EntityInfoView", tab_index: -1 }); + private readonly no_entity_view = new DisabledView("No entity selected."); private readonly table_element = el.table(); @@ -41,7 +43,7 @@ export class EntityInfoView extends ResizableWidget { private readonly entity_disposer = new Disposer(); constructor() { - super(el.div({ class: "quest_editor_EntityInfoView", tab_index: -1 })); + super(); const entity = quest_editor_store.selected_entity; const no_entity = entity.map(e => e == undefined); diff --git a/src/quest_editor/gui/NpcCountsView.ts b/src/quest_editor/gui/NpcCountsView.ts index 8029d641..b4fa5f64 100644 --- a/src/quest_editor/gui/NpcCountsView.ts +++ b/src/quest_editor/gui/NpcCountsView.ts @@ -7,12 +7,14 @@ import "./NpcCountsView.css"; import { DisabledView } from "./DisabledView"; export class NpcCountsView extends ResizableWidget { + readonly element = el.div({ class: "quest_editor_NpcCountsView" }); + private readonly table_element = el.table(); private readonly no_quest_view = new DisabledView("No quest loaded."); constructor() { - super(el.div({ class: "quest_editor_NpcCountsView" })); + super(); this.element.append(this.table_element, this.no_quest_view.element); diff --git a/src/quest_editor/gui/QuestEditorToolBar.ts b/src/quest_editor/gui/QuestEditorToolBar.ts index dbfddf99..916599eb 100644 --- a/src/quest_editor/gui/QuestEditorToolBar.ts +++ b/src/quest_editor/gui/QuestEditorToolBar.ts @@ -11,7 +11,6 @@ import { DropDown } from "../../core/gui/DropDown"; import { Episode } from "../../core/data_formats/parsing/quest/Episode"; import { area_store } from "../stores/AreaStore"; import { gui_store, GuiTool } from "../../core/stores/GuiStore"; -import { asm_editor_store } from "../stores/AsmEditorStore"; export class QuestEditorToolBar extends ToolBar { constructor() { diff --git a/src/quest_editor/gui/QuestEditorView.ts b/src/quest_editor/gui/QuestEditorView.ts index 0f08f2fd..1a9f8faf 100644 --- a/src/quest_editor/gui/QuestEditorView.ts +++ b/src/quest_editor/gui/QuestEditorView.ts @@ -94,6 +94,8 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [ ]; export class QuestEditorView extends ResizableWidget { + readonly element = el.div({ class: "quest_editor_QuestEditorView" }); + private readonly tool_bar_view = this.disposable(new QuestEditorToolBar()); private readonly layout_element = create_element("div", { class: "quest_editor_gl_container" }); @@ -102,7 +104,7 @@ export class QuestEditorView extends ResizableWidget { private readonly sub_views = new Map(); constructor() { - super(el.div({ class: "quest_editor_QuestEditorView" })); + super(); this.element.append(this.tool_bar_view.element, this.layout_element); diff --git a/src/quest_editor/gui/QuestInfoView.ts b/src/quest_editor/gui/QuestInfoView.ts index 5b62073d..1c665fbf 100644 --- a/src/quest_editor/gui/QuestInfoView.ts +++ b/src/quest_editor/gui/QuestInfoView.ts @@ -10,6 +10,8 @@ import "./QuestInfoView.css"; import { DisabledView } from "./DisabledView"; export class QuestInfoView extends ResizableWidget { + readonly element = el.div({ class: "quest_editor_QuestInfoView", tab_index: -1 }); + private readonly table_element = el.table(); private readonly episode_element: HTMLElement; private readonly id_input = this.disposable(new NumberInput()); @@ -40,7 +42,7 @@ export class QuestInfoView extends ResizableWidget { private readonly quest_disposer = this.disposable(new Disposer()); constructor() { - super(el.div({ class: "quest_editor_QuestInfoView", tab_index: -1 })); + super(); const quest = quest_editor_store.current_quest; const no_quest = quest.map(q => q == undefined); diff --git a/src/quest_editor/gui/QuestRendererView.ts b/src/quest_editor/gui/QuestRendererView.ts index 24866166..6a2b4391 100644 --- a/src/quest_editor/gui/QuestRendererView.ts +++ b/src/quest_editor/gui/QuestRendererView.ts @@ -6,10 +6,12 @@ import { gui_store, GuiTool } from "../../core/stores/GuiStore"; import { quest_editor_store } from "../stores/QuestEditorStore"; export class QuestRendererView extends ResizableWidget { + readonly element = el.div({ class: "quest_editor_QuestRendererView", tab_index: -1 }); + private renderer_view = this.disposable(new RendererWidget(new QuestRenderer())); constructor() { - super(el.div({ class: "quest_editor_QuestRendererView", tab_index: -1 })); + super(); this.element.append(this.renderer_view.element); diff --git a/src/viewer/gui/TextureView.ts b/src/viewer/gui/TextureView.ts index cf49830b..d5f07e25 100644 --- a/src/viewer/gui/TextureView.ts +++ b/src/viewer/gui/TextureView.ts @@ -8,6 +8,8 @@ import { TextureRenderer } from "../rendering/TextureRenderer"; import { gui_store, GuiTool } from "../../core/stores/GuiStore"; export class TextureView extends ResizableWidget { + readonly element = el.div({ class: "viewer_TextureView" }); + private readonly open_file_button = new FileButton("Open file...", { icon_left: Icon.File, accept: ".xvm", @@ -18,7 +20,7 @@ export class TextureView extends ResizableWidget { private readonly renderer_view = this.disposable(new RendererWidget(new TextureRenderer())); constructor() { - super(el.div({ class: "viewer_TextureView" })); + super(); this.element.append(this.tool_bar.element, this.renderer_view.element); diff --git a/src/viewer/gui/model_3d/Model3DSelectListView.ts b/src/viewer/gui/model_3d/Model3DSelectListView.ts index d8357884..ab244900 100644 --- a/src/viewer/gui/model_3d/Model3DSelectListView.ts +++ b/src/viewer/gui/model_3d/Model3DSelectListView.ts @@ -4,6 +4,8 @@ import { WritableProperty } from "../../../core/observable/property/WritableProp import "./Model3DSelectListView.css"; export class Model3DSelectListView extends ResizableWidget { + readonly element = create_element("ul", { class: "viewer_Model3DSelectListView" }); + set borders(borders: boolean) { if (borders) { this.element.style.borderLeft = "var(--border)"; @@ -18,7 +20,7 @@ export class Model3DSelectListView extends Resizable private selected_element?: HTMLLIElement; constructor(private models: T[], private selected: WritableProperty) { - super(create_element("ul", { class: "viewer_Model3DSelectListView" })); + super(); this.element.onclick = this.list_click; diff --git a/src/viewer/gui/model_3d/Model3DView.ts b/src/viewer/gui/model_3d/Model3DView.ts index 561a748f..8c97d450 100644 --- a/src/viewer/gui/model_3d/Model3DView.ts +++ b/src/viewer/gui/model_3d/Model3DView.ts @@ -14,13 +14,15 @@ const MODEL_LIST_WIDTH = 100; const ANIMATION_LIST_WIDTH = 140; export class Model3DView extends ResizableWidget { + readonly element = el.div({ class: "viewer_Model3DView" }); + private tool_bar_view: Model3DToolBar; private model_list_view: Model3DSelectListView; private animation_list_view: Model3DSelectListView; private renderer_view: RendererWidget; constructor() { - super(el.div({ class: "viewer_Model3DView" })); + super(); this.tool_bar_view = this.disposable(new Model3DToolBar()); this.model_list_view = this.disposable( @@ -33,13 +35,15 @@ export class Model3DView extends ResizableWidget { this.animation_list_view.borders = true; - const container_element = el.div({ class: "viewer_Model3DView_container" }); - container_element.append( - this.model_list_view.element, - this.animation_list_view.element, - this.renderer_view.element, + this.element.append( + this.tool_bar_view.element, + el.div( + { class: "viewer_Model3DView_container" }, + this.model_list_view.element, + this.animation_list_view.element, + this.renderer_view.element, + ), ); - this.element.append(this.tool_bar_view.element, container_element); model_store.current_model.val = model_store.models[5];