mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Its now possible to choose a section ID and body type in the model viewer.
This commit is contained in:
parent
66728f7096
commit
05d5ce6e29
@ -1,8 +1,9 @@
|
|||||||
import { initialize_application } from "./index";
|
import { initialize_application } from "./index";
|
||||||
import { DisposableThreeRenderer } from "../core/rendering/Renderer";
|
|
||||||
import { LogHandler, LogLevel, LogManager } from "../core/Logger";
|
import { LogHandler, LogLevel, LogManager } from "../core/Logger";
|
||||||
import { FileSystemHttpClient } from "../../test/src/core/FileSystemHttpClient";
|
import { FileSystemHttpClient } from "../../test/src/core/FileSystemHttpClient";
|
||||||
import { timeout } from "../../test/src/utils";
|
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"]) {
|
for (const path of [undefined, "/viewer", "/quest_editor", "/hunt_optimizer"]) {
|
||||||
const with_path = path == undefined ? "without specific path" : `with path ${path}`;
|
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(
|
const app = initialize_application(
|
||||||
new FileSystemHttpClient(),
|
new FileSystemHttpClient(),
|
||||||
() => new StubRenderer(),
|
new Random(() => 0.27),
|
||||||
|
() => new StubThreeRenderer(),
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(app).toBeDefined();
|
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
|
|
||||||
}
|
|
||||||
|
@ -8,9 +8,11 @@ import { throttle } from "lodash";
|
|||||||
import { DisposableThreeRenderer } from "../core/rendering/Renderer";
|
import { DisposableThreeRenderer } from "../core/rendering/Renderer";
|
||||||
import { Disposer } from "../core/observable/Disposer";
|
import { Disposer } from "../core/observable/Disposer";
|
||||||
import { disposable_custom_listener, disposable_listener } from "../core/gui/dom";
|
import { disposable_custom_listener, disposable_listener } from "../core/gui/dom";
|
||||||
|
import { Random } from "../core/Random";
|
||||||
|
|
||||||
export function initialize_application(
|
export function initialize_application(
|
||||||
http_client: HttpClient,
|
http_client: HttpClient,
|
||||||
|
random: Random,
|
||||||
create_three_renderer: () => DisposableThreeRenderer,
|
create_three_renderer: () => DisposableThreeRenderer,
|
||||||
): Disposable {
|
): Disposable {
|
||||||
const disposer = new Disposer();
|
const disposer = new Disposer();
|
||||||
@ -43,7 +45,7 @@ export function initialize_application(
|
|||||||
async () => {
|
async () => {
|
||||||
const { initialize_viewer } = await import("../viewer");
|
const { initialize_viewer } = await import("../viewer");
|
||||||
const viewer = disposer.add(
|
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;
|
return viewer.view;
|
||||||
|
18
src/core/Random.ts
Normal file
18
src/core/Random.ts
Normal file
@ -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<T>(array: readonly T[]): T {
|
||||||
|
return array[this.integer(0, array.length)];
|
||||||
|
}
|
||||||
|
}
|
@ -9,7 +9,7 @@ import { Menu } from "./Menu";
|
|||||||
|
|
||||||
export type SelectOptions<T> = LabelledControlOptions & {
|
export type SelectOptions<T> = LabelledControlOptions & {
|
||||||
readonly items: readonly T[] | Property<readonly T[]>;
|
readonly items: readonly T[] | Property<readonly T[]>;
|
||||||
readonly to_label: (element: T) => string;
|
readonly to_label?: (element: T) => string;
|
||||||
readonly selected?: T | Property<T>;
|
readonly selected?: T | Property<T>;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -31,7 +31,7 @@ export class Select<T> extends LabelledControl {
|
|||||||
|
|
||||||
this.preferred_label_position = "left";
|
this.preferred_label_position = "left";
|
||||||
|
|
||||||
this.to_label = options.to_label;
|
this.to_label = options.to_label ?? String;
|
||||||
this.button = this.disposable(
|
this.button = this.disposable(
|
||||||
new Button({
|
new Button({
|
||||||
text: " ",
|
text: " ",
|
||||||
|
@ -34,22 +34,6 @@ export function array_remove<T>(array: T[], ...elements: T[]): number {
|
|||||||
return count;
|
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<T>(array: readonly T[]): T {
|
|
||||||
return array[random_integer(0, array.length)];
|
|
||||||
}
|
|
||||||
|
|
||||||
export function array_buffers_equal(a: ArrayBuffer, b: ArrayBuffer): boolean {
|
export function array_buffers_equal(a: ArrayBuffer, b: ArrayBuffer): boolean {
|
||||||
if (a.byteLength !== b.byteLength) return false;
|
if (a.byteLength !== b.byteLength) return false;
|
||||||
|
|
||||||
|
@ -10,6 +10,7 @@ import { initialize_application } from "./application";
|
|||||||
import { FetchClient } from "./core/HttpClient";
|
import { FetchClient } from "./core/HttpClient";
|
||||||
import { WebGLRenderer } from "three";
|
import { WebGLRenderer } from "three";
|
||||||
import { DisposableThreeRenderer } from "./core/rendering/Renderer";
|
import { DisposableThreeRenderer } from "./core/rendering/Renderer";
|
||||||
|
import { Random } from "./core/Random";
|
||||||
|
|
||||||
function create_three_renderer(): DisposableThreeRenderer {
|
function create_three_renderer(): DisposableThreeRenderer {
|
||||||
const renderer = new WebGLRenderer({ antialias: true, alpha: true });
|
const renderer = new WebGLRenderer({ antialias: true, alpha: true });
|
||||||
@ -17,4 +18,4 @@ function create_three_renderer(): DisposableThreeRenderer {
|
|||||||
return renderer;
|
return renderer;
|
||||||
}
|
}
|
||||||
|
|
||||||
initialize_application(new FetchClient(), create_three_renderer);
|
initialize_application(new FetchClient(), new Random(), create_three_renderer);
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
import { list_property } from "../../core/observable";
|
import { Controller } from "../../core/controllers/Controller";
|
||||||
import { parse_xvm, XvrTexture } from "../../core/data_formats/parsing/ninja/texture";
|
import { filename_extension } from "../../core/util";
|
||||||
import { read_file } from "../../core/read_file";
|
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 { ArrayBufferCursor } from "../../core/data_formats/cursor/ArrayBufferCursor";
|
||||||
import { Endianness } from "../../core/data_formats/Endianness";
|
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 { LogManager } from "../../core/Logger";
|
||||||
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
|
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
|
||||||
|
import { list_property } from "../../core/observable";
|
||||||
import { ListProperty } from "../../core/observable/property/list/ListProperty";
|
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<XvrTexture> = list_property();
|
private readonly _textures: WritableListProperty<XvrTexture> = list_property();
|
||||||
readonly textures: ListProperty<XvrTexture> = this._textures;
|
readonly textures: ListProperty<XvrTexture> = this._textures;
|
||||||
|
|
@ -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<boolean>;
|
||||||
|
readonly current_section_id: Property<SectionId>;
|
||||||
|
readonly current_body_options: Property<readonly number[]>;
|
||||||
|
readonly current_body: Property<number | undefined>;
|
||||||
|
|
||||||
|
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);
|
||||||
|
};
|
||||||
|
}
|
31
src/viewer/controllers/model/ModelController.ts
Normal file
31
src/viewer/controllers/model/ModelController.ts
Normal file
@ -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<CharacterClassModel | undefined>;
|
||||||
|
|
||||||
|
readonly animations: readonly CharacterClassAnimationModel[];
|
||||||
|
readonly current_animation: Property<CharacterClassAnimationModel | undefined>;
|
||||||
|
|
||||||
|
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);
|
||||||
|
};
|
||||||
|
}
|
92
src/viewer/controllers/model/ModelToolBarController.ts
Normal file
92
src/viewer/controllers/model/ModelToolBarController.ts
Normal file
@ -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<boolean>;
|
||||||
|
readonly animation_frame_count: Property<number>;
|
||||||
|
readonly animation_frame_count_label: Property<string>;
|
||||||
|
readonly animation_controls_enabled: Property<boolean>;
|
||||||
|
readonly animation_playing: Property<boolean>;
|
||||||
|
readonly animation_frame_rate: Property<number>;
|
||||||
|
readonly animation_frame: Property<number>;
|
||||||
|
|
||||||
|
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<void> => {
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
15
src/viewer/gui/TextureView.test.ts
Normal file
15
src/viewer/gui/TextureView.test.ts
Normal file
@ -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.");
|
||||||
|
}));
|
@ -3,9 +3,8 @@ import { FileButton } from "../../core/gui/FileButton";
|
|||||||
import { ToolBar } from "../../core/gui/ToolBar";
|
import { ToolBar } from "../../core/gui/ToolBar";
|
||||||
import { RendererWidget } from "../../core/gui/RendererWidget";
|
import { RendererWidget } from "../../core/gui/RendererWidget";
|
||||||
import { TextureRenderer } from "../rendering/TextureRenderer";
|
import { TextureRenderer } from "../rendering/TextureRenderer";
|
||||||
import { TextureStore } from "../stores/TextureStore";
|
|
||||||
import { DisposableThreeRenderer } from "../../core/rendering/Renderer";
|
|
||||||
import { ResizableView } from "../../core/gui/ResizableView";
|
import { ResizableView } from "../../core/gui/ResizableView";
|
||||||
|
import { TextureController } from "../controllers/TextureController";
|
||||||
|
|
||||||
export class TextureView extends ResizableView {
|
export class TextureView extends ResizableView {
|
||||||
readonly element = div({ className: "viewer_TextureView" });
|
readonly element = div({ className: "viewer_TextureView" });
|
||||||
@ -19,18 +18,16 @@ export class TextureView extends ResizableView {
|
|||||||
|
|
||||||
private readonly renderer_view: RendererWidget;
|
private readonly renderer_view: RendererWidget;
|
||||||
|
|
||||||
constructor(texture_store: TextureStore, three_renderer: DisposableThreeRenderer) {
|
constructor(ctrl: TextureController, renderer: TextureRenderer) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
this.renderer_view = this.add(
|
this.renderer_view = this.add(new RendererWidget(renderer));
|
||||||
new RendererWidget(new TextureRenderer(three_renderer, texture_store)),
|
|
||||||
);
|
|
||||||
|
|
||||||
this.element.append(this.tool_bar.element, this.renderer_view.element);
|
this.element.append(this.tool_bar.element, this.renderer_view.element);
|
||||||
|
|
||||||
this.disposables(
|
this.disposables(
|
||||||
this.open_file_button.files.observe(({ value: files }) => {
|
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]);
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { TabContainer } from "../../core/gui/TabContainer";
|
import { TabContainer } from "../../core/gui/TabContainer";
|
||||||
import { Model3DView } from "./model_3d/Model3DView";
|
import { ModelView } from "./model/ModelView";
|
||||||
import { TextureView } from "./TextureView";
|
import { TextureView } from "./TextureView";
|
||||||
import { ResizableView } from "../../core/gui/ResizableView";
|
import { ResizableView } from "../../core/gui/ResizableView";
|
||||||
import { GuiStore } from "../../core/stores/GuiStore";
|
import { GuiStore } from "../../core/stores/GuiStore";
|
||||||
@ -13,7 +13,7 @@ export class ViewerView extends ResizableView {
|
|||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
gui_store: GuiStore,
|
gui_store: GuiStore,
|
||||||
create_model_3d_view: () => Promise<Model3DView>,
|
create_model_view: () => Promise<ModelView>,
|
||||||
create_texture_view: () => Promise<TextureView>,
|
create_texture_view: () => Promise<TextureView>,
|
||||||
) {
|
) {
|
||||||
super();
|
super();
|
||||||
@ -24,13 +24,13 @@ export class ViewerView extends ResizableView {
|
|||||||
tabs: [
|
tabs: [
|
||||||
{
|
{
|
||||||
title: "Models",
|
title: "Models",
|
||||||
key: "models",
|
key: "model",
|
||||||
path: "/models",
|
path: "/models",
|
||||||
create_view: create_model_3d_view,
|
create_view: create_model_view,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: "Textures",
|
title: "Textures",
|
||||||
key: "textures",
|
key: "texture",
|
||||||
path: "/textures",
|
path: "/textures",
|
||||||
create_view: create_texture_view,
|
create_view: create_texture_view,
|
||||||
},
|
},
|
||||||
|
48
src/viewer/gui/__snapshots__/TextureView.test.ts.snap
Normal file
48
src/viewer/gui/__snapshots__/TextureView.test.ts.snap
Normal file
@ -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`] = `
|
||||||
|
<div
|
||||||
|
class="viewer_TextureView"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="core_ToolBar"
|
||||||
|
style="height: 33px;"
|
||||||
|
>
|
||||||
|
<label
|
||||||
|
class="core_FileButton core_Button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
class="core_FileButton_inner core_Button_inner"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
class="core_FileButton_left core_Button_left"
|
||||||
|
>
|
||||||
|
<span>
|
||||||
|
<span
|
||||||
|
class="fas fa-file"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
<span
|
||||||
|
class="core_Button_center"
|
||||||
|
>
|
||||||
|
Open file...
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
<input
|
||||||
|
accept=".afs, .xvm"
|
||||||
|
class="core_FileButton_input core_Button_inner"
|
||||||
|
type="file"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="core_RendererWidget"
|
||||||
|
>
|
||||||
|
<canvas
|
||||||
|
style="outline: none;"
|
||||||
|
tabindex="0"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
23
src/viewer/gui/model/CharacterClassOptionsView.css
Normal file
23
src/viewer/gui/model/CharacterClassOptionsView.css
Normal file
@ -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;
|
||||||
|
}
|
50
src/viewer/gui/model/CharacterClassOptionsView.ts
Normal file
50
src/viewer/gui/model/CharacterClassOptionsView.ts
Normal file
@ -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<SectionId> = 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<number | undefined> = 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();
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
.viewer_Model3DSelectListView {
|
.viewer_model_CharacterClassSelectionView {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
list-style: none;
|
list-style: none;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@ -6,16 +6,16 @@
|
|||||||
overflow: auto;
|
overflow: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.viewer_Model3DSelectListView li {
|
.viewer_model_CharacterClassSelectionView li {
|
||||||
padding: 4px 8px;
|
padding: 4px 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.viewer_Model3DSelectListView li:hover {
|
.viewer_model_CharacterClassSelectionView li:hover {
|
||||||
color: hsl(0, 0%, 90%);
|
color: hsl(0, 0%, 90%);
|
||||||
background-color: hsl(0, 0%, 18%);
|
background-color: hsl(0, 0%, 18%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.viewer_Model3DSelectListView li.active {
|
.viewer_model_CharacterClassSelectionView li.active {
|
||||||
color: hsl(0, 0%, 90%);
|
color: hsl(0, 0%, 90%);
|
||||||
background-color: hsl(0, 0%, 21%);
|
background-color: hsl(0, 0%, 21%);
|
||||||
}
|
}
|
@ -1,35 +1,30 @@
|
|||||||
import "./Model3DSelectListView.css";
|
import "./CharacterClassSelectionView.css";
|
||||||
import { Property } from "../../../core/observable/property/Property";
|
import { Property } from "../../../core/observable/property/Property";
|
||||||
import { li, ul } from "../../../core/gui/dom";
|
import { li, ul } from "../../../core/gui/dom";
|
||||||
import { ResizableView } from "../../../core/gui/ResizableView";
|
import { ResizableView } from "../../../core/gui/ResizableView";
|
||||||
|
|
||||||
export class Model3DSelectListView<T extends { name: string }> extends ResizableView {
|
export class CharacterClassSelectionView<T extends { name: string }> 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";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private selected_model?: T;
|
private selected_model?: T;
|
||||||
private selected_element?: HTMLLIElement;
|
private selected_element?: HTMLLIElement;
|
||||||
|
|
||||||
|
readonly element = ul({ className: "viewer_model_CharacterClassSelectionView" });
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private models: readonly T[],
|
private character_classes: readonly T[],
|
||||||
private selected: Property<T | undefined>,
|
private selected: Property<T | undefined>,
|
||||||
private set_selected: (selected: T) => void,
|
private set_selected: (selected: T) => void,
|
||||||
|
private border_left: boolean,
|
||||||
) {
|
) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
this.element.onclick = this.list_click;
|
this.element.onclick = this.list_click;
|
||||||
|
|
||||||
models.forEach((model, index) => {
|
if (border_left) {
|
||||||
this.element.append(li({ data: { index: index.toString() } }, model.name));
|
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(
|
this.disposables(
|
||||||
@ -41,7 +36,7 @@ export class Model3DSelectListView<T extends { name: string }> extends Resizable
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (model && model !== this.selected_model) {
|
if (model && model !== this.selected_model) {
|
||||||
const index = this.models.indexOf(model);
|
const index = this.character_classes.indexOf(model);
|
||||||
|
|
||||||
if (index !== -1) {
|
if (index !== -1) {
|
||||||
this.selected_element = this.element.childNodes[index] as HTMLLIElement;
|
this.selected_element = this.element.childNodes[index] as HTMLLIElement;
|
||||||
@ -67,7 +62,7 @@ export class Model3DSelectListView<T extends { name: string }> extends Resizable
|
|||||||
const index = parseInt(e.target.dataset["index"]!, 10);
|
const index = parseInt(e.target.dataset["index"]!, 10);
|
||||||
|
|
||||||
this.selected_element = e.target;
|
this.selected_element = e.target;
|
||||||
this.set_selected(this.models[index]);
|
this.set_selected(this.character_classes[index]);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
@ -5,10 +5,10 @@ import { NumberInput } from "../../../core/gui/NumberInput";
|
|||||||
import { PSO_FRAME_RATE } from "../../../core/rendering/conversion/ninja_animation";
|
import { PSO_FRAME_RATE } from "../../../core/rendering/conversion/ninja_animation";
|
||||||
import { Label } from "../../../core/gui/Label";
|
import { Label } from "../../../core/gui/Label";
|
||||||
import { Icon } from "../../../core/gui/dom";
|
import { Icon } from "../../../core/gui/dom";
|
||||||
import { Model3DStore } from "../../stores/Model3DStore";
|
|
||||||
import { View } from "../../../core/gui/View";
|
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;
|
private readonly toolbar: ToolBar;
|
||||||
|
|
||||||
get element(): HTMLElement {
|
get element(): HTMLElement {
|
||||||
@ -19,7 +19,7 @@ export class Model3DToolBarView extends View {
|
|||||||
return this.toolbar.height;
|
return this.toolbar.height;
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor(model_3d_store: Model3DStore) {
|
constructor(ctrl: ModelToolBarController) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
const open_file_button = new FileButton("Open file...", {
|
const open_file_button = new FileButton("Open file...", {
|
||||||
@ -37,12 +37,10 @@ export class Model3DToolBarView extends View {
|
|||||||
const animation_frame_input = new NumberInput(1, {
|
const animation_frame_input = new NumberInput(1, {
|
||||||
label: "Frame:",
|
label: "Frame:",
|
||||||
min: 1,
|
min: 1,
|
||||||
max: model_3d_store.animation_frame_count,
|
max: ctrl.animation_frame_count,
|
||||||
step: 1,
|
step: 1,
|
||||||
});
|
});
|
||||||
const animation_frame_count_label = new Label(
|
const animation_frame_count_label = new Label(ctrl.animation_frame_count_label);
|
||||||
model_3d_store.animation_frame_count.map(count => `/ ${count}`),
|
|
||||||
);
|
|
||||||
|
|
||||||
this.toolbar = this.add(
|
this.toolbar = this.add(
|
||||||
new ToolBar(
|
new ToolBar(
|
||||||
@ -58,36 +56,30 @@ export class Model3DToolBarView extends View {
|
|||||||
// Always-enabled controls.
|
// Always-enabled controls.
|
||||||
this.disposables(
|
this.disposables(
|
||||||
open_file_button.files.observe(({ value: files }) => {
|
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 }) =>
|
skeleton_checkbox.checked.observe(({ value }) => ctrl.set_show_skeleton(value)),
|
||||||
model_3d_store.set_show_skeleton(value),
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
// Controls that are only enabled when an animation is selected.
|
// 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(
|
this.disposables(
|
||||||
play_animation_checkbox.enabled.bind_to(enabled),
|
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 }) =>
|
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.enabled.bind_to(enabled),
|
||||||
animation_frame_rate_input.value.observe(({ value }) =>
|
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.enabled.bind_to(enabled),
|
||||||
animation_frame_input.value.bind_to(
|
animation_frame_input.value.bind_to(ctrl.animation_frame),
|
||||||
model_3d_store.animation_frame.map(v => Math.round(v)),
|
animation_frame_input.value.observe(({ value }) => ctrl.set_animation_frame(value)),
|
||||||
),
|
|
||||||
animation_frame_input.value.observe(({ value }) =>
|
|
||||||
model_3d_store.set_animation_frame(value),
|
|
||||||
),
|
|
||||||
|
|
||||||
animation_frame_count_label.enabled.bind_to(enabled),
|
animation_frame_count_label.enabled.bind_to(enabled),
|
||||||
);
|
);
|
@ -1,4 +1,4 @@
|
|||||||
.viewer_Model3DView_container {
|
.viewer_model_ModelView_container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
}
|
}
|
31
src/viewer/gui/model/ModelView.test.ts
Normal file
31
src/viewer/gui/model/ModelView.test.ts
Normal file
@ -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();
|
||||||
|
}));
|
102
src/viewer/gui/model/ModelView.ts
Normal file
102
src/viewer/gui/model/ModelView.ts
Normal file
@ -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<CharacterClassModel>;
|
||||||
|
private options_view: CharacterClassOptionsView;
|
||||||
|
private renderer_view: RendererWidget;
|
||||||
|
private animation_selection_view: CharacterClassSelectionView<CharacterClassAnimationModel>;
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
3313
src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap
Normal file
3313
src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap
Normal file
File diff suppressed because it is too large
Load Diff
@ -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<CharacterClassModel>;
|
|
||||||
private animation_list_view: Model3DSelectListView<CharacterClassAnimationModel>;
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
|
@ -4,9 +4,18 @@ import { HttpClient } from "../core/HttpClient";
|
|||||||
import { DisposableThreeRenderer } from "../core/rendering/Renderer";
|
import { DisposableThreeRenderer } from "../core/rendering/Renderer";
|
||||||
import { Disposable } from "../core/observable/Disposable";
|
import { Disposable } from "../core/observable/Disposable";
|
||||||
import { Disposer } from "../core/observable/Disposer";
|
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(
|
export function initialize_viewer(
|
||||||
http_client: HttpClient,
|
http_client: HttpClient,
|
||||||
|
random: Random,
|
||||||
gui_store: GuiStore,
|
gui_store: GuiStore,
|
||||||
create_three_renderer: () => DisposableThreeRenderer,
|
create_three_renderer: () => DisposableThreeRenderer,
|
||||||
): { view: ViewerView } & Disposable {
|
): { view: ViewerView } & Disposable {
|
||||||
@ -14,36 +23,36 @@ export function initialize_viewer(
|
|||||||
|
|
||||||
const view = new ViewerView(
|
const view = new ViewerView(
|
||||||
gui_store,
|
gui_store,
|
||||||
|
|
||||||
async () => {
|
async () => {
|
||||||
const { Model3DStore } = await import("./stores/Model3DStore");
|
const { ModelController } = await import("./controllers/model/ModelController");
|
||||||
const { Model3DView } = await import("./gui/model_3d/Model3DView");
|
const { ModelView } = await import("./gui/model/ModelView");
|
||||||
const { CharacterClassAssetLoader } = await import(
|
const { CharacterClassAssetLoader } = await import(
|
||||||
"./loading/CharacterClassAssetLoader"
|
"./loading/CharacterClassAssetLoader"
|
||||||
);
|
);
|
||||||
const asset_loader = disposer.add(new CharacterClassAssetLoader(http_client));
|
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) {
|
return new ModelView(
|
||||||
store.dispose();
|
model_controller,
|
||||||
} else {
|
new ModelToolBarView(model_tool_bar_controller),
|
||||||
disposer.add(store);
|
new CharacterClassOptionsView(character_class_options_controller),
|
||||||
}
|
new ModelRenderer(store, create_three_renderer()),
|
||||||
|
);
|
||||||
return new Model3DView(gui_store, store, create_three_renderer());
|
|
||||||
},
|
},
|
||||||
|
|
||||||
async () => {
|
async () => {
|
||||||
const { TextureStore } = await import("./stores/TextureStore");
|
const { TextureController } = await import("./controllers/TextureController");
|
||||||
const { TextureView } = await import("./gui/TextureView");
|
const { TextureView } = await import("./gui/TextureView");
|
||||||
const store = new TextureStore();
|
const controller = disposer.add(new TextureController());
|
||||||
|
|
||||||
if (disposer.disposed) {
|
return new TextureView(
|
||||||
store.dispose();
|
controller,
|
||||||
} else {
|
new TextureRenderer(controller, create_three_renderer()),
|
||||||
disposer.add(store);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
return new TextureView(store, create_three_renderer());
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -327,7 +327,7 @@ function texture_ids(
|
|||||||
case FOMARL: {
|
case FOMARL: {
|
||||||
const body_idx = body * 16;
|
const body_idx = body * 16;
|
||||||
return {
|
return {
|
||||||
section_id: section_id + 310,
|
section_id: section_id + 326,
|
||||||
body: [body_idx, body_idx + 2, body_idx + 1, 322 /*hands*/],
|
body: [body_idx, body_idx + 2, body_idx + 1, 322 /*hands*/],
|
||||||
head: [288],
|
head: [288],
|
||||||
hair: [undefined, undefined, 308],
|
hair: [undefined, undefined, 308],
|
||||||
|
@ -1,10 +1,8 @@
|
|||||||
import {
|
import {
|
||||||
AdditiveBlending,
|
|
||||||
AnimationClip,
|
AnimationClip,
|
||||||
AnimationMixer,
|
AnimationMixer,
|
||||||
Clock,
|
Clock,
|
||||||
DoubleSide,
|
DoubleSide,
|
||||||
Mesh,
|
|
||||||
MeshBasicMaterial,
|
MeshBasicMaterial,
|
||||||
MeshLambertMaterial,
|
MeshLambertMaterial,
|
||||||
Object3D,
|
Object3D,
|
||||||
@ -25,10 +23,10 @@ import {
|
|||||||
import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer";
|
import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer";
|
||||||
import { Disposer } from "../../core/observable/Disposer";
|
import { Disposer } from "../../core/observable/Disposer";
|
||||||
import { ChangeEvent } from "../../core/observable/Observable";
|
import { ChangeEvent } from "../../core/observable/Observable";
|
||||||
import { Model3DStore } from "../stores/Model3DStore";
|
|
||||||
import { LogManager } from "../../core/Logger";
|
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({
|
const DEFAULT_MATERIAL = new MeshLambertMaterial({
|
||||||
color: 0xffffff,
|
color: 0xffffff,
|
||||||
@ -40,7 +38,7 @@ const DEFAULT_SKINNED_MATERIAL = new MeshLambertMaterial({
|
|||||||
side: DoubleSide,
|
side: DoubleSide,
|
||||||
});
|
});
|
||||||
|
|
||||||
export class Model3DRenderer extends Renderer implements Disposable {
|
export class ModelRenderer extends Renderer implements Disposable {
|
||||||
private readonly disposer = new Disposer();
|
private readonly disposer = new Disposer();
|
||||||
private readonly clock = new Clock();
|
private readonly clock = new Clock();
|
||||||
private mesh?: Object3D;
|
private mesh?: Object3D;
|
||||||
@ -53,23 +51,21 @@ export class Model3DRenderer extends Renderer implements Disposable {
|
|||||||
|
|
||||||
readonly camera = new PerspectiveCamera(75, 1, 1, 200);
|
readonly camera = new PerspectiveCamera(75, 1, 1, 200);
|
||||||
|
|
||||||
constructor(
|
constructor(private readonly store: ModelStore, three_renderer: DisposableThreeRenderer) {
|
||||||
three_renderer: DisposableThreeRenderer,
|
|
||||||
private readonly model_3d_store: Model3DStore,
|
|
||||||
) {
|
|
||||||
super(three_renderer);
|
super(three_renderer);
|
||||||
|
|
||||||
this.disposer.add_all(
|
this.disposer.add_all(
|
||||||
model_3d_store.current_nj_data.observe(this.nj_data_or_xvm_changed),
|
store.current_nj_object.observe(this.nj_object_or_xvm_changed),
|
||||||
model_3d_store.current_textures.observe(this.nj_data_or_xvm_changed),
|
store.current_textures.observe(this.nj_object_or_xvm_changed),
|
||||||
model_3d_store.current_nj_motion.observe(this.nj_motion_changed),
|
store.current_nj_motion.observe(this.nj_motion_changed),
|
||||||
model_3d_store.show_skeleton.observe(this.show_skeleton_changed),
|
store.show_skeleton.observe(this.show_skeleton_changed),
|
||||||
model_3d_store.animation_playing.observe(this.animation_playing_changed),
|
store.animation_playing.observe(this.animation_playing_changed),
|
||||||
model_3d_store.animation_frame_rate.observe(this.animation_frame_rate_changed),
|
store.animation_frame_rate.observe(this.animation_frame_rate_changed),
|
||||||
model_3d_store.animation_frame.observe(this.animation_frame_changed),
|
store.animation_frame.observe(this.animation_frame_changed),
|
||||||
);
|
);
|
||||||
|
|
||||||
this.init_camera_controls();
|
this.init_camera_controls();
|
||||||
|
this.reset_camera(new Vector3(0, 10, 20), new Vector3(0, 0, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
set_size(width: number, height: number): void {
|
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) {
|
if (this.mesh) {
|
||||||
this.scene.remove(this.mesh);
|
this.scene.remove(this.mesh);
|
||||||
this.mesh = undefined;
|
this.mesh = undefined;
|
||||||
@ -105,20 +101,22 @@ export class Model3DRenderer extends Renderer implements Disposable {
|
|||||||
this.skeleton_helper = undefined;
|
this.skeleton_helper = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.animation) {
|
const nj_object = this.store.current_nj_object.val;
|
||||||
this.animation.mixer.stopAllAction();
|
|
||||||
if (this.mesh) this.animation.mixer.uncacheRoot(this.mesh);
|
|
||||||
this.animation = undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
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) {
|
if (this.animation) {
|
||||||
const { nj_object, has_skeleton } = nj_data;
|
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;
|
// Convert textures and geometry.
|
||||||
|
const textures = this.store.current_textures.val.map(tex => {
|
||||||
const textures = this.model_3d_store.current_textures.val.map(tex => {
|
|
||||||
if (tex) {
|
if (tex) {
|
||||||
try {
|
try {
|
||||||
return xvr_texture_to_texture(tex);
|
return xvr_texture_to_texture(tex);
|
||||||
@ -130,6 +128,9 @@ export class Model3DRenderer extends Renderer implements Disposable {
|
|||||||
return undefined;
|
return undefined;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const geometry = ninja_object_to_buffer_geometry(nj_object);
|
||||||
|
const has_skeleton = geometry.getAttribute("skinIndex") != undefined;
|
||||||
|
|
||||||
const materials = textures.map(tex =>
|
const materials = textures.map(tex =>
|
||||||
tex
|
tex
|
||||||
? new MeshBasicMaterial({
|
? new MeshBasicMaterial({
|
||||||
@ -145,64 +146,66 @@ export class Model3DRenderer extends Renderer implements Disposable {
|
|||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (has_skeleton) {
|
this.mesh = has_skeleton
|
||||||
mesh = create_skinned_mesh(
|
? create_skinned_mesh(geometry, materials, DEFAULT_SKINNED_MATERIAL)
|
||||||
ninja_object_to_buffer_geometry(nj_object),
|
: create_mesh(geometry, materials, DEFAULT_MATERIAL);
|
||||||
materials,
|
|
||||||
DEFAULT_SKINNED_MATERIAL,
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
mesh = create_mesh(
|
|
||||||
ninja_object_to_buffer_geometry(nj_object),
|
|
||||||
materials,
|
|
||||||
DEFAULT_MATERIAL,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure we rotate around the center of the model instead of its origin.
|
// 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;
|
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(this.mesh);
|
||||||
this.scene.add(mesh);
|
|
||||||
|
|
||||||
this.skeleton_helper = new SkeletonHelper(mesh);
|
// Add skeleton.
|
||||||
this.skeleton_helper.visible = this.model_3d_store.show_skeleton.val;
|
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.skeleton_helper.material as any).linewidth = 3;
|
||||||
this.scene.add(this.skeleton_helper);
|
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();
|
this.schedule_render();
|
||||||
};
|
};
|
||||||
|
|
||||||
private nj_motion_changed = ({ value: nj_motion }: ChangeEvent<NjMotion | undefined>): void => {
|
private nj_motion_changed = ({ value: nj_motion }: ChangeEvent<NjMotion | undefined>): void => {
|
||||||
let mixer!: AnimationMixer;
|
let mixer: AnimationMixer | undefined;
|
||||||
|
|
||||||
if (this.animation) {
|
if (this.animation) {
|
||||||
this.animation.mixer.stopAllAction();
|
this.animation.mixer.stopAllAction();
|
||||||
|
this.animation.mixer.uncacheAction(this.animation.clip);
|
||||||
mixer = this.animation.mixer;
|
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);
|
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 = {
|
this.animation = { mixer, clip };
|
||||||
mixer,
|
|
||||||
clip,
|
|
||||||
};
|
|
||||||
|
|
||||||
this.clock.start();
|
this.clock.start();
|
||||||
this.animation.mixer.clipAction(this.animation.clip).play();
|
mixer.clipAction(clip).play();
|
||||||
this.schedule_render();
|
this.schedule_render();
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -233,7 +236,7 @@ export class Model3DRenderer extends Renderer implements Disposable {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private animation_frame_changed = ({ value: frame }: ChangeEvent<number>): void => {
|
private animation_frame_changed = ({ value: frame }: ChangeEvent<number>): 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) {
|
if (this.animation && nj_motion) {
|
||||||
const frame_count = nj_motion.frame_count;
|
const frame_count = nj_motion.frame_count;
|
||||||
@ -255,7 +258,7 @@ export class Model3DRenderer extends Renderer implements Disposable {
|
|||||||
|
|
||||||
if (!action.paused) {
|
if (!action.paused) {
|
||||||
this.update_animation_time = false;
|
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;
|
this.update_animation_time = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -10,10 +10,10 @@ import {
|
|||||||
import { Disposable } from "../../core/observable/Disposable";
|
import { Disposable } from "../../core/observable/Disposable";
|
||||||
import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer";
|
import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer";
|
||||||
import { Disposer } from "../../core/observable/Disposer";
|
import { Disposer } from "../../core/observable/Disposer";
|
||||||
import { TextureStore } from "../stores/TextureStore";
|
|
||||||
import { XvrTexture } from "../../core/data_formats/parsing/ninja/texture";
|
import { XvrTexture } from "../../core/data_formats/parsing/ninja/texture";
|
||||||
import { xvr_texture_to_texture } from "../../core/rendering/conversion/ninja_textures";
|
import { xvr_texture_to_texture } from "../../core/rendering/conversion/ninja_textures";
|
||||||
import { LogManager } from "../../core/Logger";
|
import { LogManager } from "../../core/Logger";
|
||||||
|
import { TextureController } from "../controllers/TextureController";
|
||||||
|
|
||||||
const logger = LogManager.get("viewer/rendering/TextureRenderer");
|
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);
|
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);
|
super(three_renderer);
|
||||||
|
|
||||||
this.disposer.add_all(
|
this.disposer.add_all(
|
||||||
texture_store.textures.observe(({ value: textures }) => {
|
ctrl.textures.observe(({ value: textures }) => {
|
||||||
this.scene.remove(...this.quad_meshes);
|
this.scene.remove(...this.quad_meshes);
|
||||||
|
|
||||||
this.render_textures(textures);
|
this.render_textures(textures);
|
||||||
|
@ -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<CharacterClassModel | undefined> = property(
|
|
||||||
undefined,
|
|
||||||
);
|
|
||||||
private readonly _current_nj_data = property<NjData | undefined>(undefined);
|
|
||||||
private readonly _current_textures = list_property<XvrTexture | undefined>();
|
|
||||||
private readonly _show_skeleton: WritableProperty<boolean> = property(false);
|
|
||||||
private readonly _current_animation: WritableProperty<
|
|
||||||
CharacterClassAnimationModel | undefined
|
|
||||||
> = property(undefined);
|
|
||||||
private readonly _current_nj_motion = property<NjMotion | undefined>(undefined);
|
|
||||||
private readonly _animation_playing: WritableProperty<boolean> = property(true);
|
|
||||||
private readonly _animation_frame_rate: WritableProperty<number> = property(PSO_FRAME_RATE);
|
|
||||||
private readonly _animation_frame: WritableProperty<number> = 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<CharacterClassModel | undefined> = this._current_model;
|
|
||||||
readonly current_nj_data: Property<NjData | undefined> = this._current_nj_data;
|
|
||||||
readonly current_textures: ListProperty<XvrTexture | undefined> = this._current_textures;
|
|
||||||
readonly show_skeleton: Property<boolean> = this._show_skeleton;
|
|
||||||
readonly current_animation: Property<CharacterClassAnimationModel | undefined> = this
|
|
||||||
._current_animation;
|
|
||||||
readonly current_nj_motion: Property<NjMotion | undefined> = this._current_nj_motion;
|
|
||||||
readonly animation_playing: Property<boolean> = this._animation_playing;
|
|
||||||
readonly animation_frame_rate: Property<number> = this._animation_frame_rate;
|
|
||||||
readonly animation_frame: Property<number> = this._animation_frame;
|
|
||||||
readonly animation_frame_count: Property<number> = 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<void> => {
|
|
||||||
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<void> => {
|
|
||||||
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<void> => {
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
226
src/viewer/stores/ModelStore.ts
Normal file
226
src/viewer/stores/ModelStore.ts
Normal file
@ -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<SectionId>;
|
||||||
|
private readonly _current_body: WritableProperty<number | undefined> = property(0);
|
||||||
|
private readonly _current_animation: WritableProperty<
|
||||||
|
CharacterClassAnimationModel | undefined
|
||||||
|
> = property(undefined);
|
||||||
|
|
||||||
|
// Geometry, textures and animations.
|
||||||
|
private readonly _current_nj_object = property<NjObject | undefined>(undefined);
|
||||||
|
private readonly _current_textures = list_property<XvrTexture | undefined>();
|
||||||
|
private readonly _current_nj_motion = property<NjMotion | undefined>(undefined);
|
||||||
|
|
||||||
|
// User settings.
|
||||||
|
private readonly _show_skeleton: WritableProperty<boolean> = property(false);
|
||||||
|
private readonly _animation_playing: WritableProperty<boolean> = property(true);
|
||||||
|
private readonly _animation_frame_rate: WritableProperty<number> = property(PSO_FRAME_RATE);
|
||||||
|
private readonly _animation_frame: WritableProperty<number> = 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<CharacterClassModel | undefined> = this
|
||||||
|
._current_character_class;
|
||||||
|
readonly current_section_id: Property<SectionId>;
|
||||||
|
readonly current_body: Property<number | undefined> = this._current_body;
|
||||||
|
readonly animations: readonly CharacterClassAnimationModel[] = new Array(572)
|
||||||
|
.fill(undefined)
|
||||||
|
.map((_, i) => new CharacterClassAnimationModel(i, `Animation ${i + 1}`));
|
||||||
|
readonly current_animation: Property<CharacterClassAnimationModel | undefined> = this
|
||||||
|
._current_animation;
|
||||||
|
|
||||||
|
// Geometry, textures and animations.
|
||||||
|
readonly current_nj_object: Property<NjObject | undefined> = this._current_nj_object;
|
||||||
|
readonly current_textures: ListProperty<XvrTexture | undefined> = this._current_textures;
|
||||||
|
readonly current_nj_motion: Property<NjMotion | undefined> = this._current_nj_motion;
|
||||||
|
readonly animation_frame_count: Property<number> = this.current_nj_motion.map(njm =>
|
||||||
|
njm ? njm.frame_count : 0,
|
||||||
|
);
|
||||||
|
|
||||||
|
// User settings.
|
||||||
|
readonly show_skeleton: Property<boolean> = this._show_skeleton;
|
||||||
|
readonly animation_playing: Property<boolean> = this._animation_playing;
|
||||||
|
readonly animation_frame_rate: Property<number> = this._animation_frame_rate;
|
||||||
|
readonly animation_frame: Property<number> = 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<void> => {
|
||||||
|
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<void> => {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
11
test/src/core/rendering/StubThreeRenderer.ts
Normal file
11
test/src/core/rendering/StubThreeRenderer.ts
Normal file
@ -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
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user