mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
Ported quest info view to the new GUI system.
This commit is contained in:
parent
8e13441f26
commit
17400200a0
@ -1,10 +1,10 @@
|
||||
import { NavigationView } from "./NavigationView";
|
||||
import { MainContentView } from "./MainContentView";
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
import { ResizableView } from "../../core/gui/ResizableView";
|
||||
|
||||
export class ApplicationView extends ResizableView {
|
||||
element = el("div", { class: "application_ApplicationView" });
|
||||
element = create_element("div", { class: "application_ApplicationView" });
|
||||
|
||||
private menu_view = this.disposable(new NavigationView());
|
||||
private main_content_view = this.disposable(new MainContentView());
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
import { gui_store, GuiTool } from "../../core/stores/GuiStore";
|
||||
import { LazyView } from "../../core/gui/LazyView";
|
||||
import { ResizableView } from "../../core/gui/ResizableView";
|
||||
@ -12,7 +12,7 @@ const TOOLS: [GuiTool, () => Promise<ResizableView>][] = [
|
||||
];
|
||||
|
||||
export class MainContentView extends ResizableView {
|
||||
element = el("div", { class: "application_MainContentView" });
|
||||
element = create_element("div", { class: "application_MainContentView" });
|
||||
|
||||
private tool_views = new Map(
|
||||
TOOLS.map(([tool, create_view]) => [tool, this.disposable(new LazyView(create_view))]),
|
||||
@ -26,7 +26,7 @@ export class MainContentView extends ResizableView {
|
||||
}
|
||||
|
||||
const tool_view = this.tool_views.get(gui_store.tool.val);
|
||||
if (tool_view) tool_view.visible = true;
|
||||
if (tool_view) tool_view.visible.val = true;
|
||||
|
||||
this.disposable(gui_store.tool.observe(this.tool_changed));
|
||||
}
|
||||
@ -43,10 +43,10 @@ export class MainContentView extends ResizableView {
|
||||
|
||||
private tool_changed = (new_tool: GuiTool) => {
|
||||
for (const tool of this.tool_views.values()) {
|
||||
tool.visible = false;
|
||||
tool.visible.val = false;
|
||||
}
|
||||
|
||||
const new_view = this.tool_views.get(new_tool);
|
||||
if (new_view) new_view.visible = true;
|
||||
if (new_view) new_view.visible.val = true;
|
||||
};
|
||||
}
|
||||
|
@ -16,7 +16,7 @@
|
||||
display: inline-flex;
|
||||
flex-direction: row;
|
||||
align-items: center;
|
||||
font-size: 15px;
|
||||
font-size: 13px;
|
||||
height: 100%;
|
||||
padding: 0 20px;
|
||||
color: hsl(0, 0%, 65%);
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
import "./NavigationView.css";
|
||||
import { gui_store, GuiTool } from "../../core/stores/GuiStore";
|
||||
import { View } from "../../core/gui/View";
|
||||
@ -10,7 +10,7 @@ const TOOLS: [GuiTool, string][] = [
|
||||
];
|
||||
|
||||
export class NavigationView extends View {
|
||||
readonly element = el("div", { class: "application_NavigationView" });
|
||||
readonly element = create_element("div", { class: "application_NavigationView" });
|
||||
|
||||
readonly height = 30;
|
||||
|
||||
@ -22,7 +22,7 @@ export class NavigationView extends View {
|
||||
super();
|
||||
|
||||
this.element.style.height = `${this.height}px`;
|
||||
this.element.onclick = this.click;
|
||||
this.element.onmousedown = this.mousedown;
|
||||
|
||||
for (const button of this.buttons.values()) {
|
||||
this.element.append(button.element);
|
||||
@ -32,7 +32,7 @@ export class NavigationView extends View {
|
||||
this.disposable(gui_store.tool.observe(this.tool_changed));
|
||||
}
|
||||
|
||||
private click(e: MouseEvent): void {
|
||||
private mousedown(e: MouseEvent): void {
|
||||
if (e.target instanceof HTMLLabelElement && e.target.control instanceof HTMLInputElement) {
|
||||
gui_store.tool.val = (GuiTool as any)[e.target.control.value];
|
||||
}
|
||||
@ -45,10 +45,10 @@ export class NavigationView extends View {
|
||||
}
|
||||
|
||||
class ToolButton extends View {
|
||||
element: HTMLElement = el("span");
|
||||
element: HTMLElement = create_element("span");
|
||||
|
||||
private input: HTMLInputElement = el("input");
|
||||
private label: HTMLLabelElement = el("label");
|
||||
private input: HTMLInputElement = create_element("input");
|
||||
private label: HTMLLabelElement = create_element("label");
|
||||
|
||||
constructor(tool: GuiTool, text: string) {
|
||||
super();
|
||||
|
@ -2,15 +2,19 @@
|
||||
display: inline-flex;
|
||||
flex-direction: row;
|
||||
align-items: stretch;
|
||||
align-content: stretch;
|
||||
box-sizing: border-box;
|
||||
height: 26px;
|
||||
padding: 0;
|
||||
border: solid 1px hsl(0, 0%, 10%);
|
||||
color: hsl(0, 0%, 80%);
|
||||
outline: none;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.core_Button .core_Button_inner {
|
||||
display: flex;
|
||||
flex: 1;
|
||||
display: inline-flex;
|
||||
flex-direction: row;
|
||||
align-items: center;
|
||||
box-sizing: border-box;
|
||||
|
@ -1,11 +1,11 @@
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import "./Button.css";
|
||||
import { Observable } from "../observable/Observable";
|
||||
import { emitter } from "../observable";
|
||||
import { Control } from "./Control";
|
||||
|
||||
export class Button extends Control {
|
||||
readonly element: HTMLButtonElement = el("button", { class: "core_Button" });
|
||||
readonly element: HTMLButtonElement = create_element("button", { class: "core_Button" });
|
||||
|
||||
private readonly _click = emitter<MouseEvent>();
|
||||
readonly click: Observable<MouseEvent> = this._click;
|
||||
@ -13,7 +13,7 @@ export class Button extends Control {
|
||||
constructor(text: string) {
|
||||
super();
|
||||
|
||||
this.element.append(el("span", { class: "core_Button_inner", text }));
|
||||
this.element.append(create_element("span", { class: "core_Button_inner", text }));
|
||||
|
||||
this.enabled.observe(enabled => (this.element.disabled = !enabled));
|
||||
|
||||
|
@ -1,10 +1,10 @@
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import { WritableProperty } from "../observable/WritableProperty";
|
||||
import { property } from "../observable";
|
||||
import { LabelledControl } from "./LabelledControl";
|
||||
|
||||
export class CheckBox extends LabelledControl {
|
||||
readonly element: HTMLInputElement = el("input", { class: "core_CheckBox" });
|
||||
readonly element: HTMLInputElement = create_element("input", { class: "core_CheckBox" });
|
||||
|
||||
readonly checked: WritableProperty<boolean> = property(false);
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import "./FileButton.css";
|
||||
import "./Button.css";
|
||||
import { property } from "../observable";
|
||||
@ -6,14 +6,14 @@ import { Property } from "../observable/Property";
|
||||
import { Control } from "./Control";
|
||||
|
||||
export class FileButton extends Control {
|
||||
readonly element: HTMLLabelElement = el("label", {
|
||||
readonly element: HTMLLabelElement = create_element("label", {
|
||||
class: "core_FileButton core_Button",
|
||||
});
|
||||
|
||||
private readonly _files = property<File[]>([]);
|
||||
readonly files: Property<File[]> = this._files;
|
||||
|
||||
private input: HTMLInputElement = el("input", {
|
||||
private input: HTMLInputElement = create_element("input", {
|
||||
class: "core_FileButton_input core_Button_inner",
|
||||
});
|
||||
|
||||
@ -31,7 +31,7 @@ export class FileButton extends Control {
|
||||
};
|
||||
|
||||
this.element.append(
|
||||
el("span", {
|
||||
create_element("span", {
|
||||
class: "core_FileButton_inner core_Button_inner",
|
||||
text,
|
||||
}),
|
||||
|
@ -1,32 +1,35 @@
|
||||
.core_Input {
|
||||
display: inline-block;
|
||||
box-sizing: border-box;
|
||||
border: solid 1px hsl(0, 0%, 25%);
|
||||
height: 24px;
|
||||
border: var(--input-border);
|
||||
}
|
||||
|
||||
.core_Input .core_Input_inner {
|
||||
box-sizing: border-box;
|
||||
width: 100%;
|
||||
height: 24px;
|
||||
height: 100%;
|
||||
padding: 0 3px;
|
||||
border: solid 1px hsl(0, 0%, 0%);
|
||||
background-color: hsl(0, 0%, 12%);
|
||||
color: hsl(0, 0%, 75%);
|
||||
border: var(--input-inner-border);
|
||||
background-color: var(--input-bg-color);
|
||||
color: var(--input-text-color);
|
||||
outline: none;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.core_Input:hover {
|
||||
border-color: hsl(0, 0%, 35%);
|
||||
border-color: var(--input-border-hover);
|
||||
}
|
||||
|
||||
.core_Input:focus-within {
|
||||
border-color: hsl(0, 0%, 45%);
|
||||
border-color: var(--input-border-focus);
|
||||
}
|
||||
|
||||
.core_Input.disabled {
|
||||
border: solid 1px hsl(0, 0%, 20%);
|
||||
border: var(--input-border-disabled);
|
||||
}
|
||||
|
||||
.core_Input.disabled .core_Input_inner {
|
||||
background-color: hsl(0, 0%, 15%);
|
||||
color: var(--text-color-disabled);
|
||||
color: var(--input-text-color-disabled);
|
||||
background-color: var(--input-bg-color-disabled);
|
||||
}
|
||||
|
83
src/core/gui/Input.ts
Normal file
83
src/core/gui/Input.ts
Normal file
@ -0,0 +1,83 @@
|
||||
/* eslint-disable no-dupe-class-members */
|
||||
import { LabelledControl } from "./LabelledControl";
|
||||
import { create_element } from "./dom";
|
||||
import { WritableProperty } from "../observable/WritableProperty";
|
||||
import { is_any_property, Property } from "../observable/Property";
|
||||
import "./Input.css";
|
||||
|
||||
export abstract class Input<T> extends LabelledControl {
|
||||
readonly element: HTMLElement;
|
||||
|
||||
readonly value: WritableProperty<T>;
|
||||
|
||||
protected readonly input: HTMLInputElement;
|
||||
|
||||
protected constructor(
|
||||
value: WritableProperty<T>,
|
||||
class_name: string,
|
||||
input_type: string,
|
||||
input_class_name: string,
|
||||
label?: string,
|
||||
) {
|
||||
super(label);
|
||||
|
||||
this.value = value;
|
||||
|
||||
this.element = create_element("span", { class: `${class_name} core_Input` });
|
||||
|
||||
this.input = create_element("input", {
|
||||
class: `${input_class_name} core_Input_inner`,
|
||||
});
|
||||
this.input.type = input_type;
|
||||
this.input.onchange = () => (this.value.val = this.get_input_value());
|
||||
this.set_input_value(value.val);
|
||||
|
||||
this.element.append(this.input);
|
||||
|
||||
this.disposables(
|
||||
this.value.observe(value => this.set_input_value(value)),
|
||||
|
||||
this.enabled.observe(enabled => {
|
||||
this.input.disabled = !enabled;
|
||||
|
||||
if (enabled) {
|
||||
this.element.classList.remove("disabled");
|
||||
} else {
|
||||
this.element.classList.add("disabled");
|
||||
}
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
protected abstract get_input_value(): T;
|
||||
|
||||
protected abstract set_input_value(value: T): void;
|
||||
|
||||
protected set_attr<T>(attr: InputAttrsOfType<T>, value?: T | Property<T>): void;
|
||||
protected set_attr<T, U>(
|
||||
attr: InputAttrsOfType<U>,
|
||||
value: T | Property<T> | undefined,
|
||||
convert: (value: T) => U,
|
||||
): void;
|
||||
protected set_attr<T, U>(
|
||||
attr: InputAttrsOfType<U>,
|
||||
value?: T | Property<T>,
|
||||
convert?: (value: T) => U,
|
||||
): void {
|
||||
if (value == undefined) return;
|
||||
|
||||
const input = this.input as any;
|
||||
const cvt = convert ? convert : (v: T) => (v as any) as U;
|
||||
|
||||
if (is_any_property(value)) {
|
||||
input[attr] = cvt(value.val);
|
||||
this.disposable(value.observe(v => (input[attr] = cvt(v))));
|
||||
} else {
|
||||
input[attr] = cvt(value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
type InputAttrsOfType<T> = {
|
||||
[K in keyof HTMLInputElement]: T extends HTMLInputElement[K] ? K : never;
|
||||
}[keyof HTMLInputElement];
|
@ -1,12 +1,12 @@
|
||||
import { View } from "./View";
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import { WritableProperty } from "../observable/WritableProperty";
|
||||
import "./Label.css";
|
||||
import { property } from "../observable";
|
||||
import { Property } from "../observable/Property";
|
||||
|
||||
export class Label extends View {
|
||||
readonly element = el<HTMLLabelElement>("label", { class: "core_Label" });
|
||||
readonly element = create_element<HTMLLabelElement>("label", { class: "core_Label" });
|
||||
|
||||
set for(id: string) {
|
||||
this.element.htmlFor = id;
|
||||
@ -14,7 +14,7 @@ export class Label extends View {
|
||||
|
||||
readonly enabled: WritableProperty<boolean> = property(true);
|
||||
|
||||
constructor(text: string | Property<string>) {
|
||||
constructor(text: string | Property<string>, options: { enabled?: boolean } = {}) {
|
||||
super();
|
||||
|
||||
if (typeof text === "string") {
|
||||
@ -33,5 +33,7 @@ export class Label extends View {
|
||||
}
|
||||
}),
|
||||
);
|
||||
|
||||
if (options.enabled != undefined) this.enabled.val = options.enabled;
|
||||
}
|
||||
}
|
||||
|
@ -2,7 +2,7 @@ import { Label } from "./Label";
|
||||
import { Control } from "./Control";
|
||||
|
||||
export abstract class LabelledControl extends Control {
|
||||
abstract readonly preferred_label_position: "left" | "right";
|
||||
abstract readonly preferred_label_position: "left" | "right" | "top" | "bottom";
|
||||
|
||||
private readonly _label_text: string;
|
||||
private _label?: Label;
|
||||
|
@ -1,29 +1,10 @@
|
||||
import { View } from "./View";
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import { Resizable } from "./Resizable";
|
||||
import { ResizableView } from "./ResizableView";
|
||||
|
||||
export class LazyView extends ResizableView {
|
||||
readonly element = el("div", { class: "core_LazyView" });
|
||||
|
||||
private _visible = false;
|
||||
|
||||
set visible(visible: boolean) {
|
||||
if (this._visible !== visible) {
|
||||
this._visible = visible;
|
||||
this.element.hidden = !visible;
|
||||
|
||||
if (visible && !this.initialized) {
|
||||
this.initialized = true;
|
||||
|
||||
this.create_view().then(view => {
|
||||
this.view = this.disposable(view);
|
||||
this.view.resize(this.width, this.height);
|
||||
this.element.append(view.element);
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
readonly element = create_element("div", { class: "core_LazyView" });
|
||||
|
||||
private initialized = false;
|
||||
private view: View & Resizable | undefined;
|
||||
@ -31,7 +12,21 @@ export class LazyView extends ResizableView {
|
||||
constructor(private create_view: () => Promise<View & Resizable>) {
|
||||
super();
|
||||
|
||||
this.element.hidden = true;
|
||||
this.visible.val = false;
|
||||
|
||||
this.disposables(
|
||||
this.visible.observe(visible => {
|
||||
if (visible && !this.initialized) {
|
||||
this.initialized = true;
|
||||
|
||||
this.create_view().then(view => {
|
||||
this.view = this.disposable(view);
|
||||
this.view.resize(this.width, this.height);
|
||||
this.element.append(view.element);
|
||||
});
|
||||
}
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
resize(width: number, height: number): this {
|
||||
|
@ -1,3 +1,3 @@
|
||||
.core_NumberInput .core_NumberInput_inner {
|
||||
text-align: right;
|
||||
padding-right: 1px;
|
||||
}
|
@ -1,65 +1,43 @@
|
||||
import "./NumberInput.css";
|
||||
import "./Input.css";
|
||||
import { el } from "./dom";
|
||||
import { WritableProperty } from "../observable/WritableProperty";
|
||||
import { property } from "../observable";
|
||||
import { LabelledControl } from "./LabelledControl";
|
||||
import { is_any_property, Property } from "../observable/Property";
|
||||
|
||||
export class NumberInput extends LabelledControl {
|
||||
readonly element = el("span", { class: "core_NumberInput core_Input" });
|
||||
|
||||
readonly value: WritableProperty<number> = property(0);
|
||||
import { Property } from "../observable/Property";
|
||||
import { Input } from "./Input";
|
||||
import "./NumberInput.css"
|
||||
|
||||
export class NumberInput extends Input<number> {
|
||||
readonly preferred_label_position = "left";
|
||||
|
||||
private readonly input: HTMLInputElement = el("input", {
|
||||
class: "core_NumberInput_inner core_Input_inner",
|
||||
});
|
||||
|
||||
constructor(
|
||||
value = 0,
|
||||
label?: string,
|
||||
min: number | Property<number> = -Infinity,
|
||||
max: number | Property<number> = Infinity,
|
||||
step: number | Property<number> = 1,
|
||||
value: number = 0,
|
||||
options?: {
|
||||
label?: string;
|
||||
min?: number | Property<number>;
|
||||
max?: number | Property<number>;
|
||||
step?: number | Property<number>;
|
||||
},
|
||||
) {
|
||||
super(label);
|
||||
|
||||
this.input.type = "number";
|
||||
this.input.valueAsNumber = value;
|
||||
|
||||
this.set_prop("min", min);
|
||||
this.set_prop("max", max);
|
||||
this.set_prop("step", step);
|
||||
|
||||
this.input.onchange = () => (this.value.val = this.input.valueAsNumber);
|
||||
|
||||
this.element.append(this.input);
|
||||
|
||||
this.disposables(
|
||||
this.value.observe(value => (this.input.valueAsNumber = value)),
|
||||
|
||||
this.enabled.observe(enabled => {
|
||||
this.input.disabled = !enabled;
|
||||
|
||||
if (enabled) {
|
||||
this.element.classList.remove("disabled");
|
||||
} else {
|
||||
this.element.classList.add("disabled");
|
||||
}
|
||||
}),
|
||||
super(
|
||||
property(value),
|
||||
"core_NumberInput",
|
||||
"number",
|
||||
"core_NumberInput_inner",
|
||||
options && options.label,
|
||||
);
|
||||
|
||||
this.element.style.width = "50px";
|
||||
if (options) {
|
||||
const { min, max, step } = options;
|
||||
this.set_attr("min", min, String);
|
||||
this.set_attr("max", max, String);
|
||||
this.set_attr("step", step, String);
|
||||
}
|
||||
|
||||
this.element.style.width = "54px";
|
||||
}
|
||||
|
||||
private set_prop<T>(prop: "min" | "max" | "step", value: T | Property<T>): void {
|
||||
if (is_any_property(value)) {
|
||||
this.input[prop] = String(value.val);
|
||||
this.disposable(value.observe(v => (this.input[prop] = String(v))));
|
||||
} else {
|
||||
this.input[prop] = String(value);
|
||||
}
|
||||
protected get_input_value(): number {
|
||||
return this.input.valueAsNumber;
|
||||
}
|
||||
|
||||
protected set_input_value(value: number): void {
|
||||
this.input.valueAsNumber = value;
|
||||
}
|
||||
}
|
||||
|
@ -1,9 +1,9 @@
|
||||
import { ResizableView } from "./ResizableView";
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import { Renderer } from "../rendering/Renderer";
|
||||
|
||||
export class RendererView extends ResizableView {
|
||||
readonly element = el("div");
|
||||
readonly element = create_element("div");
|
||||
|
||||
constructor(private renderer: Renderer) {
|
||||
super();
|
||||
|
@ -14,7 +14,7 @@
|
||||
margin: 0 1px -1px 1px;
|
||||
background-color: hsl(0, 0%, 12%);
|
||||
color: hsl(0, 0%, 75%);
|
||||
font-size: 15px;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.core_TabContainer_Tab:hover {
|
||||
|
@ -1,5 +1,5 @@
|
||||
import { View } from "./View";
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import { LazyView } from "./LazyView";
|
||||
import { Resizable } from "./Resizable";
|
||||
import { ResizableView } from "./ResizableView";
|
||||
@ -16,19 +16,19 @@ type TabInfo = Tab & { tab_element: HTMLSpanElement; lazy_view: LazyView };
|
||||
const BAR_HEIGHT = 28;
|
||||
|
||||
export class TabContainer extends ResizableView {
|
||||
readonly element = el("div", { class: "core_TabContainer" });
|
||||
readonly element = create_element("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" });
|
||||
private bar_element = create_element("div", { class: "core_TabContainer_Bar" });
|
||||
private panes_element = create_element("div", { class: "core_TabContainer_Panes" });
|
||||
|
||||
constructor(...tabs: Tab[]) {
|
||||
super();
|
||||
|
||||
this.bar_element.onclick = this.bar_click;
|
||||
this.bar_element.onmousedown = this.bar_mousedown;
|
||||
|
||||
for (const tab of tabs) {
|
||||
const tab_element = el("span", {
|
||||
const tab_element = create_element("span", {
|
||||
class: "core_TabContainer_Tab",
|
||||
text: tab.title,
|
||||
data: { key: tab.key },
|
||||
@ -72,7 +72,7 @@ export class TabContainer extends ResizableView {
|
||||
return this;
|
||||
}
|
||||
|
||||
private bar_click = (e: MouseEvent) => {
|
||||
private bar_mousedown = (e: MouseEvent) => {
|
||||
if (e.target instanceof HTMLElement) {
|
||||
const key = e.target.dataset["key"];
|
||||
if (key) this.activate(key);
|
||||
@ -89,7 +89,7 @@ export class TabContainer extends ResizableView {
|
||||
tab.tab_element.classList.remove("active");
|
||||
}
|
||||
|
||||
tab.lazy_view.visible = active;
|
||||
tab.lazy_view.visible.val = active;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
33
src/core/gui/TextArea.css
Normal file
33
src/core/gui/TextArea.css
Normal file
@ -0,0 +1,33 @@
|
||||
.core_TextArea {
|
||||
box-sizing: border-box;
|
||||
display: inline-block;
|
||||
border: var(--input-border);
|
||||
}
|
||||
|
||||
.core_TextArea .core_TextArea_inner {
|
||||
box-sizing: border-box;
|
||||
vertical-align: top;
|
||||
padding: 3px;
|
||||
border: var(--input-inner-border);
|
||||
background-color: var(--input-bg-color);
|
||||
color: var(--input-text-color);
|
||||
outline: none;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.core_TextArea:hover {
|
||||
border-color: var(--input-border-hover);
|
||||
}
|
||||
|
||||
.core_TextArea:focus-within {
|
||||
border-color: var(--input-border-focus);
|
||||
}
|
||||
|
||||
.core_TextArea.disabled {
|
||||
border: var(--input-border-disabled);
|
||||
}
|
||||
|
||||
.core_TextArea.disabled .core_TextArea_inner {
|
||||
color: var(--input-text-color-disabled);
|
||||
background-color: var(--input-bg-color-disabled);
|
||||
}
|
46
src/core/gui/TextArea.ts
Normal file
46
src/core/gui/TextArea.ts
Normal file
@ -0,0 +1,46 @@
|
||||
import { LabelledControl } from "./LabelledControl";
|
||||
import { el } from "./dom";
|
||||
import { property } from "../observable";
|
||||
import { WritableProperty } from "../observable/WritableProperty";
|
||||
import "./TextArea.css";
|
||||
|
||||
export class TextArea extends LabelledControl {
|
||||
readonly element: HTMLElement = el.div({ class: "core_TextArea" });
|
||||
|
||||
readonly preferred_label_position = "left";
|
||||
|
||||
readonly value: WritableProperty<string>;
|
||||
|
||||
private readonly text_element: HTMLTextAreaElement = el.textarea({
|
||||
class: "core_TextArea_inner",
|
||||
});
|
||||
|
||||
constructor(
|
||||
value = "",
|
||||
options?: {
|
||||
label?: string;
|
||||
max_length?: number;
|
||||
font_family?: string;
|
||||
rows?: number;
|
||||
cols?: number;
|
||||
},
|
||||
) {
|
||||
super(options && options.label);
|
||||
|
||||
if (options) {
|
||||
if (options.max_length != undefined) this.text_element.maxLength = options.max_length;
|
||||
if (options.font_family != undefined)
|
||||
this.text_element.style.fontFamily = options.font_family;
|
||||
if (options.rows != undefined) this.text_element.rows = options.rows;
|
||||
if (options.cols != undefined) this.text_element.cols = options.cols;
|
||||
}
|
||||
|
||||
this.value = property(value);
|
||||
|
||||
this.text_element.onchange = () => (this.value.val = this.text_element.value);
|
||||
|
||||
this.disposables(this.value.observe(value => (this.text_element.value = value)));
|
||||
|
||||
this.element.append(this.text_element);
|
||||
}
|
||||
}
|
36
src/core/gui/TextInput.ts
Normal file
36
src/core/gui/TextInput.ts
Normal file
@ -0,0 +1,36 @@
|
||||
import { Input } from "./Input";
|
||||
import { Property } from "../observable/Property";
|
||||
import { property } from "../observable";
|
||||
|
||||
export class TextInput extends Input<string> {
|
||||
readonly preferred_label_position = "left";
|
||||
|
||||
constructor(
|
||||
value = "",
|
||||
options?: {
|
||||
label?: string;
|
||||
max_length?: number | Property<number>;
|
||||
},
|
||||
) {
|
||||
super(
|
||||
property(value),
|
||||
"core_TextInput",
|
||||
"text",
|
||||
"core_TextInput_inner",
|
||||
options && options.label,
|
||||
);
|
||||
|
||||
if (options) {
|
||||
const { max_length } = options;
|
||||
this.set_attr("maxLength", max_length);
|
||||
}
|
||||
}
|
||||
|
||||
protected get_input_value(): string {
|
||||
return this.input.value;
|
||||
}
|
||||
|
||||
protected set_input_value(value: string): void {
|
||||
this.input.value = value;
|
||||
}
|
||||
}
|
@ -20,3 +20,7 @@
|
||||
.core_ToolBar > .core_ToolBar_group > * {
|
||||
margin: 0 2px;
|
||||
}
|
||||
|
||||
.core_ToolBar .core_Input {
|
||||
height: 26px;
|
||||
}
|
||||
|
@ -1,10 +1,10 @@
|
||||
import { View } from "./View";
|
||||
import { el } from "./dom";
|
||||
import { create_element } from "./dom";
|
||||
import "./ToolBar.css";
|
||||
import { LabelledControl } from "./LabelledControl";
|
||||
|
||||
export class ToolBar extends View {
|
||||
readonly element = el("div", { class: "core_ToolBar" });
|
||||
readonly element = create_element("div", { class: "core_ToolBar" });
|
||||
readonly height = 33;
|
||||
|
||||
constructor(...children: View[]) {
|
||||
@ -14,9 +14,12 @@ export class ToolBar extends View {
|
||||
|
||||
for (const child of children) {
|
||||
if (child instanceof LabelledControl) {
|
||||
const group = el("div", { class: "core_ToolBar_group" });
|
||||
const group = create_element("div", { class: "core_ToolBar_group" });
|
||||
|
||||
if (child.preferred_label_position === "left") {
|
||||
if (
|
||||
child.preferred_label_position === "left" ||
|
||||
child.preferred_label_position === "top"
|
||||
) {
|
||||
group.append(child.label.element, child.element);
|
||||
} else {
|
||||
group.append(child.element, child.label.element);
|
||||
|
@ -2,6 +2,8 @@ import { Disposable } from "../observable/Disposable";
|
||||
import { Disposer } from "../observable/Disposer";
|
||||
import { Observable } from "../observable/Observable";
|
||||
import { bind_hidden } from "./dom";
|
||||
import { WritableProperty } from "../observable/WritableProperty";
|
||||
import { property } from "../observable";
|
||||
|
||||
export abstract class View implements Disposable {
|
||||
abstract readonly element: HTMLElement;
|
||||
@ -14,8 +16,14 @@ export abstract class View implements Disposable {
|
||||
this.element.id = id;
|
||||
}
|
||||
|
||||
readonly visible: WritableProperty<boolean> = property(true);
|
||||
|
||||
private disposer = new Disposer();
|
||||
|
||||
constructor() {
|
||||
this.disposables(this.visible.observe(visible => (this.element.hidden = !visible)));
|
||||
}
|
||||
|
||||
dispose(): void {
|
||||
this.element.remove();
|
||||
this.disposer.dispose();
|
||||
|
@ -2,16 +2,41 @@ import { Disposable } from "../observable/Disposable";
|
||||
import { Observable } from "../observable/Observable";
|
||||
import { is_property } from "../observable/Property";
|
||||
|
||||
export function el<T extends HTMLElement>(
|
||||
export const el = {
|
||||
div: (attributes?: {}, ...children: HTMLElement[]): HTMLDivElement =>
|
||||
create_element("div", attributes, ...children),
|
||||
|
||||
table: (attributes?: {}, ...children: HTMLElement[]): HTMLTableElement =>
|
||||
create_element("table", attributes, ...children),
|
||||
|
||||
tr: (attributes?: {}, ...children: HTMLElement[]): HTMLTableRowElement =>
|
||||
create_element("tr", attributes, ...children),
|
||||
|
||||
th: (
|
||||
attributes?: { text?: string; col_span?: number },
|
||||
...children: HTMLElement[]
|
||||
): HTMLTableHeaderCellElement => create_element("th", attributes, ...children),
|
||||
|
||||
td: (
|
||||
attributes?: { text?: string; col_span?: number },
|
||||
...children: HTMLElement[]
|
||||
): HTMLTableCellElement => create_element("td", attributes, ...children),
|
||||
|
||||
textarea: (attributes?: {}, ...children: HTMLElement[]): HTMLTextAreaElement =>
|
||||
create_element("textarea", attributes, ...children),
|
||||
};
|
||||
|
||||
export function create_element<T extends HTMLElement>(
|
||||
tag_name: string,
|
||||
attributes?: {
|
||||
class?: string;
|
||||
text?: string ;
|
||||
text?: string;
|
||||
data?: { [key: string]: string };
|
||||
col_span?: number;
|
||||
},
|
||||
...children: HTMLElement[]
|
||||
): T {
|
||||
const element = document.createElement(tag_name) as T;
|
||||
const element = document.createElement(tag_name) as HTMLTableCellElement;
|
||||
|
||||
if (attributes) {
|
||||
if (attributes.class) element.className = attributes.class;
|
||||
@ -22,11 +47,13 @@ export function el<T extends HTMLElement>(
|
||||
element.dataset[key] = val;
|
||||
}
|
||||
}
|
||||
|
||||
if (attributes.col_span) element.colSpan = attributes.col_span;
|
||||
}
|
||||
|
||||
element.append(...children);
|
||||
|
||||
return element;
|
||||
return (element as HTMLElement) as T;
|
||||
}
|
||||
|
||||
export function bind_hidden(element: HTMLElement, observable: Observable<boolean>): Disposable {
|
||||
|
@ -17,7 +17,7 @@
|
||||
margin: 0 1px -1px 1px;
|
||||
background-color: hsl(0, 0%, 12%);
|
||||
color: hsl(0, 0%, 75%);
|
||||
font-size: 15px;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
#root .lm_tab:hover {
|
||||
|
@ -1,11 +1,28 @@
|
||||
:root {
|
||||
/* Basic view variables */
|
||||
|
||||
--bg-color: hsl(0, 0%, 15%);
|
||||
--text-color: hsl(0, 0%, 80%);
|
||||
--text-color-disabled: hsl(0, 0%, 55%);
|
||||
--border-color: hsl(0, 0%, 25%);
|
||||
|
||||
/* Scrollbars */
|
||||
|
||||
--scrollbar-color: hsl(0, 0%, 13%);
|
||||
--scrollbar-thumb-color: hsl(0, 0%, 17%);
|
||||
|
||||
/* Inputs */
|
||||
|
||||
--input-bg-color: hsl(0, 0%, 12%);
|
||||
--input-bg-color-disabled: hsl(0, 0%, 15%);
|
||||
--input-text-color: hsl(0, 0%, 75%);
|
||||
--input-text-color-disabled: var(--text-color-disabled);
|
||||
--input-border: solid 1px hsl(0, 0%, 25%);
|
||||
--input-border-hover: hsl(0, 0%, 35%);
|
||||
--input-border-focus: hsl(0, 0%, 45%);
|
||||
--input-border-disabled: solid 1px hsl(0, 0%, 20%);
|
||||
|
||||
--input-inner-border: solid 1px hsl(0, 0%, 5%);
|
||||
}
|
||||
|
||||
* {
|
||||
@ -33,9 +50,15 @@ body {
|
||||
user-select: none;
|
||||
overflow: hidden;
|
||||
margin: 0;
|
||||
font-size: 15px;
|
||||
font-family: "Helvetica Neue Light", "HelveticaNeue-Light", "Helvetica Neue", Calibri, Helvetica,
|
||||
Arial, sans-serif;
|
||||
font-size: 13px;
|
||||
background-color: var(--bg-color);
|
||||
color: var(--text-color);
|
||||
}
|
||||
|
||||
* {
|
||||
font-family: Verdana, Geneva, sans-serif;
|
||||
}
|
||||
|
||||
#root *[hidden] {
|
||||
display: none;
|
||||
}
|
@ -15,7 +15,7 @@ export class DependentProperty<T> extends AbstractMinimalProperty<T> implements
|
||||
private _val?: T;
|
||||
|
||||
get val(): T {
|
||||
if (this.dependency_disposables) {
|
||||
if (this.dependency_disposables.length) {
|
||||
return this._val as T;
|
||||
} else {
|
||||
return this.f();
|
||||
|
@ -15,5 +15,5 @@ export function is_property<T>(observable: Observable<T>): observable is Propert
|
||||
}
|
||||
|
||||
export function is_any_property(observable: any): observable is Property<any> {
|
||||
return (observable as any).is_property;
|
||||
return observable && (observable as any).is_property;
|
||||
}
|
||||
|
@ -1,6 +1,6 @@
|
||||
import { ApplicationView } from "./application/gui/ApplicationView";
|
||||
import { Disposable } from "./core/observable/Disposable";
|
||||
import "./index.css";
|
||||
import "./core/gui/index.css";
|
||||
import { throttle } from "lodash";
|
||||
import Logger from "js-logger";
|
||||
|
||||
|
@ -1,6 +1,6 @@
|
||||
import { ResizableView } from "../../core/gui/ResizableView";
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
|
||||
export class NpcCountsView extends ResizableView {
|
||||
readonly element = el("div");
|
||||
readonly element = create_element("div");
|
||||
}
|
||||
|
33
src/quest_editor/gui/QuesInfoView.css
Normal file
33
src/quest_editor/gui/QuesInfoView.css
Normal file
@ -0,0 +1,33 @@
|
||||
.quest_editor_QuesInfoView {
|
||||
box-sizing: border-box;
|
||||
padding: 3px;
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
.quest_editor_QuesInfoView table {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.quest_editor_QuesInfoView th {
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.quest_editor_QuesInfoView .core_TextInput {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.quest_editor_QuesInfoView .core_TextArea {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.quest_editor_QuesInfoView textarea {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.quest_editor_QuesInfoView_no_quest {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
@ -2,36 +2,78 @@ import { ResizableView } from "../../core/gui/ResizableView";
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { quest_editor_store } from "../stores/QuestEditorStore";
|
||||
import { Episode } from "../../core/data_formats/parsing/quest/Episode";
|
||||
import { NumberInput } from "../../core/gui/NumberInput";
|
||||
import { Disposer } from "../../core/observable/Disposer";
|
||||
import { TextInput } from "../../core/gui/TextInput";
|
||||
import { TextArea } from "../../core/gui/TextArea";
|
||||
import "./QuesInfoView.css";
|
||||
import { Label } from "../../core/gui/Label";
|
||||
|
||||
export class QuesInfoView extends ResizableView {
|
||||
readonly element = el("div", { class: "quest_editor_QuesInfoView" });
|
||||
readonly element = el.div({ class: "quest_editor_QuesInfoView" });
|
||||
|
||||
private readonly table_element = el("table");
|
||||
private readonly table_element = el.table();
|
||||
private readonly episode_element: HTMLElement;
|
||||
private readonly id_element: HTMLElement;
|
||||
private readonly name_element: HTMLElement;
|
||||
private readonly id_input = this.disposable(new NumberInput());
|
||||
private readonly name_input = this.disposable(new TextInput());
|
||||
private readonly short_description_input = this.disposable(
|
||||
new TextArea("", {
|
||||
max_length: 128,
|
||||
font_family: '"Courier New", monospace',
|
||||
cols: 25,
|
||||
rows: 5,
|
||||
}),
|
||||
);
|
||||
private readonly long_description_input = this.disposable(
|
||||
new TextArea("", {
|
||||
max_length: 288,
|
||||
font_family: '"Courier New", monospace',
|
||||
cols: 25,
|
||||
rows: 10,
|
||||
}),
|
||||
);
|
||||
|
||||
private readonly no_quest_element = el.div({ class: "quest_editor_QuesInfoView_no_quest" });
|
||||
private readonly no_quest_label = this.disposable(
|
||||
new Label("No quest loaded.", { enabled: false }),
|
||||
);
|
||||
|
||||
private readonly quest_disposer = this.disposable(new Disposer());
|
||||
|
||||
constructor() {
|
||||
super();
|
||||
|
||||
const quest = quest_editor_store.current_quest;
|
||||
|
||||
this.bind_hidden(this.table_element, quest.map(q => q == undefined));
|
||||
this.no_quest_element.append(this.no_quest_label.element);
|
||||
this.bind_hidden(this.no_quest_element, quest.map(q => q != undefined));
|
||||
|
||||
this.table_element.append(
|
||||
el("tr", {}, el("th", { text: "Episode:" }), (this.episode_element = el("td"))),
|
||||
el("tr", {}, el("th", { text: "ID:" }), (this.id_element = el("td"))),
|
||||
el("tr", {}, el("th", { text: "Name:" }), (this.name_element = el("td"))),
|
||||
el.tr({}, el.th({ text: "Episode:" }), (this.episode_element = el.td())),
|
||||
el.tr({}, el.th({ text: "ID:" }), el.td({}, this.id_input.element)),
|
||||
el.tr({}, el.th({ text: "Name:" }), el.td({}, this.name_input.element)),
|
||||
el.tr({}, el.th({ text: "Short description:", col_span: 2 })),
|
||||
el.tr({}, el.td({ col_span: 2 }, this.short_description_input.element)),
|
||||
el.tr({}, el.th({ text: "Long description:", col_span: 2 })),
|
||||
el.tr({}, el.td({ col_span: 2 }, this.long_description_input.element)),
|
||||
);
|
||||
this.bind_hidden(this.table_element, quest.map(q => q == undefined));
|
||||
|
||||
this.element.append(this.table_element);
|
||||
this.element.append(this.table_element, this.no_quest_element);
|
||||
|
||||
this.disposables(
|
||||
quest.observe(q => {
|
||||
this.quest_disposer.dispose();
|
||||
|
||||
this.episode_element.textContent = q ? Episode[q.episode] : "";
|
||||
|
||||
if (q) {
|
||||
this.episode_element.textContent = Episode[q.episode];
|
||||
this.id_element.textContent = q.id.val.toString();
|
||||
this.name_element.textContent = q.name.val;
|
||||
this.quest_disposer.add_all(
|
||||
this.id_input.value.bind_bi(q.id),
|
||||
this.name_input.value.bind_bi(q.name),
|
||||
this.short_description_input.value.bind_bi(q.short_description),
|
||||
this.long_description_input.value.bind_bi(q.long_description),
|
||||
);
|
||||
}
|
||||
}),
|
||||
);
|
||||
|
@ -1,5 +1,5 @@
|
||||
import { ResizableView } from "../../core/gui/ResizableView";
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
import { ToolBarView } from "./ToolBarView";
|
||||
import GoldenLayout, { Container, ContentItem, ItemConfigType } from "golden-layout";
|
||||
import { quest_editor_ui_persister } from "../persistence/QuestEditorUiPersister";
|
||||
@ -89,11 +89,11 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
|
||||
];
|
||||
|
||||
export class QuestEditorView extends ResizableView {
|
||||
readonly element = el("div", { class: "quest_editor_QuestEditorView" });
|
||||
readonly element = create_element("div", { class: "quest_editor_QuestEditorView" });
|
||||
|
||||
private readonly tool_bar_view = this.disposable(new ToolBarView());
|
||||
|
||||
private readonly layout_element = el("div", { class: "quest_editor_gl_container" });
|
||||
private readonly layout_element = create_element("div", { class: "quest_editor_gl_container" });
|
||||
private readonly layout: Promise<GoldenLayout>;
|
||||
|
||||
constructor() {
|
||||
|
@ -6,14 +6,20 @@ import { parse_quest } from "../../core/data_formats/parsing/quest";
|
||||
import { ArrayBufferCursor } from "../../core/data_formats/cursor/ArrayBufferCursor";
|
||||
import { Endianness } from "../../core/data_formats/Endianness";
|
||||
import { SimpleUndo, UndoStack } from "../../old/core/undo";
|
||||
import { WritableProperty } from "../../core/observable/WritableProperty";
|
||||
import Logger = require("js-logger");
|
||||
|
||||
const logger = Logger.get("quest_editor/gui/QuestEditorStore");
|
||||
|
||||
export class QuestEditorStore {
|
||||
readonly debug: WritableProperty<boolean> = property(false);
|
||||
|
||||
readonly undo = new UndoStack();
|
||||
readonly script_undo = new SimpleUndo("Text edits", () => {}, () => {});
|
||||
|
||||
private readonly _current_quest_filename = property<string | undefined>(undefined);
|
||||
readonly current_quest_filename: Property<string | undefined> = this._current_quest_filename;
|
||||
|
||||
private readonly _current_quest = property<ObservableQuest | undefined>(undefined);
|
||||
readonly current_quest: Property<ObservableQuest | undefined> = this._current_quest;
|
||||
|
||||
@ -74,7 +80,7 @@ export class QuestEditorStore {
|
||||
};
|
||||
|
||||
private set_quest(quest?: ObservableQuest, filename?: string): void {
|
||||
// this.current_quest_filename = filename;
|
||||
this._current_quest_filename.val = filename;
|
||||
this.undo.reset();
|
||||
this.script_undo.reset();
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
import { ResizableView } from "../../core/gui/ResizableView";
|
||||
import { ToolBar } from "../../core/gui/ToolBar";
|
||||
import "./ModelView.css";
|
||||
@ -18,10 +18,10 @@ const MODEL_LIST_WIDTH = 100;
|
||||
const ANIMATION_LIST_WIDTH = 130;
|
||||
|
||||
export class ModelView extends ResizableView {
|
||||
readonly element = el("div", { class: "viewer_ModelView" });
|
||||
readonly element = create_element("div", { class: "viewer_ModelView" });
|
||||
|
||||
private tool_bar_view = this.disposable(new ToolBarView());
|
||||
private container_element = el("div", { class: "viewer_ModelView_container" });
|
||||
private container_element = create_element("div", { class: "viewer_ModelView_container" });
|
||||
private model_list_view = this.disposable(
|
||||
new ModelSelectListView(model_store.models, model_store.current_model),
|
||||
);
|
||||
@ -78,20 +78,18 @@ class ToolBarView extends View {
|
||||
private readonly open_file_button = new FileButton("Open file...", ".nj, .njm, .xj, .xvm");
|
||||
private readonly skeleton_checkbox = new CheckBox(false, "Show skeleton");
|
||||
private readonly play_animation_checkbox = new CheckBox(true, "Play animation");
|
||||
private readonly animation_frame_rate_input = new NumberInput(
|
||||
PSO_FRAME_RATE,
|
||||
"Frame rate:",
|
||||
1,
|
||||
240,
|
||||
1,
|
||||
);
|
||||
private readonly animation_frame_input = new NumberInput(
|
||||
1,
|
||||
"Frame:",
|
||||
1,
|
||||
model_store.animation_frame_count,
|
||||
1,
|
||||
);
|
||||
private readonly animation_frame_rate_input = new NumberInput(PSO_FRAME_RATE, {
|
||||
label: "Frame rate:",
|
||||
min: 1,
|
||||
max: 240,
|
||||
step: 1,
|
||||
});
|
||||
private readonly animation_frame_input = new NumberInput(1, {
|
||||
label: "Frame:",
|
||||
min: 1,
|
||||
max: model_store.animation_frame_count,
|
||||
step: 1,
|
||||
});
|
||||
private readonly animation_frame_count_label = new Label(
|
||||
model_store.animation_frame_count.map(count => `/ ${count}`),
|
||||
);
|
||||
@ -147,7 +145,7 @@ class ToolBarView extends View {
|
||||
}
|
||||
|
||||
class ModelSelectListView<T extends { name: string }> extends ResizableView {
|
||||
element = el("ul", { class: "viewer_ModelSelectListView" });
|
||||
element = create_element("ul", { class: "viewer_ModelSelectListView" });
|
||||
|
||||
set borders(borders: boolean) {
|
||||
if (borders) {
|
||||
@ -169,7 +167,7 @@ class ModelSelectListView<T extends { name: string }> extends ResizableView {
|
||||
|
||||
models.forEach((model, index) => {
|
||||
this.element.append(
|
||||
el("li", { text: model.name, data: { index: index.toString() } }),
|
||||
create_element("li", { text: model.name, data: { index: index.toString() } }),
|
||||
);
|
||||
});
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { create_element } from "../../core/gui/dom";
|
||||
import { ResizableView } from "../../core/gui/ResizableView";
|
||||
import { FileButton } from "../../core/gui/FileButton";
|
||||
import { ToolBar } from "../../core/gui/ToolBar";
|
||||
@ -8,7 +8,7 @@ import { TextureRenderer } from "../rendering/TextureRenderer";
|
||||
import { gui_store, GuiTool } from "../../core/stores/GuiStore";
|
||||
|
||||
export class TextureView extends ResizableView {
|
||||
readonly element = el("div", { class: "viewer_TextureView" });
|
||||
readonly element = create_element("div", { class: "viewer_TextureView" });
|
||||
|
||||
private readonly open_file_button = new FileButton("Open file...", ".xvm");
|
||||
|
||||
|
@ -36,6 +36,7 @@ export class ModelRenderer extends Renderer implements Disposable {
|
||||
clip: AnimationClip;
|
||||
action: AnimationAction;
|
||||
};
|
||||
private update_animation_time = true;
|
||||
|
||||
constructor() {
|
||||
super(new PerspectiveCamera(75, 1, 1, 200));
|
||||
@ -201,7 +202,11 @@ export class ModelRenderer extends Renderer implements Disposable {
|
||||
const frame_count = nj_motion.frame_count;
|
||||
if (frame > frame_count) frame = 1;
|
||||
if (frame < 1) frame = frame_count;
|
||||
this.animation.action.time = (frame - 1) / PSO_FRAME_RATE;
|
||||
|
||||
if (this.update_animation_time) {
|
||||
this.animation.action.time = (frame - 1) / PSO_FRAME_RATE;
|
||||
}
|
||||
|
||||
this.schedule_render();
|
||||
}
|
||||
};
|
||||
@ -209,7 +214,9 @@ export class ModelRenderer extends Renderer implements Disposable {
|
||||
private update_animation_frame(): void {
|
||||
if (this.animation && !this.animation.action.paused) {
|
||||
const time = this.animation.action.time;
|
||||
this.update_animation_time = false;
|
||||
model_store.animation_frame.val = time * PSO_FRAME_RATE + 1;
|
||||
this.update_animation_time = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user