mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
225 lines
7.2 KiB
TypeScript
225 lines
7.2 KiB
TypeScript
import {
|
|
AnimationAction,
|
|
AnimationClip,
|
|
AnimationMixer,
|
|
Clock,
|
|
DoubleSide,
|
|
Mesh,
|
|
MeshLambertMaterial,
|
|
Object3D,
|
|
PerspectiveCamera,
|
|
SkeletonHelper,
|
|
SkinnedMesh,
|
|
Vector3,
|
|
} from "three";
|
|
import { model_store } from "../stores/Model3DStore";
|
|
import { Disposable } from "../../core/observable/Disposable";
|
|
import { NjMotion } from "../../core/data_formats/parsing/ninja/motion";
|
|
import { xvm_to_textures } from "../../core/rendering/conversion/ninja_textures";
|
|
import { create_mesh, create_skinned_mesh } from "../../core/rendering/conversion/create_mesh";
|
|
import { ninja_object_to_buffer_geometry } from "../../core/rendering/conversion/ninja_geometry";
|
|
import {
|
|
create_animation_clip,
|
|
PSO_FRAME_RATE,
|
|
} from "../../core/rendering/conversion/ninja_animation";
|
|
import { Renderer } from "../../core/rendering/Renderer";
|
|
import { Disposer } from "../../core/observable/Disposer";
|
|
import { ChangeEvent } from "../../core/observable/Observable";
|
|
|
|
export class Model3DRenderer extends Renderer implements Disposable {
|
|
private readonly disposer = new Disposer();
|
|
private readonly clock = new Clock();
|
|
private mesh?: Object3D;
|
|
private skeleton_helper?: SkeletonHelper;
|
|
private animation?: {
|
|
mixer: AnimationMixer;
|
|
clip: AnimationClip;
|
|
action: AnimationAction;
|
|
};
|
|
private update_animation_time = true;
|
|
|
|
readonly camera = new PerspectiveCamera(75, 1, 1, 200);
|
|
|
|
constructor() {
|
|
super();
|
|
|
|
this.disposer.add_all(
|
|
model_store.current_nj_data.observe(this.nj_data_or_xvm_changed),
|
|
model_store.current_xvm.observe(this.nj_data_or_xvm_changed),
|
|
model_store.current_nj_motion.observe(this.nj_motion_changed),
|
|
model_store.show_skeleton.observe(this.show_skeleton_changed),
|
|
model_store.animation_playing.observe(this.animation_playing_changed),
|
|
model_store.animation_frame_rate.observe(this.animation_frame_rate_changed),
|
|
model_store.animation_frame.observe(this.animation_frame_changed),
|
|
);
|
|
|
|
this.init_camera_controls();
|
|
}
|
|
|
|
set_size(width: number, height: number): void {
|
|
this.camera.aspect = width / height;
|
|
this.camera.updateProjectionMatrix();
|
|
super.set_size(width, height);
|
|
}
|
|
|
|
dispose(): void {
|
|
super.dispose();
|
|
this.disposer.dispose();
|
|
}
|
|
|
|
protected render(): void {
|
|
if (this.animation) {
|
|
this.animation.mixer.update(this.clock.getDelta());
|
|
}
|
|
|
|
this.light_holder.quaternion.copy(this.camera.quaternion);
|
|
super.render();
|
|
|
|
if (this.animation && !this.animation.action.paused) {
|
|
this.update_animation_frame();
|
|
this.schedule_render();
|
|
}
|
|
}
|
|
|
|
private nj_data_or_xvm_changed = () => {
|
|
if (this.mesh) {
|
|
this.scene.remove(this.mesh);
|
|
this.mesh = undefined;
|
|
this.scene.remove(this.skeleton_helper!);
|
|
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_data = model_store.current_nj_data.val;
|
|
|
|
if (nj_data) {
|
|
const { nj_object, has_skeleton } = nj_data;
|
|
|
|
let mesh: Mesh;
|
|
|
|
const xvm = model_store.current_xvm.val;
|
|
const textures = xvm ? xvm_to_textures(xvm) : undefined;
|
|
|
|
const materials =
|
|
textures &&
|
|
textures.map(
|
|
tex =>
|
|
new MeshLambertMaterial({
|
|
skinning: has_skeleton,
|
|
map: tex,
|
|
side: DoubleSide,
|
|
alphaTest: 0.5,
|
|
}),
|
|
);
|
|
|
|
if (has_skeleton) {
|
|
mesh = create_skinned_mesh(ninja_object_to_buffer_geometry(nj_object), materials);
|
|
} else {
|
|
mesh = create_mesh(ninja_object_to_buffer_geometry(nj_object), materials);
|
|
}
|
|
|
|
// Make sure we rotate around the center of the model instead of its origin.
|
|
const bb = mesh.geometry.boundingBox;
|
|
const height = bb.max.y - bb.min.y;
|
|
mesh.translateY(-height / 2 - bb.min.y);
|
|
|
|
this.mesh = mesh;
|
|
this.scene.add(mesh);
|
|
|
|
this.skeleton_helper = new SkeletonHelper(mesh);
|
|
this.skeleton_helper.visible = model_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));
|
|
}
|
|
|
|
this.schedule_render();
|
|
};
|
|
|
|
private nj_motion_changed = ({ value: nj_motion }: ChangeEvent<NjMotion | undefined>) => {
|
|
let mixer!: AnimationMixer;
|
|
|
|
if (this.animation) {
|
|
this.animation.mixer.stopAllAction();
|
|
mixer = this.animation.mixer;
|
|
}
|
|
|
|
const nj_data = model_store.current_nj_data.val;
|
|
|
|
if (!this.mesh || !(this.mesh instanceof SkinnedMesh) || !nj_motion || !nj_data) return;
|
|
|
|
if (!this.animation) {
|
|
mixer = new AnimationMixer(this.mesh);
|
|
}
|
|
|
|
const clip = create_animation_clip(nj_data.nj_object, nj_motion);
|
|
|
|
this.animation = {
|
|
mixer,
|
|
clip,
|
|
action: mixer.clipAction(clip),
|
|
};
|
|
|
|
this.clock.start();
|
|
this.animation.action.play();
|
|
this.schedule_render();
|
|
};
|
|
|
|
private show_skeleton_changed = ({ value: show_skeleton }: ChangeEvent<boolean>) => {
|
|
if (this.skeleton_helper) {
|
|
this.skeleton_helper.visible = show_skeleton;
|
|
this.schedule_render();
|
|
}
|
|
};
|
|
|
|
private animation_playing_changed = ({ value: playing }: ChangeEvent<boolean>) => {
|
|
if (this.animation) {
|
|
this.animation.action.paused = !playing;
|
|
|
|
if (playing) {
|
|
this.clock.start();
|
|
this.schedule_render();
|
|
} else {
|
|
this.clock.stop();
|
|
}
|
|
}
|
|
};
|
|
|
|
private animation_frame_rate_changed = ({ value: frame_rate }: ChangeEvent<number>) => {
|
|
if (this.animation) {
|
|
this.animation.mixer.timeScale = frame_rate / PSO_FRAME_RATE;
|
|
}
|
|
};
|
|
|
|
private animation_frame_changed = ({ value: frame }: ChangeEvent<number>) => {
|
|
const nj_motion = model_store.current_nj_motion.val;
|
|
|
|
if (this.animation && nj_motion) {
|
|
const frame_count = nj_motion.frame_count;
|
|
if (frame > frame_count) frame = 1;
|
|
if (frame < 1) frame = frame_count;
|
|
|
|
if (this.update_animation_time) {
|
|
this.animation.action.time = (frame - 1) / PSO_FRAME_RATE;
|
|
}
|
|
|
|
this.schedule_render();
|
|
}
|
|
};
|
|
|
|
private update_animation_frame(): void {
|
|
if (this.animation && !this.animation.action.paused) {
|
|
const time = this.animation.action.time;
|
|
this.update_animation_time = false;
|
|
model_store.animation_frame.val = time * PSO_FRAME_RATE + 1;
|
|
this.update_animation_time = true;
|
|
}
|
|
}
|
|
}
|