diff --git a/src/application/index.test.ts b/src/application/index.test.ts index 206fc2f9..37b30c05 100644 --- a/src/application/index.test.ts +++ b/src/application/index.test.ts @@ -1,8 +1,9 @@ import { initialize_application } from "./index"; -import { DisposableThreeRenderer } from "../core/rendering/Renderer"; import { LogHandler, LogLevel, LogManager } from "../core/Logger"; import { FileSystemHttpClient } from "../../test/src/core/FileSystemHttpClient"; import { timeout } from "../../test/src/utils"; +import { StubThreeRenderer } from "../../test/src/core/rendering/StubThreeRenderer"; +import { Random } from "../core/Random"; for (const path of [undefined, "/viewer", "/quest_editor", "/hunt_optimizer"]) { const with_path = path == undefined ? "without specific path" : `with path ${path}`; @@ -23,7 +24,8 @@ for (const path of [undefined, "/viewer", "/quest_editor", "/hunt_optimizer"]) { const app = initialize_application( new FileSystemHttpClient(), - () => new StubRenderer(), + new Random(() => 0.27), + () => new StubThreeRenderer(), ); expect(app).toBeDefined(); @@ -37,13 +39,3 @@ for (const path of [undefined, "/viewer", "/quest_editor", "/hunt_optimizer"]) { }); }); } - -class StubRenderer implements DisposableThreeRenderer { - domElement: HTMLCanvasElement = document.createElement("canvas"); - - dispose(): void {} // eslint-disable-line - - render(): void {} // eslint-disable-line - - setSize(): void {} // eslint-disable-line -} diff --git a/src/application/index.ts b/src/application/index.ts index 8dfb5b35..7d743ef6 100644 --- a/src/application/index.ts +++ b/src/application/index.ts @@ -8,9 +8,11 @@ import { throttle } from "lodash"; import { DisposableThreeRenderer } from "../core/rendering/Renderer"; import { Disposer } from "../core/observable/Disposer"; import { disposable_custom_listener, disposable_listener } from "../core/gui/dom"; +import { Random } from "../core/Random"; export function initialize_application( http_client: HttpClient, + random: Random, create_three_renderer: () => DisposableThreeRenderer, ): Disposable { const disposer = new Disposer(); @@ -43,7 +45,7 @@ export function initialize_application( async () => { const { initialize_viewer } = await import("../viewer"); const viewer = disposer.add( - initialize_viewer(http_client, gui_store, create_three_renderer), + initialize_viewer(http_client, random, gui_store, create_three_renderer), ); return viewer.view; diff --git a/src/core/Random.ts b/src/core/Random.ts new file mode 100644 index 00000000..d2c5840b --- /dev/null +++ b/src/core/Random.ts @@ -0,0 +1,18 @@ +export class Random { + constructor(private readonly random_number: () => number = Math.random) {} + /** + * @param min - The minimum value, inclusive. + * @param max - The maximum value, exclusive. + * @returns A random integer between `min` and `max`. + */ + integer(min: number, max: number): number { + return min + Math.floor(this.random_number() * (max - min)); + } + + /** + * @returns A random element from `array`. + */ + sample_array(array: readonly T[]): T { + return array[this.integer(0, array.length)]; + } +} diff --git a/src/core/gui/Select.ts b/src/core/gui/Select.ts index c5337c05..a3718e3a 100644 --- a/src/core/gui/Select.ts +++ b/src/core/gui/Select.ts @@ -9,7 +9,7 @@ import { Menu } from "./Menu"; export type SelectOptions = LabelledControlOptions & { readonly items: readonly T[] | Property; - readonly to_label: (element: T) => string; + readonly to_label?: (element: T) => string; readonly selected?: T | Property; }; @@ -31,7 +31,7 @@ export class Select extends LabelledControl { this.preferred_label_position = "left"; - this.to_label = options.to_label; + this.to_label = options.to_label ?? String; this.button = this.disposable( new Button({ text: " ", diff --git a/src/core/util.ts b/src/core/util.ts index 4fdc323c..505f7d34 100644 --- a/src/core/util.ts +++ b/src/core/util.ts @@ -34,22 +34,6 @@ export function array_remove(array: T[], ...elements: T[]): number { return count; } -/** - * @param min - The minimum value, inclusive. - * @param max - The maximum value, exclusive. - * @returns A random integer between `min` and `max`. - */ -export function random_integer(min: number, max: number): number { - return min + Math.floor(Math.random() * (max - min)); -} - -/** - * @returns A random element from `array`. - */ -export function random_array_element(array: readonly T[]): T { - return array[random_integer(0, array.length)]; -} - export function array_buffers_equal(a: ArrayBuffer, b: ArrayBuffer): boolean { if (a.byteLength !== b.byteLength) return false; diff --git a/src/index.ts b/src/index.ts index ae08941f..c6b0a612 100644 --- a/src/index.ts +++ b/src/index.ts @@ -10,6 +10,7 @@ import { initialize_application } from "./application"; import { FetchClient } from "./core/HttpClient"; import { WebGLRenderer } from "three"; import { DisposableThreeRenderer } from "./core/rendering/Renderer"; +import { Random } from "./core/Random"; function create_three_renderer(): DisposableThreeRenderer { const renderer = new WebGLRenderer({ antialias: true, alpha: true }); @@ -17,4 +18,4 @@ function create_three_renderer(): DisposableThreeRenderer { return renderer; } -initialize_application(new FetchClient(), create_three_renderer); +initialize_application(new FetchClient(), new Random(), create_three_renderer); diff --git a/src/viewer/stores/TextureStore.ts b/src/viewer/controllers/TextureController.ts similarity index 90% rename from src/viewer/stores/TextureStore.ts rename to src/viewer/controllers/TextureController.ts index 7176427d..550c456f 100644 --- a/src/viewer/stores/TextureStore.ts +++ b/src/viewer/controllers/TextureController.ts @@ -1,18 +1,18 @@ -import { list_property } from "../../core/observable"; -import { parse_xvm, XvrTexture } from "../../core/data_formats/parsing/ninja/texture"; +import { Controller } from "../../core/controllers/Controller"; +import { filename_extension } from "../../core/util"; import { read_file } from "../../core/read_file"; +import { parse_xvm, XvrTexture } from "../../core/data_formats/parsing/ninja/texture"; import { ArrayBufferCursor } from "../../core/data_formats/cursor/ArrayBufferCursor"; import { Endianness } from "../../core/data_formats/Endianness"; -import { Store } from "../../core/stores/Store"; +import { parse_afs } from "../../core/data_formats/parsing/afs"; import { LogManager } from "../../core/Logger"; import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty"; +import { list_property } from "../../core/observable"; import { ListProperty } from "../../core/observable/property/list/ListProperty"; -import { filename_extension } from "../../core/util"; -import { parse_afs } from "../../core/data_formats/parsing/afs"; -const logger = LogManager.get("viewer/stores/TextureStore"); +const logger = LogManager.get("viewer/controllers/TextureController"); -export class TextureStore extends Store { +export class TextureController extends Controller { private readonly _textures: WritableListProperty = list_property(); readonly textures: ListProperty = this._textures; diff --git a/src/viewer/controllers/model/CharacterClassOptionsController.ts b/src/viewer/controllers/model/CharacterClassOptionsController.ts new file mode 100644 index 00000000..59362409 --- /dev/null +++ b/src/viewer/controllers/model/CharacterClassOptionsController.ts @@ -0,0 +1,38 @@ +import { Controller } from "../../../core/controllers/Controller"; +import { ModelStore } from "../../stores/ModelStore"; +import { range } from "lodash"; +import { Property } from "../../../core/observable/property/Property"; +import { SectionId } from "../../../core/model"; + +export class CharacterClassOptionsController extends Controller { + readonly enabled: Property; + readonly current_section_id: Property; + readonly current_body_options: Property; + readonly current_body: Property; + + constructor(private readonly store: ModelStore) { + super(); + + this.enabled = store.current_character_class.map(cc => cc != undefined); + + this.current_section_id = store.current_section_id; + + this.current_body_options = store.current_character_class.map(character_class => + character_class ? range(1, character_class.body_style_count + 1) : [], + ); + + this.current_body = store.current_body.map(body => + body == undefined ? undefined : body + 1, + ); + } + + set_current_section_id = (section_id?: SectionId): void => { + if (section_id != undefined) { + this.store.set_current_section_id(section_id); + } + }; + + set_current_body = (body?: number): void => { + this.store.set_current_body(body == undefined ? undefined : body - 1); + }; +} diff --git a/src/viewer/controllers/model/ModelController.ts b/src/viewer/controllers/model/ModelController.ts new file mode 100644 index 00000000..da309eb8 --- /dev/null +++ b/src/viewer/controllers/model/ModelController.ts @@ -0,0 +1,31 @@ +import { Controller } from "../../../core/controllers/Controller"; +import { ModelStore } from "../../stores/ModelStore"; +import { CharacterClassModel } from "../../model/CharacterClassModel"; +import { Property } from "../../../core/observable/property/Property"; +import { CharacterClassAnimationModel } from "../../model/CharacterClassAnimationModel"; + +export class ModelController extends Controller { + readonly character_classes: readonly CharacterClassModel[]; + readonly current_character_class: Property; + + readonly animations: readonly CharacterClassAnimationModel[]; + readonly current_animation: Property; + + constructor(private readonly store: ModelStore) { + super(); + + this.character_classes = store.character_classes; + this.current_character_class = store.current_character_class; + + this.animations = store.animations; + this.current_animation = store.current_animation; + } + + set_current_character_class = (character_class: CharacterClassModel): void => { + this.store.set_current_character_class(character_class); + }; + + set_current_animation = (animation: CharacterClassAnimationModel): void => { + this.store.set_current_animation(animation); + }; +} diff --git a/src/viewer/controllers/model/ModelToolBarController.ts b/src/viewer/controllers/model/ModelToolBarController.ts new file mode 100644 index 00000000..b5f20caf --- /dev/null +++ b/src/viewer/controllers/model/ModelToolBarController.ts @@ -0,0 +1,92 @@ +import { Controller } from "../../../core/controllers/Controller"; +import { Property } from "../../../core/observable/property/Property"; +import { ModelStore } from "../../stores/ModelStore"; +import { read_file } from "../../../core/read_file"; +import { ArrayBufferCursor } from "../../../core/data_formats/cursor/ArrayBufferCursor"; +import { Endianness } from "../../../core/data_formats/Endianness"; +import { parse_nj, parse_xj } from "../../../core/data_formats/parsing/ninja"; +import { parse_njm } from "../../../core/data_formats/parsing/ninja/motion"; +import { parse_xvm, XvrTexture } from "../../../core/data_formats/parsing/ninja/texture"; +import { parse_afs } from "../../../core/data_formats/parsing/afs"; +import { LogManager } from "../../../core/Logger"; + +const logger = LogManager.get("viewer/controllers/model/ModelToolBarController"); + +export class ModelToolBarController extends Controller { + readonly show_skeleton: Property; + readonly animation_frame_count: Property; + readonly animation_frame_count_label: Property; + readonly animation_controls_enabled: Property; + readonly animation_playing: Property; + readonly animation_frame_rate: Property; + readonly animation_frame: Property; + + constructor(private readonly store: ModelStore) { + super(); + + this.show_skeleton = store.show_skeleton; + this.animation_frame_count = store.animation_frame_count; + this.animation_frame_count_label = store.animation_frame_count.map(count => `/ ${count}`); + this.animation_controls_enabled = store.current_nj_motion.map(njm => njm != undefined); + this.animation_playing = store.animation_playing; + this.animation_frame_rate = store.animation_frame_rate; + this.animation_frame = store.animation_frame.map(v => Math.round(v)); + } + + set_show_skeleton = (show_skeleton: boolean): void => { + this.store.set_show_skeleton(show_skeleton); + }; + + set_animation_playing = (playing: boolean): void => { + this.store.set_animation_playing(playing); + }; + + set_animation_frame_rate = (frame_rate: number): void => { + this.store.set_animation_frame_rate(frame_rate); + }; + + set_animation_frame = (frame: number): void => { + this.store.set_animation_frame(frame); + }; + + // TODO: notify user of problems. + load_file = async (file: File): Promise => { + try { + const buffer = await read_file(file); + const cursor = new ArrayBufferCursor(buffer, Endianness.Little); + + if (file.name.endsWith(".nj")) { + const nj_object = parse_nj(cursor)[0]; + + this.store.set_current_nj_object(nj_object); + } else if (file.name.endsWith(".xj")) { + const nj_object = parse_xj(cursor)[0]; + + this.store.set_current_nj_object(nj_object); + } else if (file.name.endsWith(".njm")) { + this.store.set_current_animation(undefined); + this.store.set_current_nj_motion(undefined); + + const nj_object = this.store.current_nj_object.val; + + if (nj_object) { + this.set_animation_playing(true); + this.store.set_current_nj_motion(parse_njm(cursor, nj_object.bone_count())); + } + } else if (file.name.endsWith(".xvm")) { + this.store.set_current_textures(parse_xvm(cursor).textures); + } else if (file.name.endsWith(".afs")) { + const files = parse_afs(cursor); + const textures: XvrTexture[] = files.flatMap( + file => parse_xvm(new ArrayBufferCursor(file, Endianness.Little)).textures, + ); + + this.store.set_current_textures(textures); + } else { + logger.error(`Unknown file extension in filename "${file.name}".`); + } + } catch (e) { + logger.error("Couldn't read file.", e); + } + }; +} diff --git a/src/viewer/gui/TextureView.test.ts b/src/viewer/gui/TextureView.test.ts new file mode 100644 index 00000000..2ac40256 --- /dev/null +++ b/src/viewer/gui/TextureView.test.ts @@ -0,0 +1,15 @@ +import { TextureView } from "./TextureView"; +import { with_disposer } from "../../../test/src/core/observables/disposable_helpers"; +import { TextureController } from "../controllers/TextureController"; +import { TextureRenderer } from "../rendering/TextureRenderer"; +import { StubThreeRenderer } from "../../../test/src/core/rendering/StubThreeRenderer"; + +test("Renders correctly without textures.", () => + with_disposer(disposer => { + const ctrl = disposer.add(new TextureController()); + const view = disposer.add( + new TextureView(ctrl, new TextureRenderer(ctrl, new StubThreeRenderer())), + ); + + expect(view.element).toMatchSnapshot("Should render a toolbar and a renderer widget."); + })); diff --git a/src/viewer/gui/TextureView.ts b/src/viewer/gui/TextureView.ts index 03d0138b..3d8d0c4c 100644 --- a/src/viewer/gui/TextureView.ts +++ b/src/viewer/gui/TextureView.ts @@ -3,9 +3,8 @@ import { FileButton } from "../../core/gui/FileButton"; import { ToolBar } from "../../core/gui/ToolBar"; import { RendererWidget } from "../../core/gui/RendererWidget"; import { TextureRenderer } from "../rendering/TextureRenderer"; -import { TextureStore } from "../stores/TextureStore"; -import { DisposableThreeRenderer } from "../../core/rendering/Renderer"; import { ResizableView } from "../../core/gui/ResizableView"; +import { TextureController } from "../controllers/TextureController"; export class TextureView extends ResizableView { readonly element = div({ className: "viewer_TextureView" }); @@ -19,18 +18,16 @@ export class TextureView extends ResizableView { private readonly renderer_view: RendererWidget; - constructor(texture_store: TextureStore, three_renderer: DisposableThreeRenderer) { + constructor(ctrl: TextureController, renderer: TextureRenderer) { super(); - this.renderer_view = this.add( - new RendererWidget(new TextureRenderer(three_renderer, texture_store)), - ); + this.renderer_view = this.add(new RendererWidget(renderer)); this.element.append(this.tool_bar.element, this.renderer_view.element); this.disposables( this.open_file_button.files.observe(({ value: files }) => { - if (files.length) texture_store.load_file(files[0]); + if (files.length) ctrl.load_file(files[0]); }), ); diff --git a/src/viewer/gui/ViewerView.ts b/src/viewer/gui/ViewerView.ts index 843b7833..80e65872 100644 --- a/src/viewer/gui/ViewerView.ts +++ b/src/viewer/gui/ViewerView.ts @@ -1,5 +1,5 @@ import { TabContainer } from "../../core/gui/TabContainer"; -import { Model3DView } from "./model_3d/Model3DView"; +import { ModelView } from "./model/ModelView"; import { TextureView } from "./TextureView"; import { ResizableView } from "../../core/gui/ResizableView"; import { GuiStore } from "../../core/stores/GuiStore"; @@ -13,7 +13,7 @@ export class ViewerView extends ResizableView { constructor( gui_store: GuiStore, - create_model_3d_view: () => Promise, + create_model_view: () => Promise, create_texture_view: () => Promise, ) { super(); @@ -24,13 +24,13 @@ export class ViewerView extends ResizableView { tabs: [ { title: "Models", - key: "models", + key: "model", path: "/models", - create_view: create_model_3d_view, + create_view: create_model_view, }, { title: "Textures", - key: "textures", + key: "texture", path: "/textures", create_view: create_texture_view, }, diff --git a/src/viewer/gui/__snapshots__/TextureView.test.ts.snap b/src/viewer/gui/__snapshots__/TextureView.test.ts.snap new file mode 100644 index 00000000..323187ec --- /dev/null +++ b/src/viewer/gui/__snapshots__/TextureView.test.ts.snap @@ -0,0 +1,48 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`Renders correctly without textures.: Should render a toolbar and a renderer widget. 1`] = ` +
+
+ +
+
+ +
+
+`; diff --git a/src/viewer/gui/model/CharacterClassOptionsView.css b/src/viewer/gui/model/CharacterClassOptionsView.css new file mode 100644 index 00000000..99a12314 --- /dev/null +++ b/src/viewer/gui/model/CharacterClassOptionsView.css @@ -0,0 +1,23 @@ +.viewer_model_CharacterClassOptionsView { + box-sizing: border-box; + border-left: var(--border); + border-right: var(--border); + padding: 0 2px; +} + +.viewer_model_CharacterClassOptionsView table { + table-layout: fixed; + width: 100%; +} + +.viewer_model_CharacterClassOptionsView td:first-child { + width: 80px; +} + +.viewer_model_CharacterClassOptionsView_section_id { + width: 100%; +} + +.viewer_model_CharacterClassOptionsView_body { + width: 60px; +} diff --git a/src/viewer/gui/model/CharacterClassOptionsView.ts b/src/viewer/gui/model/CharacterClassOptionsView.ts new file mode 100644 index 00000000..419ca8c4 --- /dev/null +++ b/src/viewer/gui/model/CharacterClassOptionsView.ts @@ -0,0 +1,50 @@ +import { ResizableView } from "../../../core/gui/ResizableView"; +import { div, table, td, tr } from "../../../core/gui/dom"; +import { Select } from "../../../core/gui/Select"; +import { CharacterClassOptionsController } from "../../controllers/model/CharacterClassOptionsController"; +import "./CharacterClassOptionsView.css"; +import { SectionId, SectionIds } from "../../../core/model"; + +export class CharacterClassOptionsView extends ResizableView { + readonly element: HTMLElement; + + constructor(ctrl: CharacterClassOptionsController) { + super(); + + const section_id_select: Select = this.add( + new Select({ + class: "viewer_model_CharacterClassOptionsView_section_id", + label: "Section ID:", + items: SectionIds, + selected: ctrl.current_section_id, + to_label: section_id => SectionId[section_id], + enabled: ctrl.enabled, + }), + ); + + const body_select: Select = this.add( + new Select({ + class: "viewer_model_CharacterClassOptionsView_body", + label: "Body:", + items: ctrl.current_body_options, + selected: ctrl.current_body, + enabled: ctrl.enabled, + }), + ); + + this.element = div( + { className: "viewer_model_CharacterClassOptionsView" }, + table( + tr(td(section_id_select.label?.element), td(section_id_select.element)), + tr(td(body_select.label?.element), td(body_select.element)), + ), + ); + + this.disposables( + section_id_select.selected.observe(({ value }) => ctrl.set_current_section_id(value)), + body_select.selected.observe(({ value }) => ctrl.set_current_body(value)), + ); + + this.finalize_construction(); + } +} diff --git a/src/viewer/gui/model_3d/Model3DSelectListView.css b/src/viewer/gui/model/CharacterClassSelectionView.css similarity index 57% rename from src/viewer/gui/model_3d/Model3DSelectListView.css rename to src/viewer/gui/model/CharacterClassSelectionView.css index 7b409738..5ebe9311 100644 --- a/src/viewer/gui/model_3d/Model3DSelectListView.css +++ b/src/viewer/gui/model/CharacterClassSelectionView.css @@ -1,4 +1,4 @@ -.viewer_Model3DSelectListView { +.viewer_model_CharacterClassSelectionView { box-sizing: border-box; list-style: none; padding: 0; @@ -6,16 +6,16 @@ overflow: auto; } -.viewer_Model3DSelectListView li { +.viewer_model_CharacterClassSelectionView li { padding: 4px 8px; } -.viewer_Model3DSelectListView li:hover { +.viewer_model_CharacterClassSelectionView li:hover { color: hsl(0, 0%, 90%); background-color: hsl(0, 0%, 18%); } -.viewer_Model3DSelectListView li.active { +.viewer_model_CharacterClassSelectionView li.active { color: hsl(0, 0%, 90%); background-color: hsl(0, 0%, 21%); } diff --git a/src/viewer/gui/model_3d/Model3DSelectListView.ts b/src/viewer/gui/model/CharacterClassSelectionView.ts similarity index 72% rename from src/viewer/gui/model_3d/Model3DSelectListView.ts rename to src/viewer/gui/model/CharacterClassSelectionView.ts index d9f1ed93..dc985c30 100644 --- a/src/viewer/gui/model_3d/Model3DSelectListView.ts +++ b/src/viewer/gui/model/CharacterClassSelectionView.ts @@ -1,35 +1,30 @@ -import "./Model3DSelectListView.css"; +import "./CharacterClassSelectionView.css"; import { Property } from "../../../core/observable/property/Property"; import { li, ul } from "../../../core/gui/dom"; import { ResizableView } from "../../../core/gui/ResizableView"; -export class Model3DSelectListView extends ResizableView { - readonly element = ul({ className: "viewer_Model3DSelectListView" }); - - set borders(borders: boolean) { - if (borders) { - this.element.style.borderLeft = "var(--border)"; - this.element.style.borderRight = "var(--border)"; - } else { - this.element.style.borderLeft = "none"; - this.element.style.borderRight = "none"; - } - } - +export class CharacterClassSelectionView extends ResizableView { private selected_model?: T; private selected_element?: HTMLLIElement; + readonly element = ul({ className: "viewer_model_CharacterClassSelectionView" }); + constructor( - private models: readonly T[], + private character_classes: readonly T[], private selected: Property, private set_selected: (selected: T) => void, + private border_left: boolean, ) { super(); this.element.onclick = this.list_click; - models.forEach((model, index) => { - this.element.append(li({ data: { index: index.toString() } }, model.name)); + if (border_left) { + this.element.style.borderLeft = "var(--border)"; + } + + character_classes.forEach((character_class, index) => { + this.element.append(li({ data: { index: index.toString() } }, character_class.name)); }); this.disposables( @@ -41,7 +36,7 @@ export class Model3DSelectListView extends Resizable } if (model && model !== this.selected_model) { - const index = this.models.indexOf(model); + const index = this.character_classes.indexOf(model); if (index !== -1) { this.selected_element = this.element.childNodes[index] as HTMLLIElement; @@ -67,7 +62,7 @@ export class Model3DSelectListView extends Resizable const index = parseInt(e.target.dataset["index"]!, 10); this.selected_element = e.target; - this.set_selected(this.models[index]); + this.set_selected(this.character_classes[index]); } }; } diff --git a/src/viewer/gui/model_3d/Model3DToolBarView.ts b/src/viewer/gui/model/ModelToolBarView.ts similarity index 67% rename from src/viewer/gui/model_3d/Model3DToolBarView.ts rename to src/viewer/gui/model/ModelToolBarView.ts index 861f1b56..c8201c5c 100644 --- a/src/viewer/gui/model_3d/Model3DToolBarView.ts +++ b/src/viewer/gui/model/ModelToolBarView.ts @@ -5,10 +5,10 @@ import { NumberInput } from "../../../core/gui/NumberInput"; import { PSO_FRAME_RATE } from "../../../core/rendering/conversion/ninja_animation"; import { Label } from "../../../core/gui/Label"; import { Icon } from "../../../core/gui/dom"; -import { Model3DStore } from "../../stores/Model3DStore"; import { View } from "../../../core/gui/View"; +import { ModelToolBarController } from "../../controllers/model/ModelToolBarController"; -export class Model3DToolBarView extends View { +export class ModelToolBarView extends View { private readonly toolbar: ToolBar; get element(): HTMLElement { @@ -19,7 +19,7 @@ export class Model3DToolBarView extends View { return this.toolbar.height; } - constructor(model_3d_store: Model3DStore) { + constructor(ctrl: ModelToolBarController) { super(); const open_file_button = new FileButton("Open file...", { @@ -37,12 +37,10 @@ export class Model3DToolBarView extends View { const animation_frame_input = new NumberInput(1, { label: "Frame:", min: 1, - max: model_3d_store.animation_frame_count, + max: ctrl.animation_frame_count, step: 1, }); - const animation_frame_count_label = new Label( - model_3d_store.animation_frame_count.map(count => `/ ${count}`), - ); + const animation_frame_count_label = new Label(ctrl.animation_frame_count_label); this.toolbar = this.add( new ToolBar( @@ -58,36 +56,30 @@ export class Model3DToolBarView extends View { // Always-enabled controls. this.disposables( open_file_button.files.observe(({ value: files }) => { - if (files.length) model_3d_store.load_file(files[0]); + if (files.length) ctrl.load_file(files[0]); }), - skeleton_checkbox.checked.observe(({ value }) => - model_3d_store.set_show_skeleton(value), - ), + skeleton_checkbox.checked.observe(({ value }) => ctrl.set_show_skeleton(value)), ); // Controls that are only enabled when an animation is selected. - const enabled = model_3d_store.current_nj_motion.map(njm => njm != undefined); + const enabled = ctrl.animation_controls_enabled; this.disposables( play_animation_checkbox.enabled.bind_to(enabled), - play_animation_checkbox.checked.bind_to(model_3d_store.animation_playing), + play_animation_checkbox.checked.bind_to(ctrl.animation_playing), play_animation_checkbox.checked.observe(({ value }) => - model_3d_store.set_animation_playing(value), + ctrl.set_animation_playing(value), ), animation_frame_rate_input.enabled.bind_to(enabled), animation_frame_rate_input.value.observe(({ value }) => - model_3d_store.set_animation_frame_rate(value), + ctrl.set_animation_frame_rate(value), ), animation_frame_input.enabled.bind_to(enabled), - animation_frame_input.value.bind_to( - model_3d_store.animation_frame.map(v => Math.round(v)), - ), - animation_frame_input.value.observe(({ value }) => - model_3d_store.set_animation_frame(value), - ), + animation_frame_input.value.bind_to(ctrl.animation_frame), + animation_frame_input.value.observe(({ value }) => ctrl.set_animation_frame(value)), animation_frame_count_label.enabled.bind_to(enabled), ); diff --git a/src/viewer/gui/model_3d/Model3DView.css b/src/viewer/gui/model/ModelView.css similarity index 56% rename from src/viewer/gui/model_3d/Model3DView.css rename to src/viewer/gui/model/ModelView.css index 8b796079..ce56ab08 100644 --- a/src/viewer/gui/model_3d/Model3DView.css +++ b/src/viewer/gui/model/ModelView.css @@ -1,4 +1,4 @@ -.viewer_Model3DView_container { +.viewer_model_ModelView_container { display: flex; flex-direction: row; } diff --git a/src/viewer/gui/model/ModelView.test.ts b/src/viewer/gui/model/ModelView.test.ts new file mode 100644 index 00000000..89e0190e --- /dev/null +++ b/src/viewer/gui/model/ModelView.test.ts @@ -0,0 +1,31 @@ +import { with_disposer } from "../../../../test/src/core/observables/disposable_helpers"; +import { ModelController } from "../../controllers/model/ModelController"; +import { CharacterClassAssetLoader } from "../../loading/CharacterClassAssetLoader"; +import { FileSystemHttpClient } from "../../../../test/src/core/FileSystemHttpClient"; +import { ModelView } from "./ModelView"; +import { ModelRenderer } from "../../rendering/ModelRenderer"; +import { StubThreeRenderer } from "../../../../test/src/core/rendering/StubThreeRenderer"; +import { Random } from "../../../core/Random"; +import { ModelStore } from "../../stores/ModelStore"; +import { ModelToolBarView } from "./ModelToolBarView"; +import { ModelToolBarController } from "../../controllers/model/ModelToolBarController"; +import { CharacterClassOptionsView } from "./CharacterClassOptionsView"; +import { CharacterClassOptionsController } from "../../controllers/model/CharacterClassOptionsController"; + +test("Renders correctly.", () => + with_disposer(disposer => { + const store = disposer.add( + new ModelStore( + disposer.add(new CharacterClassAssetLoader(new FileSystemHttpClient())), + new Random(() => 0.04), + ), + ); + const view = new ModelView( + disposer.add(new ModelController(store)), + new ModelToolBarView(disposer.add(new ModelToolBarController(store))), + new CharacterClassOptionsView(disposer.add(new CharacterClassOptionsController(store))), + new ModelRenderer(store, new StubThreeRenderer()), + ); + + expect(view.element).toMatchSnapshot(); + })); diff --git a/src/viewer/gui/model/ModelView.ts b/src/viewer/gui/model/ModelView.ts new file mode 100644 index 00000000..599e0161 --- /dev/null +++ b/src/viewer/gui/model/ModelView.ts @@ -0,0 +1,102 @@ +import "./ModelView.css"; +import { RendererWidget } from "../../../core/gui/RendererWidget"; +import { ModelRenderer } from "../../rendering/ModelRenderer"; +import { ModelToolBarView } from "./ModelToolBarView"; +import { CharacterClassSelectionView } from "./CharacterClassSelectionView"; +import { CharacterClassModel } from "../../model/CharacterClassModel"; +import { CharacterClassAnimationModel } from "../../model/CharacterClassAnimationModel"; +import { ModelController } from "../../controllers/model/ModelController"; +import { div } from "../../../core/gui/dom"; +import { ResizableView } from "../../../core/gui/ResizableView"; +import { CharacterClassOptionsView } from "./CharacterClassOptionsView"; + +const CHARACTER_CLASS_SELECTION_WIDTH = 100; +const CHARACTER_CLASS_OPTIONS_WIDTH = 220; +const ANIMATION_SELECTION_WIDTH = 140; + +export class ModelView extends ResizableView { + readonly element = div({ className: "viewer_model_ModelView" }); + + private tool_bar_view: ModelToolBarView; + private character_class_selection_view: CharacterClassSelectionView; + private options_view: CharacterClassOptionsView; + private renderer_view: RendererWidget; + private animation_selection_view: CharacterClassSelectionView; + + constructor( + ctrl: ModelController, + tool_bar_view: ModelToolBarView, + options_view: CharacterClassOptionsView, + renderer: ModelRenderer, + ) { + super(); + + this.tool_bar_view = this.add(tool_bar_view); + this.character_class_selection_view = this.add( + new CharacterClassSelectionView( + ctrl.character_classes, + ctrl.current_character_class, + ctrl.set_current_character_class, + false, + ), + ); + this.options_view = this.add(options_view); + this.renderer_view = this.add(new RendererWidget(renderer)); + this.animation_selection_view = this.add( + new CharacterClassSelectionView( + ctrl.animations, + ctrl.current_animation, + ctrl.set_current_animation, + true, + ), + ); + + this.element.append( + this.tool_bar_view.element, + div( + { className: "viewer_model_ModelView_container" }, + this.character_class_selection_view.element, + this.options_view.element, + this.renderer_view.element, + this.animation_selection_view.element, + ), + ); + + this.finalize_construction(); + } + + activate(): void { + this.renderer_view.start_rendering(); + super.activate(); + } + + deactivate(): void { + super.deactivate(); + this.renderer_view.stop_rendering(); + } + + resize(width: number, height: number): this { + super.resize(width, height); + + const container_height = Math.max(0, height - this.tool_bar_view.height); + + this.character_class_selection_view.resize( + CHARACTER_CLASS_SELECTION_WIDTH, + container_height, + ); + this.options_view.resize(CHARACTER_CLASS_OPTIONS_WIDTH, container_height); + this.renderer_view.resize( + Math.max( + 0, + width - + CHARACTER_CLASS_SELECTION_WIDTH - + CHARACTER_CLASS_OPTIONS_WIDTH - + ANIMATION_SELECTION_WIDTH, + ), + container_height, + ); + this.animation_selection_view.resize(ANIMATION_SELECTION_WIDTH, container_height); + + return this; + } +} diff --git a/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap b/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap new file mode 100644 index 00000000..323e5c70 --- /dev/null +++ b/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap @@ -0,0 +1,3313 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`Renders correctly. 1`] = ` +
+
+ +
+ + +
+
+ + +
+
+ + + + +
+
+ + + + +
+ +
+
+
    +
  • + HUmar +
  • +
  • + HUnewearl +
  • +
  • + HUcast +
  • +
  • + HUcaseal +
  • +
  • + RAmar +
  • +
  • + RAmarl +
  • +
  • + RAcast +
  • +
  • + RAcaseal +
  • +
  • + FOmar +
  • +
  • + FOmarl +
  • +
  • + FOnewm +
  • +
  • + FOnewearl +
  • +
+
+ + + + + + + + + +
+ + +
+ + +
+
+ + +
+ + +
+
+
+
+ +
+
    +
  • + Animation 1 +
  • +
  • + Animation 2 +
  • +
  • + Animation 3 +
  • +
  • + Animation 4 +
  • +
  • + Animation 5 +
  • +
  • + Animation 6 +
  • +
  • + Animation 7 +
  • +
  • + Animation 8 +
  • +
  • + Animation 9 +
  • +
  • + Animation 10 +
  • +
  • + Animation 11 +
  • +
  • + Animation 12 +
  • +
  • + Animation 13 +
  • +
  • + Animation 14 +
  • +
  • + Animation 15 +
  • +
  • + Animation 16 +
  • +
  • + Animation 17 +
  • +
  • + Animation 18 +
  • +
  • + Animation 19 +
  • +
  • + Animation 20 +
  • +
  • + Animation 21 +
  • +
  • + Animation 22 +
  • +
  • + Animation 23 +
  • +
  • + Animation 24 +
  • +
  • + Animation 25 +
  • +
  • + Animation 26 +
  • +
  • + Animation 27 +
  • +
  • + Animation 28 +
  • +
  • + Animation 29 +
  • +
  • + Animation 30 +
  • +
  • + Animation 31 +
  • +
  • + Animation 32 +
  • +
  • + Animation 33 +
  • +
  • + Animation 34 +
  • +
  • + Animation 35 +
  • +
  • + Animation 36 +
  • +
  • + Animation 37 +
  • +
  • + Animation 38 +
  • +
  • + Animation 39 +
  • +
  • + Animation 40 +
  • +
  • + Animation 41 +
  • +
  • + Animation 42 +
  • +
  • + Animation 43 +
  • +
  • + Animation 44 +
  • +
  • + Animation 45 +
  • +
  • + Animation 46 +
  • +
  • + Animation 47 +
  • +
  • + Animation 48 +
  • +
  • + Animation 49 +
  • +
  • + Animation 50 +
  • +
  • + Animation 51 +
  • +
  • + Animation 52 +
  • +
  • + Animation 53 +
  • +
  • + Animation 54 +
  • +
  • + Animation 55 +
  • +
  • + Animation 56 +
  • +
  • + Animation 57 +
  • +
  • + Animation 58 +
  • +
  • + Animation 59 +
  • +
  • + Animation 60 +
  • +
  • + Animation 61 +
  • +
  • + Animation 62 +
  • +
  • + Animation 63 +
  • +
  • + Animation 64 +
  • +
  • + Animation 65 +
  • +
  • + Animation 66 +
  • +
  • + Animation 67 +
  • +
  • + Animation 68 +
  • +
  • + Animation 69 +
  • +
  • + Animation 70 +
  • +
  • + Animation 71 +
  • +
  • + Animation 72 +
  • +
  • + Animation 73 +
  • +
  • + Animation 74 +
  • +
  • + Animation 75 +
  • +
  • + Animation 76 +
  • +
  • + Animation 77 +
  • +
  • + Animation 78 +
  • +
  • + Animation 79 +
  • +
  • + Animation 80 +
  • +
  • + Animation 81 +
  • +
  • + Animation 82 +
  • +
  • + Animation 83 +
  • +
  • + Animation 84 +
  • +
  • + Animation 85 +
  • +
  • + Animation 86 +
  • +
  • + Animation 87 +
  • +
  • + Animation 88 +
  • +
  • + Animation 89 +
  • +
  • + Animation 90 +
  • +
  • + Animation 91 +
  • +
  • + Animation 92 +
  • +
  • + Animation 93 +
  • +
  • + Animation 94 +
  • +
  • + Animation 95 +
  • +
  • + Animation 96 +
  • +
  • + Animation 97 +
  • +
  • + Animation 98 +
  • +
  • + Animation 99 +
  • +
  • + Animation 100 +
  • +
  • + Animation 101 +
  • +
  • + Animation 102 +
  • +
  • + Animation 103 +
  • +
  • + Animation 104 +
  • +
  • + Animation 105 +
  • +
  • + Animation 106 +
  • +
  • + Animation 107 +
  • +
  • + Animation 108 +
  • +
  • + Animation 109 +
  • +
  • + Animation 110 +
  • +
  • + Animation 111 +
  • +
  • + Animation 112 +
  • +
  • + Animation 113 +
  • +
  • + Animation 114 +
  • +
  • + Animation 115 +
  • +
  • + Animation 116 +
  • +
  • + Animation 117 +
  • +
  • + Animation 118 +
  • +
  • + Animation 119 +
  • +
  • + Animation 120 +
  • +
  • + Animation 121 +
  • +
  • + Animation 122 +
  • +
  • + Animation 123 +
  • +
  • + Animation 124 +
  • +
  • + Animation 125 +
  • +
  • + Animation 126 +
  • +
  • + Animation 127 +
  • +
  • + Animation 128 +
  • +
  • + Animation 129 +
  • +
  • + Animation 130 +
  • +
  • + Animation 131 +
  • +
  • + Animation 132 +
  • +
  • + Animation 133 +
  • +
  • + Animation 134 +
  • +
  • + Animation 135 +
  • +
  • + Animation 136 +
  • +
  • + Animation 137 +
  • +
  • + Animation 138 +
  • +
  • + Animation 139 +
  • +
  • + Animation 140 +
  • +
  • + Animation 141 +
  • +
  • + Animation 142 +
  • +
  • + Animation 143 +
  • +
  • + Animation 144 +
  • +
  • + Animation 145 +
  • +
  • + Animation 146 +
  • +
  • + Animation 147 +
  • +
  • + Animation 148 +
  • +
  • + Animation 149 +
  • +
  • + Animation 150 +
  • +
  • + Animation 151 +
  • +
  • + Animation 152 +
  • +
  • + Animation 153 +
  • +
  • + Animation 154 +
  • +
  • + Animation 155 +
  • +
  • + Animation 156 +
  • +
  • + Animation 157 +
  • +
  • + Animation 158 +
  • +
  • + Animation 159 +
  • +
  • + Animation 160 +
  • +
  • + Animation 161 +
  • +
  • + Animation 162 +
  • +
  • + Animation 163 +
  • +
  • + Animation 164 +
  • +
  • + Animation 165 +
  • +
  • + Animation 166 +
  • +
  • + Animation 167 +
  • +
  • + Animation 168 +
  • +
  • + Animation 169 +
  • +
  • + Animation 170 +
  • +
  • + Animation 171 +
  • +
  • + Animation 172 +
  • +
  • + Animation 173 +
  • +
  • + Animation 174 +
  • +
  • + Animation 175 +
  • +
  • + Animation 176 +
  • +
  • + Animation 177 +
  • +
  • + Animation 178 +
  • +
  • + Animation 179 +
  • +
  • + Animation 180 +
  • +
  • + Animation 181 +
  • +
  • + Animation 182 +
  • +
  • + Animation 183 +
  • +
  • + Animation 184 +
  • +
  • + Animation 185 +
  • +
  • + Animation 186 +
  • +
  • + Animation 187 +
  • +
  • + Animation 188 +
  • +
  • + Animation 189 +
  • +
  • + Animation 190 +
  • +
  • + Animation 191 +
  • +
  • + Animation 192 +
  • +
  • + Animation 193 +
  • +
  • + Animation 194 +
  • +
  • + Animation 195 +
  • +
  • + Animation 196 +
  • +
  • + Animation 197 +
  • +
  • + Animation 198 +
  • +
  • + Animation 199 +
  • +
  • + Animation 200 +
  • +
  • + Animation 201 +
  • +
  • + Animation 202 +
  • +
  • + Animation 203 +
  • +
  • + Animation 204 +
  • +
  • + Animation 205 +
  • +
  • + Animation 206 +
  • +
  • + Animation 207 +
  • +
  • + Animation 208 +
  • +
  • + Animation 209 +
  • +
  • + Animation 210 +
  • +
  • + Animation 211 +
  • +
  • + Animation 212 +
  • +
  • + Animation 213 +
  • +
  • + Animation 214 +
  • +
  • + Animation 215 +
  • +
  • + Animation 216 +
  • +
  • + Animation 217 +
  • +
  • + Animation 218 +
  • +
  • + Animation 219 +
  • +
  • + Animation 220 +
  • +
  • + Animation 221 +
  • +
  • + Animation 222 +
  • +
  • + Animation 223 +
  • +
  • + Animation 224 +
  • +
  • + Animation 225 +
  • +
  • + Animation 226 +
  • +
  • + Animation 227 +
  • +
  • + Animation 228 +
  • +
  • + Animation 229 +
  • +
  • + Animation 230 +
  • +
  • + Animation 231 +
  • +
  • + Animation 232 +
  • +
  • + Animation 233 +
  • +
  • + Animation 234 +
  • +
  • + Animation 235 +
  • +
  • + Animation 236 +
  • +
  • + Animation 237 +
  • +
  • + Animation 238 +
  • +
  • + Animation 239 +
  • +
  • + Animation 240 +
  • +
  • + Animation 241 +
  • +
  • + Animation 242 +
  • +
  • + Animation 243 +
  • +
  • + Animation 244 +
  • +
  • + Animation 245 +
  • +
  • + Animation 246 +
  • +
  • + Animation 247 +
  • +
  • + Animation 248 +
  • +
  • + Animation 249 +
  • +
  • + Animation 250 +
  • +
  • + Animation 251 +
  • +
  • + Animation 252 +
  • +
  • + Animation 253 +
  • +
  • + Animation 254 +
  • +
  • + Animation 255 +
  • +
  • + Animation 256 +
  • +
  • + Animation 257 +
  • +
  • + Animation 258 +
  • +
  • + Animation 259 +
  • +
  • + Animation 260 +
  • +
  • + Animation 261 +
  • +
  • + Animation 262 +
  • +
  • + Animation 263 +
  • +
  • + Animation 264 +
  • +
  • + Animation 265 +
  • +
  • + Animation 266 +
  • +
  • + Animation 267 +
  • +
  • + Animation 268 +
  • +
  • + Animation 269 +
  • +
  • + Animation 270 +
  • +
  • + Animation 271 +
  • +
  • + Animation 272 +
  • +
  • + Animation 273 +
  • +
  • + Animation 274 +
  • +
  • + Animation 275 +
  • +
  • + Animation 276 +
  • +
  • + Animation 277 +
  • +
  • + Animation 278 +
  • +
  • + Animation 279 +
  • +
  • + Animation 280 +
  • +
  • + Animation 281 +
  • +
  • + Animation 282 +
  • +
  • + Animation 283 +
  • +
  • + Animation 284 +
  • +
  • + Animation 285 +
  • +
  • + Animation 286 +
  • +
  • + Animation 287 +
  • +
  • + Animation 288 +
  • +
  • + Animation 289 +
  • +
  • + Animation 290 +
  • +
  • + Animation 291 +
  • +
  • + Animation 292 +
  • +
  • + Animation 293 +
  • +
  • + Animation 294 +
  • +
  • + Animation 295 +
  • +
  • + Animation 296 +
  • +
  • + Animation 297 +
  • +
  • + Animation 298 +
  • +
  • + Animation 299 +
  • +
  • + Animation 300 +
  • +
  • + Animation 301 +
  • +
  • + Animation 302 +
  • +
  • + Animation 303 +
  • +
  • + Animation 304 +
  • +
  • + Animation 305 +
  • +
  • + Animation 306 +
  • +
  • + Animation 307 +
  • +
  • + Animation 308 +
  • +
  • + Animation 309 +
  • +
  • + Animation 310 +
  • +
  • + Animation 311 +
  • +
  • + Animation 312 +
  • +
  • + Animation 313 +
  • +
  • + Animation 314 +
  • +
  • + Animation 315 +
  • +
  • + Animation 316 +
  • +
  • + Animation 317 +
  • +
  • + Animation 318 +
  • +
  • + Animation 319 +
  • +
  • + Animation 320 +
  • +
  • + Animation 321 +
  • +
  • + Animation 322 +
  • +
  • + Animation 323 +
  • +
  • + Animation 324 +
  • +
  • + Animation 325 +
  • +
  • + Animation 326 +
  • +
  • + Animation 327 +
  • +
  • + Animation 328 +
  • +
  • + Animation 329 +
  • +
  • + Animation 330 +
  • +
  • + Animation 331 +
  • +
  • + Animation 332 +
  • +
  • + Animation 333 +
  • +
  • + Animation 334 +
  • +
  • + Animation 335 +
  • +
  • + Animation 336 +
  • +
  • + Animation 337 +
  • +
  • + Animation 338 +
  • +
  • + Animation 339 +
  • +
  • + Animation 340 +
  • +
  • + Animation 341 +
  • +
  • + Animation 342 +
  • +
  • + Animation 343 +
  • +
  • + Animation 344 +
  • +
  • + Animation 345 +
  • +
  • + Animation 346 +
  • +
  • + Animation 347 +
  • +
  • + Animation 348 +
  • +
  • + Animation 349 +
  • +
  • + Animation 350 +
  • +
  • + Animation 351 +
  • +
  • + Animation 352 +
  • +
  • + Animation 353 +
  • +
  • + Animation 354 +
  • +
  • + Animation 355 +
  • +
  • + Animation 356 +
  • +
  • + Animation 357 +
  • +
  • + Animation 358 +
  • +
  • + Animation 359 +
  • +
  • + Animation 360 +
  • +
  • + Animation 361 +
  • +
  • + Animation 362 +
  • +
  • + Animation 363 +
  • +
  • + Animation 364 +
  • +
  • + Animation 365 +
  • +
  • + Animation 366 +
  • +
  • + Animation 367 +
  • +
  • + Animation 368 +
  • +
  • + Animation 369 +
  • +
  • + Animation 370 +
  • +
  • + Animation 371 +
  • +
  • + Animation 372 +
  • +
  • + Animation 373 +
  • +
  • + Animation 374 +
  • +
  • + Animation 375 +
  • +
  • + Animation 376 +
  • +
  • + Animation 377 +
  • +
  • + Animation 378 +
  • +
  • + Animation 379 +
  • +
  • + Animation 380 +
  • +
  • + Animation 381 +
  • +
  • + Animation 382 +
  • +
  • + Animation 383 +
  • +
  • + Animation 384 +
  • +
  • + Animation 385 +
  • +
  • + Animation 386 +
  • +
  • + Animation 387 +
  • +
  • + Animation 388 +
  • +
  • + Animation 389 +
  • +
  • + Animation 390 +
  • +
  • + Animation 391 +
  • +
  • + Animation 392 +
  • +
  • + Animation 393 +
  • +
  • + Animation 394 +
  • +
  • + Animation 395 +
  • +
  • + Animation 396 +
  • +
  • + Animation 397 +
  • +
  • + Animation 398 +
  • +
  • + Animation 399 +
  • +
  • + Animation 400 +
  • +
  • + Animation 401 +
  • +
  • + Animation 402 +
  • +
  • + Animation 403 +
  • +
  • + Animation 404 +
  • +
  • + Animation 405 +
  • +
  • + Animation 406 +
  • +
  • + Animation 407 +
  • +
  • + Animation 408 +
  • +
  • + Animation 409 +
  • +
  • + Animation 410 +
  • +
  • + Animation 411 +
  • +
  • + Animation 412 +
  • +
  • + Animation 413 +
  • +
  • + Animation 414 +
  • +
  • + Animation 415 +
  • +
  • + Animation 416 +
  • +
  • + Animation 417 +
  • +
  • + Animation 418 +
  • +
  • + Animation 419 +
  • +
  • + Animation 420 +
  • +
  • + Animation 421 +
  • +
  • + Animation 422 +
  • +
  • + Animation 423 +
  • +
  • + Animation 424 +
  • +
  • + Animation 425 +
  • +
  • + Animation 426 +
  • +
  • + Animation 427 +
  • +
  • + Animation 428 +
  • +
  • + Animation 429 +
  • +
  • + Animation 430 +
  • +
  • + Animation 431 +
  • +
  • + Animation 432 +
  • +
  • + Animation 433 +
  • +
  • + Animation 434 +
  • +
  • + Animation 435 +
  • +
  • + Animation 436 +
  • +
  • + Animation 437 +
  • +
  • + Animation 438 +
  • +
  • + Animation 439 +
  • +
  • + Animation 440 +
  • +
  • + Animation 441 +
  • +
  • + Animation 442 +
  • +
  • + Animation 443 +
  • +
  • + Animation 444 +
  • +
  • + Animation 445 +
  • +
  • + Animation 446 +
  • +
  • + Animation 447 +
  • +
  • + Animation 448 +
  • +
  • + Animation 449 +
  • +
  • + Animation 450 +
  • +
  • + Animation 451 +
  • +
  • + Animation 452 +
  • +
  • + Animation 453 +
  • +
  • + Animation 454 +
  • +
  • + Animation 455 +
  • +
  • + Animation 456 +
  • +
  • + Animation 457 +
  • +
  • + Animation 458 +
  • +
  • + Animation 459 +
  • +
  • + Animation 460 +
  • +
  • + Animation 461 +
  • +
  • + Animation 462 +
  • +
  • + Animation 463 +
  • +
  • + Animation 464 +
  • +
  • + Animation 465 +
  • +
  • + Animation 466 +
  • +
  • + Animation 467 +
  • +
  • + Animation 468 +
  • +
  • + Animation 469 +
  • +
  • + Animation 470 +
  • +
  • + Animation 471 +
  • +
  • + Animation 472 +
  • +
  • + Animation 473 +
  • +
  • + Animation 474 +
  • +
  • + Animation 475 +
  • +
  • + Animation 476 +
  • +
  • + Animation 477 +
  • +
  • + Animation 478 +
  • +
  • + Animation 479 +
  • +
  • + Animation 480 +
  • +
  • + Animation 481 +
  • +
  • + Animation 482 +
  • +
  • + Animation 483 +
  • +
  • + Animation 484 +
  • +
  • + Animation 485 +
  • +
  • + Animation 486 +
  • +
  • + Animation 487 +
  • +
  • + Animation 488 +
  • +
  • + Animation 489 +
  • +
  • + Animation 490 +
  • +
  • + Animation 491 +
  • +
  • + Animation 492 +
  • +
  • + Animation 493 +
  • +
  • + Animation 494 +
  • +
  • + Animation 495 +
  • +
  • + Animation 496 +
  • +
  • + Animation 497 +
  • +
  • + Animation 498 +
  • +
  • + Animation 499 +
  • +
  • + Animation 500 +
  • +
  • + Animation 501 +
  • +
  • + Animation 502 +
  • +
  • + Animation 503 +
  • +
  • + Animation 504 +
  • +
  • + Animation 505 +
  • +
  • + Animation 506 +
  • +
  • + Animation 507 +
  • +
  • + Animation 508 +
  • +
  • + Animation 509 +
  • +
  • + Animation 510 +
  • +
  • + Animation 511 +
  • +
  • + Animation 512 +
  • +
  • + Animation 513 +
  • +
  • + Animation 514 +
  • +
  • + Animation 515 +
  • +
  • + Animation 516 +
  • +
  • + Animation 517 +
  • +
  • + Animation 518 +
  • +
  • + Animation 519 +
  • +
  • + Animation 520 +
  • +
  • + Animation 521 +
  • +
  • + Animation 522 +
  • +
  • + Animation 523 +
  • +
  • + Animation 524 +
  • +
  • + Animation 525 +
  • +
  • + Animation 526 +
  • +
  • + Animation 527 +
  • +
  • + Animation 528 +
  • +
  • + Animation 529 +
  • +
  • + Animation 530 +
  • +
  • + Animation 531 +
  • +
  • + Animation 532 +
  • +
  • + Animation 533 +
  • +
  • + Animation 534 +
  • +
  • + Animation 535 +
  • +
  • + Animation 536 +
  • +
  • + Animation 537 +
  • +
  • + Animation 538 +
  • +
  • + Animation 539 +
  • +
  • + Animation 540 +
  • +
  • + Animation 541 +
  • +
  • + Animation 542 +
  • +
  • + Animation 543 +
  • +
  • + Animation 544 +
  • +
  • + Animation 545 +
  • +
  • + Animation 546 +
  • +
  • + Animation 547 +
  • +
  • + Animation 548 +
  • +
  • + Animation 549 +
  • +
  • + Animation 550 +
  • +
  • + Animation 551 +
  • +
  • + Animation 552 +
  • +
  • + Animation 553 +
  • +
  • + Animation 554 +
  • +
  • + Animation 555 +
  • +
  • + Animation 556 +
  • +
  • + Animation 557 +
  • +
  • + Animation 558 +
  • +
  • + Animation 559 +
  • +
  • + Animation 560 +
  • +
  • + Animation 561 +
  • +
  • + Animation 562 +
  • +
  • + Animation 563 +
  • +
  • + Animation 564 +
  • +
  • + Animation 565 +
  • +
  • + Animation 566 +
  • +
  • + Animation 567 +
  • +
  • + Animation 568 +
  • +
  • + Animation 569 +
  • +
  • + Animation 570 +
  • +
  • + Animation 571 +
  • +
  • + Animation 572 +
  • +
+
+
+`; diff --git a/src/viewer/gui/model_3d/Model3DView.ts b/src/viewer/gui/model_3d/Model3DView.ts deleted file mode 100644 index f98e28df..00000000 --- a/src/viewer/gui/model_3d/Model3DView.ts +++ /dev/null @@ -1,90 +0,0 @@ -import "./Model3DView.css"; -import { RendererWidget } from "../../../core/gui/RendererWidget"; -import { Model3DRenderer } from "../../rendering/Model3DRenderer"; -import { Model3DToolBarView } from "./Model3DToolBarView"; -import { Model3DSelectListView } from "./Model3DSelectListView"; -import { CharacterClassModel } from "../../model/CharacterClassModel"; -import { CharacterClassAnimationModel } from "../../model/CharacterClassAnimationModel"; -import { Model3DStore } from "../../stores/Model3DStore"; -import { DisposableThreeRenderer } from "../../../core/rendering/Renderer"; -import { div } from "../../../core/gui/dom"; -import { ResizableView } from "../../../core/gui/ResizableView"; -import { GuiStore } from "../../../core/stores/GuiStore"; - -const MODEL_LIST_WIDTH = 100; -const ANIMATION_LIST_WIDTH = 140; - -export class Model3DView extends ResizableView { - readonly element = div({ className: "viewer_Model3DView" }); - - private tool_bar_view: Model3DToolBarView; - private model_list_view: Model3DSelectListView; - private animation_list_view: Model3DSelectListView; - private renderer_view: RendererWidget; - - constructor( - private readonly gui_store: GuiStore, - model_3d_store: Model3DStore, - three_renderer: DisposableThreeRenderer, - ) { - super(); - - this.tool_bar_view = this.add(new Model3DToolBarView(model_3d_store)); - this.model_list_view = this.add( - new Model3DSelectListView( - model_3d_store.models, - model_3d_store.current_model, - model_3d_store.set_current_model, - ), - ); - this.animation_list_view = this.add( - new Model3DSelectListView( - model_3d_store.animations, - model_3d_store.current_animation, - model_3d_store.set_current_animation, - ), - ); - this.renderer_view = this.add( - new RendererWidget(new Model3DRenderer(three_renderer, model_3d_store)), - ); - - this.animation_list_view.borders = true; - - this.element.append( - this.tool_bar_view.element, - div( - { className: "viewer_Model3DView_container" }, - this.model_list_view.element, - this.animation_list_view.element, - this.renderer_view.element, - ), - ); - - this.finalize_construction(); - } - - activate(): void { - this.renderer_view.start_rendering(); - super.activate(); - } - - deactivate(): void { - super.deactivate(); - this.renderer_view.stop_rendering(); - } - - resize(width: number, height: number): this { - super.resize(width, height); - - const container_height = Math.max(0, height - this.tool_bar_view.height); - - this.model_list_view.resize(MODEL_LIST_WIDTH, container_height); - this.animation_list_view.resize(ANIMATION_LIST_WIDTH, container_height); - this.renderer_view.resize( - Math.max(0, width - MODEL_LIST_WIDTH - ANIMATION_LIST_WIDTH), - container_height, - ); - - return this; - } -} diff --git a/src/viewer/index.ts b/src/viewer/index.ts index 2afbf33a..5d7b6188 100644 --- a/src/viewer/index.ts +++ b/src/viewer/index.ts @@ -4,9 +4,18 @@ import { HttpClient } from "../core/HttpClient"; import { DisposableThreeRenderer } from "../core/rendering/Renderer"; import { Disposable } from "../core/observable/Disposable"; import { Disposer } from "../core/observable/Disposer"; +import { TextureRenderer } from "./rendering/TextureRenderer"; +import { ModelRenderer } from "./rendering/ModelRenderer"; +import { Random } from "../core/Random"; +import { ModelToolBarView } from "./gui/model/ModelToolBarView"; +import { ModelStore } from "./stores/ModelStore"; +import { ModelToolBarController } from "./controllers/model/ModelToolBarController"; +import { CharacterClassOptionsView } from "./gui/model/CharacterClassOptionsView"; +import { CharacterClassOptionsController } from "./controllers/model/CharacterClassOptionsController"; export function initialize_viewer( http_client: HttpClient, + random: Random, gui_store: GuiStore, create_three_renderer: () => DisposableThreeRenderer, ): { view: ViewerView } & Disposable { @@ -14,36 +23,36 @@ export function initialize_viewer( const view = new ViewerView( gui_store, + async () => { - const { Model3DStore } = await import("./stores/Model3DStore"); - const { Model3DView } = await import("./gui/model_3d/Model3DView"); + const { ModelController } = await import("./controllers/model/ModelController"); + const { ModelView } = await import("./gui/model/ModelView"); const { CharacterClassAssetLoader } = await import( "./loading/CharacterClassAssetLoader" ); const asset_loader = disposer.add(new CharacterClassAssetLoader(http_client)); - const store = new Model3DStore(asset_loader); + const store = disposer.add(new ModelStore(asset_loader, random)); + const model_controller = new ModelController(store); + const model_tool_bar_controller = new ModelToolBarController(store); + const character_class_options_controller = new CharacterClassOptionsController(store); - if (disposer.disposed) { - store.dispose(); - } else { - disposer.add(store); - } - - return new Model3DView(gui_store, store, create_three_renderer()); + return new ModelView( + model_controller, + new ModelToolBarView(model_tool_bar_controller), + new CharacterClassOptionsView(character_class_options_controller), + new ModelRenderer(store, create_three_renderer()), + ); }, async () => { - const { TextureStore } = await import("./stores/TextureStore"); + const { TextureController } = await import("./controllers/TextureController"); const { TextureView } = await import("./gui/TextureView"); - const store = new TextureStore(); + const controller = disposer.add(new TextureController()); - if (disposer.disposed) { - store.dispose(); - } else { - disposer.add(store); - } - - return new TextureView(store, create_three_renderer()); + return new TextureView( + controller, + new TextureRenderer(controller, create_three_renderer()), + ); }, ); diff --git a/src/viewer/loading/CharacterClassAssetLoader.ts b/src/viewer/loading/CharacterClassAssetLoader.ts index 2accd6f3..5c7d6359 100644 --- a/src/viewer/loading/CharacterClassAssetLoader.ts +++ b/src/viewer/loading/CharacterClassAssetLoader.ts @@ -327,7 +327,7 @@ function texture_ids( case FOMARL: { const body_idx = body * 16; return { - section_id: section_id + 310, + section_id: section_id + 326, body: [body_idx, body_idx + 2, body_idx + 1, 322 /*hands*/], head: [288], hair: [undefined, undefined, 308], diff --git a/src/viewer/rendering/Model3DRenderer.ts b/src/viewer/rendering/ModelRenderer.ts similarity index 64% rename from src/viewer/rendering/Model3DRenderer.ts rename to src/viewer/rendering/ModelRenderer.ts index 2418b959..cd8ef734 100644 --- a/src/viewer/rendering/Model3DRenderer.ts +++ b/src/viewer/rendering/ModelRenderer.ts @@ -1,10 +1,8 @@ import { - AdditiveBlending, AnimationClip, AnimationMixer, Clock, DoubleSide, - Mesh, MeshBasicMaterial, MeshLambertMaterial, Object3D, @@ -25,10 +23,10 @@ import { import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer"; import { Disposer } from "../../core/observable/Disposer"; import { ChangeEvent } from "../../core/observable/Observable"; -import { Model3DStore } from "../stores/Model3DStore"; import { LogManager } from "../../core/Logger"; +import { ModelStore } from "../stores/ModelStore"; -const logger = LogManager.get("viewer/rendering/Model3DRenderer"); +const logger = LogManager.get("viewer/rendering/ModelRenderer"); const DEFAULT_MATERIAL = new MeshLambertMaterial({ color: 0xffffff, @@ -40,7 +38,7 @@ const DEFAULT_SKINNED_MATERIAL = new MeshLambertMaterial({ side: DoubleSide, }); -export class Model3DRenderer extends Renderer implements Disposable { +export class ModelRenderer extends Renderer implements Disposable { private readonly disposer = new Disposer(); private readonly clock = new Clock(); private mesh?: Object3D; @@ -53,23 +51,21 @@ export class Model3DRenderer extends Renderer implements Disposable { readonly camera = new PerspectiveCamera(75, 1, 1, 200); - constructor( - three_renderer: DisposableThreeRenderer, - private readonly model_3d_store: Model3DStore, - ) { + constructor(private readonly store: ModelStore, three_renderer: DisposableThreeRenderer) { super(three_renderer); this.disposer.add_all( - model_3d_store.current_nj_data.observe(this.nj_data_or_xvm_changed), - model_3d_store.current_textures.observe(this.nj_data_or_xvm_changed), - model_3d_store.current_nj_motion.observe(this.nj_motion_changed), - model_3d_store.show_skeleton.observe(this.show_skeleton_changed), - model_3d_store.animation_playing.observe(this.animation_playing_changed), - model_3d_store.animation_frame_rate.observe(this.animation_frame_rate_changed), - model_3d_store.animation_frame.observe(this.animation_frame_changed), + store.current_nj_object.observe(this.nj_object_or_xvm_changed), + store.current_textures.observe(this.nj_object_or_xvm_changed), + store.current_nj_motion.observe(this.nj_motion_changed), + store.show_skeleton.observe(this.show_skeleton_changed), + store.animation_playing.observe(this.animation_playing_changed), + store.animation_frame_rate.observe(this.animation_frame_rate_changed), + store.animation_frame.observe(this.animation_frame_changed), ); this.init_camera_controls(); + this.reset_camera(new Vector3(0, 10, 20), new Vector3(0, 0, 0)); } set_size(width: number, height: number): void { @@ -97,7 +93,7 @@ export class Model3DRenderer extends Renderer implements Disposable { } } - private nj_data_or_xvm_changed = (): void => { + private nj_object_or_xvm_changed = (): void => { if (this.mesh) { this.scene.remove(this.mesh); this.mesh = undefined; @@ -105,20 +101,22 @@ export class Model3DRenderer extends Renderer implements Disposable { this.skeleton_helper = undefined; } - if (this.animation) { - this.animation.mixer.stopAllAction(); - if (this.mesh) this.animation.mixer.uncacheRoot(this.mesh); - this.animation = undefined; - } + const nj_object = this.store.current_nj_object.val; - const nj_data = this.model_3d_store.current_nj_data.val; + if (nj_object) { + // Stop animation and store animation time. + let animation_time: number | undefined; - if (nj_data) { - const { nj_object, has_skeleton } = nj_data; + if (this.animation) { + const mixer = this.animation.mixer; + animation_time = mixer.existingAction(this.animation.clip)?.time; + mixer.stopAllAction(); + mixer.uncacheAction(this.animation.clip); + this.animation = undefined; + } - let mesh: Mesh; - - const textures = this.model_3d_store.current_textures.val.map(tex => { + // Convert textures and geometry. + const textures = this.store.current_textures.val.map(tex => { if (tex) { try { return xvr_texture_to_texture(tex); @@ -130,6 +128,9 @@ export class Model3DRenderer extends Renderer implements Disposable { return undefined; }); + const geometry = ninja_object_to_buffer_geometry(nj_object); + const has_skeleton = geometry.getAttribute("skinIndex") != undefined; + const materials = textures.map(tex => tex ? new MeshBasicMaterial({ @@ -145,64 +146,66 @@ export class Model3DRenderer extends Renderer implements Disposable { }), ); - if (has_skeleton) { - mesh = create_skinned_mesh( - ninja_object_to_buffer_geometry(nj_object), - materials, - DEFAULT_SKINNED_MATERIAL, - ); - } else { - mesh = create_mesh( - ninja_object_to_buffer_geometry(nj_object), - materials, - DEFAULT_MATERIAL, - ); - } + this.mesh = has_skeleton + ? create_skinned_mesh(geometry, materials, DEFAULT_SKINNED_MATERIAL) + : create_mesh(geometry, materials, DEFAULT_MATERIAL); // Make sure we rotate around the center of the model instead of its origin. - const bb = mesh.geometry.boundingBox; + const bb = geometry.boundingBox; const height = bb.max.y - bb.min.y; - mesh.translateY(-height / 2 - bb.min.y); + this.mesh.translateY(-height / 2 - bb.min.y); - this.mesh = mesh; - this.scene.add(mesh); + this.scene.add(this.mesh); - this.skeleton_helper = new SkeletonHelper(mesh); - this.skeleton_helper.visible = this.model_3d_store.show_skeleton.val; + // Add skeleton. + this.skeleton_helper = new SkeletonHelper(this.mesh); + this.skeleton_helper.visible = this.store.show_skeleton.val; (this.skeleton_helper.material as any).linewidth = 3; this.scene.add(this.skeleton_helper); - this.reset_camera(new Vector3(0, 10, 20), new Vector3(0, 0, 0)); + // Create a new animation mixer and clip. + const nj_motion = this.store.current_nj_motion.val; + + if (nj_motion) { + const mixer = new AnimationMixer(this.mesh); + mixer.timeScale = this.store.animation_frame_rate.val / PSO_FRAME_RATE; + + const clip = create_animation_clip(nj_object, nj_motion); + + this.animation = { mixer, clip }; + + const action = mixer.clipAction(clip, this.mesh); + action.time = animation_time ?? 0; + action.play(); + } } this.schedule_render(); }; private nj_motion_changed = ({ value: nj_motion }: ChangeEvent): void => { - let mixer!: AnimationMixer; + let mixer: AnimationMixer | undefined; if (this.animation) { this.animation.mixer.stopAllAction(); + this.animation.mixer.uncacheAction(this.animation.clip); mixer = this.animation.mixer; } - const nj_data = this.model_3d_store.current_nj_data.val; + const nj_object = this.store.current_nj_object.val; - if (!this.mesh || !(this.mesh instanceof SkinnedMesh) || !nj_motion || !nj_data) return; + if (!this.mesh || !(this.mesh instanceof SkinnedMesh) || !nj_motion || !nj_object) return; - if (!this.animation) { + if (!mixer) { mixer = new AnimationMixer(this.mesh); } - const clip = create_animation_clip(nj_data.nj_object, nj_motion); + const clip = create_animation_clip(nj_object, nj_motion); - this.animation = { - mixer, - clip, - }; + this.animation = { mixer, clip }; this.clock.start(); - this.animation.mixer.clipAction(this.animation.clip).play(); + mixer.clipAction(clip).play(); this.schedule_render(); }; @@ -233,7 +236,7 @@ export class Model3DRenderer extends Renderer implements Disposable { }; private animation_frame_changed = ({ value: frame }: ChangeEvent): void => { - const nj_motion = this.model_3d_store.current_nj_motion.val; + const nj_motion = this.store.current_nj_motion.val; if (this.animation && nj_motion) { const frame_count = nj_motion.frame_count; @@ -255,7 +258,7 @@ export class Model3DRenderer extends Renderer implements Disposable { if (!action.paused) { this.update_animation_time = false; - this.model_3d_store.set_animation_frame(action.time * PSO_FRAME_RATE + 1); + this.store.set_animation_frame(action.time * PSO_FRAME_RATE + 1); this.update_animation_time = true; } } diff --git a/src/viewer/rendering/TextureRenderer.ts b/src/viewer/rendering/TextureRenderer.ts index 42d534fb..ad79cc5d 100644 --- a/src/viewer/rendering/TextureRenderer.ts +++ b/src/viewer/rendering/TextureRenderer.ts @@ -10,10 +10,10 @@ import { import { Disposable } from "../../core/observable/Disposable"; import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer"; import { Disposer } from "../../core/observable/Disposer"; -import { TextureStore } from "../stores/TextureStore"; import { XvrTexture } from "../../core/data_formats/parsing/ninja/texture"; import { xvr_texture_to_texture } from "../../core/rendering/conversion/ninja_textures"; import { LogManager } from "../../core/Logger"; +import { TextureController } from "../controllers/TextureController"; const logger = LogManager.get("viewer/rendering/TextureRenderer"); @@ -23,11 +23,11 @@ export class TextureRenderer extends Renderer implements Disposable { readonly camera = new OrthographicCamera(-400, 400, 300, -300, 1, 10); - constructor(three_renderer: DisposableThreeRenderer, texture_store: TextureStore) { + constructor(ctrl: TextureController, three_renderer: DisposableThreeRenderer) { super(three_renderer); this.disposer.add_all( - texture_store.textures.observe(({ value: textures }) => { + ctrl.textures.observe(({ value: textures }) => { this.scene.remove(...this.quad_meshes); this.render_textures(textures); diff --git a/src/viewer/stores/Model3DStore.ts b/src/viewer/stores/Model3DStore.ts deleted file mode 100644 index 912d1852..00000000 --- a/src/viewer/stores/Model3DStore.ts +++ /dev/null @@ -1,237 +0,0 @@ -import { ArrayBufferCursor } from "../../core/data_formats/cursor/ArrayBufferCursor"; -import { Endianness } from "../../core/data_formats/Endianness"; -import { NjMotion, parse_njm } from "../../core/data_formats/parsing/ninja/motion"; -import { NjObject, parse_nj, parse_xj } from "../../core/data_formats/parsing/ninja"; -import { - CharacterClassModel, - FOMAR, - FOMARL, - FONEWEARL, - FONEWM, - HUCASEAL, - HUCAST, - HUMAR, - HUNEWEARL, - RACASEAL, - RACAST, - RAMAR, - RAMARL, -} from "../model/CharacterClassModel"; -import { CharacterClassAnimationModel } from "../model/CharacterClassAnimationModel"; -import { WritableProperty } from "../../core/observable/property/WritableProperty"; -import { read_file } from "../../core/read_file"; -import { list_property, property } from "../../core/observable"; -import { Property } from "../../core/observable/property/Property"; -import { PSO_FRAME_RATE } from "../../core/rendering/conversion/ninja_animation"; -import { parse_xvm, XvrTexture } from "../../core/data_formats/parsing/ninja/texture"; -import { CharacterClassAssetLoader } from "../loading/CharacterClassAssetLoader"; -import { Store } from "../../core/stores/Store"; -import { LogManager } from "../../core/Logger"; -import { ListProperty } from "../../core/observable/property/list/ListProperty"; -import { parse_afs } from "../../core/data_formats/parsing/afs"; -import { random_array_element, random_integer } from "../../core/util"; -import { SectionIds } from "../../core/model"; - -const logger = LogManager.get("viewer/stores/ModelStore"); - -export type NjData = { - nj_object: NjObject; - bone_count: number; - has_skeleton: boolean; -}; - -export class Model3DStore extends Store { - private readonly _current_model: WritableProperty = property( - undefined, - ); - private readonly _current_nj_data = property(undefined); - private readonly _current_textures = list_property(); - private readonly _show_skeleton: WritableProperty = property(false); - private readonly _current_animation: WritableProperty< - CharacterClassAnimationModel | undefined - > = property(undefined); - private readonly _current_nj_motion = property(undefined); - private readonly _animation_playing: WritableProperty = property(true); - private readonly _animation_frame_rate: WritableProperty = property(PSO_FRAME_RATE); - private readonly _animation_frame: WritableProperty = property(0); - - readonly models: readonly CharacterClassModel[] = [ - HUMAR, - HUNEWEARL, - HUCAST, - HUCASEAL, - RAMAR, - RAMARL, - RACAST, - RACASEAL, - FOMAR, - FOMARL, - FONEWM, - FONEWEARL, - ]; - - readonly animations: readonly CharacterClassAnimationModel[] = new Array(572) - .fill(undefined) - .map((_, i) => new CharacterClassAnimationModel(i, `Animation ${i + 1}`)); - - readonly current_model: Property = this._current_model; - readonly current_nj_data: Property = this._current_nj_data; - readonly current_textures: ListProperty = this._current_textures; - readonly show_skeleton: Property = this._show_skeleton; - readonly current_animation: Property = this - ._current_animation; - readonly current_nj_motion: Property = this._current_nj_motion; - readonly animation_playing: Property = this._animation_playing; - readonly animation_frame_rate: Property = this._animation_frame_rate; - readonly animation_frame: Property = this._animation_frame; - readonly animation_frame_count: Property = this.current_nj_motion.map(njm => - njm ? njm.frame_count : 0, - ); - - constructor(private readonly asset_loader: CharacterClassAssetLoader) { - super(); - - this.disposables( - this.current_model.observe(({ value }) => this.load_model(value)), - this.current_animation.observe(({ value }) => this.load_animation(value)), - ); - - this.set_current_model(random_array_element(this.models)); - } - - set_current_model = (current_model?: CharacterClassModel): void => { - this._current_model.val = current_model; - }; - - set_show_skeleton = (show_skeleton: boolean): void => { - this._show_skeleton.val = show_skeleton; - }; - - set_current_animation = (animation?: CharacterClassAnimationModel): void => { - this._current_animation.val = animation; - }; - - set_animation_playing = (playing: boolean): void => { - this._animation_playing.val = playing; - }; - - set_animation_frame_rate = (frame_rate: number): void => { - this._animation_frame_rate.val = frame_rate; - }; - - set_animation_frame = (frame: number): void => { - this._animation_frame.val = frame; - }; - - // TODO: notify user of problems. - load_file = async (file: File): Promise => { - try { - const buffer = await read_file(file); - const cursor = new ArrayBufferCursor(buffer, Endianness.Little); - - if (file.name.endsWith(".nj")) { - this.set_current_model(undefined); - this._current_textures.clear(); - - const nj_object = parse_nj(cursor)[0]; - - this.set_current_nj_data({ - nj_object, - bone_count: nj_object.bone_count(), - has_skeleton: true, - }); - } else if (file.name.endsWith(".xj")) { - this.set_current_model(undefined); - this._current_textures.clear(); - - const nj_object = parse_xj(cursor)[0]; - - this.set_current_nj_data({ - nj_object, - bone_count: 0, - has_skeleton: false, - }); - } else if (file.name.endsWith(".njm")) { - this.set_current_animation(undefined); - this._current_nj_motion.val = undefined; - - const nj_data = this.current_nj_data.val; - - if (nj_data) { - this.set_animation_playing(true); - this._current_nj_motion.val = parse_njm(cursor, nj_data.bone_count); - } - } else if (file.name.endsWith(".xvm")) { - this._current_textures.val = parse_xvm(cursor).textures; - } else if (file.name.endsWith(".afs")) { - const files = parse_afs(cursor); - const textures: XvrTexture[] = []; - - for (const file of files) { - textures.push( - ...parse_xvm(new ArrayBufferCursor(file, Endianness.Little)).textures, - ); - } - - this._current_textures.val = textures; - } else { - logger.error(`Unknown file extension in filename "${file.name}".`); - } - } catch (e) { - logger.error("Couldn't read file.", e); - } - }; - - private load_model = async (model?: CharacterClassModel): Promise => { - this.set_current_animation(undefined); - - if (model) { - try { - this.set_current_nj_data(undefined); - - const nj_object = await this.asset_loader.load_geometry(model); - - this._current_textures.val = await this.asset_loader.load_textures( - model, - random_array_element(SectionIds), - random_integer(0, model.body_style_count), - ); - - this.set_current_nj_data({ - nj_object, - // Ignore the bones from the head parts. - bone_count: model ? 64 : nj_object.bone_count(), - has_skeleton: true, - }); - } catch (e) { - logger.error(`Couldn't load model for ${model.name}.`); - this._current_nj_data.val = undefined; - } - } else { - this._current_nj_data.val = undefined; - } - }; - - private set_current_nj_data(nj_data?: NjData): void { - this._current_nj_data.val = nj_data; - } - - private load_animation = async (animation?: CharacterClassAnimationModel): Promise => { - const nj_data = this.current_nj_data.val; - - if (nj_data && animation) { - try { - this._current_nj_motion.val = await this.asset_loader.load_animation( - animation.id, - nj_data.bone_count, - ); - this.set_animation_playing(true); - } catch (e) { - logger.error(`Couldn't load animation "${animation.name}".`); - this._current_nj_motion.val = undefined; - } - } else { - this._current_nj_motion.val = undefined; - } - }; -} diff --git a/src/viewer/stores/ModelStore.ts b/src/viewer/stores/ModelStore.ts new file mode 100644 index 00000000..167134ed --- /dev/null +++ b/src/viewer/stores/ModelStore.ts @@ -0,0 +1,226 @@ +import { Store } from "../../core/stores/Store"; +import { WritableProperty } from "../../core/observable/property/WritableProperty"; +import { + CharacterClassModel, + FOMAR, + FOMARL, + FONEWEARL, + FONEWM, + HUCASEAL, + HUCAST, + HUMAR, + HUNEWEARL, + RACASEAL, + RACAST, + RAMAR, + RAMARL, +} from "../model/CharacterClassModel"; +import { list_property, property } from "../../core/observable"; +import { XvrTexture } from "../../core/data_formats/parsing/ninja/texture"; +import { CharacterClassAnimationModel } from "../model/CharacterClassAnimationModel"; +import { NjMotion } from "../../core/data_formats/parsing/ninja/motion"; +import { PSO_FRAME_RATE } from "../../core/rendering/conversion/ninja_animation"; +import { Property } from "../../core/observable/property/Property"; +import { ListProperty } from "../../core/observable/property/list/ListProperty"; +import { CharacterClassAssetLoader } from "../loading/CharacterClassAssetLoader"; +import { Random } from "../../core/Random"; +import { SectionId, SectionIds } from "../../core/model"; +import { LogManager } from "../../core/Logger"; +import { NjObject } from "../../core/data_formats/parsing/ninja"; + +const logger = LogManager.get("viewer/stores/ModelStore"); + +export type NjData = { + readonly nj_object: NjObject; + readonly bone_count: number; + readonly has_skeleton: boolean; +}; + +export class ModelStore extends Store { + // Character classes and their animations. + private readonly _current_character_class: WritableProperty< + CharacterClassModel | undefined + > = property(undefined); + private readonly _current_section_id: WritableProperty; + private readonly _current_body: WritableProperty = property(0); + private readonly _current_animation: WritableProperty< + CharacterClassAnimationModel | undefined + > = property(undefined); + + // Geometry, textures and animations. + private readonly _current_nj_object = property(undefined); + private readonly _current_textures = list_property(); + private readonly _current_nj_motion = property(undefined); + + // User settings. + private readonly _show_skeleton: WritableProperty = property(false); + private readonly _animation_playing: WritableProperty = property(true); + private readonly _animation_frame_rate: WritableProperty = property(PSO_FRAME_RATE); + private readonly _animation_frame: WritableProperty = property(0); + + // Character classes and their animations. + readonly character_classes: readonly CharacterClassModel[] = [ + HUMAR, + HUNEWEARL, + HUCAST, + HUCASEAL, + RAMAR, + RAMARL, + RACAST, + RACASEAL, + FOMAR, + FOMARL, + FONEWM, + FONEWEARL, + ]; + readonly current_character_class: Property = this + ._current_character_class; + readonly current_section_id: Property; + readonly current_body: Property = this._current_body; + readonly animations: readonly CharacterClassAnimationModel[] = new Array(572) + .fill(undefined) + .map((_, i) => new CharacterClassAnimationModel(i, `Animation ${i + 1}`)); + readonly current_animation: Property = this + ._current_animation; + + // Geometry, textures and animations. + readonly current_nj_object: Property = this._current_nj_object; + readonly current_textures: ListProperty = this._current_textures; + readonly current_nj_motion: Property = this._current_nj_motion; + readonly animation_frame_count: Property = this.current_nj_motion.map(njm => + njm ? njm.frame_count : 0, + ); + + // User settings. + readonly show_skeleton: Property = this._show_skeleton; + readonly animation_playing: Property = this._animation_playing; + readonly animation_frame_rate: Property = this._animation_frame_rate; + readonly animation_frame: Property = this._animation_frame; + + constructor( + private readonly asset_loader: CharacterClassAssetLoader, + private readonly random: Random, + ) { + super(); + + this._current_section_id = property(random.sample_array(SectionIds)); + this.current_section_id = this._current_section_id; + + this.disposables( + this.current_character_class.observe(this.load_character_class_model), + this.current_section_id.observe(this.load_character_class_model), + this.current_body.observe(this.load_character_class_model), + this.current_animation.observe(this.load_animation), + ); + + const character_class = random.sample_array(this.character_classes); + this.set_current_character_class(character_class); + } + + set_current_character_class = (character_class?: CharacterClassModel): void => { + if (this._current_character_class.val !== character_class) { + this._current_character_class.val = character_class; + this.set_current_body( + character_class + ? this.random.integer(0, character_class.body_style_count) + : undefined, + ); + + if (this.current_animation.val == undefined) { + this.set_current_nj_motion(undefined); + } + } + }; + + set_current_section_id = (section_id: SectionId): void => { + this._current_section_id.val = section_id; + }; + + set_current_body = (body?: number): void => { + this._current_body.val = body; + }; + + set_current_animation = (animation?: CharacterClassAnimationModel): void => { + if (this._current_animation.val !== animation) { + this._current_animation.val = animation; + } + }; + + set_current_nj_object = (nj_object?: NjObject): void => { + this.set_current_character_class(undefined); + this.set_current_animation(undefined); + this.set_current_textures([]); + this.set_current_nj_motion(undefined); + this._current_nj_object.val = nj_object; + }; + + set_current_textures = (textures: (XvrTexture | undefined)[]): void => { + this._current_textures.val = textures; + }; + + set_current_nj_motion = (nj_motion?: NjMotion): void => { + this.set_current_animation(undefined); + this._current_nj_motion.val = nj_motion; + }; + + set_show_skeleton = (show_skeleton: boolean): void => { + this._show_skeleton.val = show_skeleton; + }; + + set_animation_playing = (playing: boolean): void => { + this._animation_playing.val = playing; + }; + + set_animation_frame_rate = (frame_rate: number): void => { + this._animation_frame_rate.val = frame_rate; + }; + + set_animation_frame = (frame: number): void => { + this._animation_frame.val = frame; + }; + + private load_character_class_model = async (): Promise => { + const character_class = this.current_character_class.val; + if (character_class == undefined) return; + + const body = this.current_body.val; + if (body == undefined) return; + + try { + this._current_nj_object.val = undefined; + + const nj_object = await this.asset_loader.load_geometry(character_class); + + this._current_textures.val = await this.asset_loader.load_textures( + character_class, + this.current_section_id.val, + body, + ); + + this._current_nj_object.val = nj_object; + } catch (e) { + logger.error(`Couldn't load model for ${character_class.name}.`); + this._current_nj_object.val = undefined; + } + }; + + private load_animation = async (): Promise => { + const nj_object = this._current_nj_object.val; + const animation = this.current_animation.val; + + if (nj_object && animation) { + try { + this._current_nj_motion.val = await this.asset_loader.load_animation( + animation.id, + 64, + ); + this.set_animation_playing(true); + } catch (e) { + logger.error(`Couldn't load animation "${animation.name}".`, e); + this._current_nj_motion.val = undefined; + } + } else { + this._current_nj_motion.val = undefined; + } + }; +} diff --git a/test/src/core/rendering/StubThreeRenderer.ts b/test/src/core/rendering/StubThreeRenderer.ts new file mode 100644 index 00000000..7806c5b5 --- /dev/null +++ b/test/src/core/rendering/StubThreeRenderer.ts @@ -0,0 +1,11 @@ +import { DisposableThreeRenderer } from "../../../../src/core/rendering/Renderer"; + +export class StubThreeRenderer implements DisposableThreeRenderer { + domElement: HTMLCanvasElement = document.createElement("canvas"); + + dispose(): void {} // eslint-disable-line + + render(): void {} // eslint-disable-line + + setSize(): void {} // eslint-disable-line +}