phantasmal-world/src/stores/ModelViewerStore.ts

328 lines
12 KiB
TypeScript
Raw Normal View History

import Logger from "js-logger";
import { action, observable } from "mobx";
2019-07-11 23:30:23 +08:00
import {
AnimationAction,
AnimationClip,
AnimationMixer,
Clock,
2019-07-11 23:30:23 +08:00
Mesh,
MeshLambertMaterial,
SkinnedMesh,
Texture,
Vector3,
DoubleSide,
2019-07-11 23:30:23 +08:00
} from "three";
import { Endianness } from "../data_formats";
import { ArrayBufferCursor } from "../data_formats/cursor/ArrayBufferCursor";
2019-07-07 03:55:02 +08:00
import { NjModel, NjObject, parse_nj, parse_xj } from "../data_formats/parsing/ninja";
import { NjMotion, parse_njm } from "../data_formats/parsing/ninja/motion";
import { parse_xvm } from "../data_formats/parsing/ninja/texture";
import { PlayerAnimation, PlayerModel } from "../domain";
2019-07-11 23:30:23 +08:00
import { read_file } from "../read_file";
import { create_animation_clip, PSO_FRAME_RATE } from "../rendering/animation";
2019-07-11 23:30:23 +08:00
import { ninja_object_to_buffer_geometry, ninja_object_to_skinned_mesh } from "../rendering/models";
import { xvm_to_textures } from "../rendering/textures";
import { get_player_animation_data, get_player_data } from "./binary_assets";
const logger = Logger.get("stores/ModelViewerStore");
2019-07-07 03:55:02 +08:00
const nj_object_cache: Map<string, Promise<NjObject<NjModel>>> = new Map();
const nj_motion_cache: Map<number, Promise<NjMotion>> = new Map();
class ModelViewerStore {
readonly models: PlayerModel[] = [
new PlayerModel("HUmar", 1, 10, new Set([6])),
new PlayerModel("HUnewearl", 1, 10, new Set()),
new PlayerModel("HUcast", 5, 0, new Set()),
new PlayerModel("HUcaseal", 5, 0, new Set()),
new PlayerModel("RAmar", 1, 10, new Set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])),
new PlayerModel("RAmarl", 1, 10, new Set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])),
new PlayerModel("RAcast", 5, 0, new Set()),
new PlayerModel("RAcaseal", 5, 0, new Set()),
new PlayerModel("FOmar", 1, 10, new Set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])),
new PlayerModel("FOmarl", 1, 10, new Set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])),
new PlayerModel("FOnewm", 1, 10, new Set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])),
new PlayerModel("FOnewearl", 1, 10, new Set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])),
];
readonly animations: PlayerAnimation[] = new Array(572)
.fill(undefined)
.map((_, i) => new PlayerAnimation(i, `Animation ${i + 1}`));
readonly clock = new Clock();
@observable.ref current_player_model?: PlayerModel;
2019-07-07 03:55:02 +08:00
@observable.ref current_model?: NjObject<NjModel>;
2019-07-02 03:20:09 +08:00
@observable.ref current_bone_count: number = 0;
2019-07-11 23:30:23 +08:00
@observable.ref current_obj3d?: Mesh;
@observable.ref animation?: {
player_animation?: PlayerAnimation;
mixer: AnimationMixer;
clip: AnimationClip;
action: AnimationAction;
};
@observable animation_playing: boolean = false;
@observable animation_frame_rate: number = PSO_FRAME_RATE;
@observable animation_frame: number = 0;
@observable animation_frame_count: number = 0;
private has_skeleton = false;
@observable show_skeleton: boolean = false;
set_animation_frame_rate = action("set_animation_frame_rate", (rate: number) => {
if (this.animation) {
this.animation.mixer.timeScale = rate / PSO_FRAME_RATE;
this.animation_frame_rate = rate;
}
});
set_animation_frame = action("set_animation_frame", (frame: number) => {
if (this.animation) {
const frame_count = this.animation_frame_count;
if (frame > frame_count) frame = 1;
if (frame < 1) frame = frame_count;
this.animation.action.time = (frame - 1) / PSO_FRAME_RATE;
this.animation_frame = frame;
}
});
load_model = async (model: PlayerModel) => {
2019-07-01 14:53:16 +08:00
const object = await this.get_player_ninja_object(model);
2019-07-11 23:30:23 +08:00
this.set_model(object, true, model);
2019-07-02 03:20:09 +08:00
// Ignore the bones from the head parts.
this.current_bone_count = 64;
};
load_animation = async (animation: PlayerAnimation) => {
const nj_motion = await this.get_nj_motion(animation);
if (this.current_model) {
this.set_animation(create_animation_clip(this.current_model, nj_motion), animation);
}
};
2019-07-11 23:30:23 +08:00
// TODO: notify user of problems.
load_file = async (file: File) => {
try {
const buffer = await read_file(file);
const cursor = new ArrayBufferCursor(buffer, Endianness.Little);
if (file.name.endsWith(".nj")) {
const model = parse_nj(cursor)[0];
this.set_model(model, true);
} else if (file.name.endsWith(".xj")) {
const model = parse_xj(cursor)[0];
this.set_model(model, false);
} else if (file.name.endsWith(".njm")) {
if (this.current_model) {
const njm = parse_njm(cursor, this.current_bone_count);
this.set_animation(create_animation_clip(this.current_model, njm));
}
} else if (file.name.endsWith(".xvm")) {
if (this.current_model) {
const xvm = parse_xvm(cursor);
this.set_textures(xvm_to_textures(xvm));
}
2019-07-11 23:30:23 +08:00
} else {
logger.error(`Unknown file extension in filename "${file.name}".`);
}
} catch (e) {
logger.error("Couldn't read file.", e);
}
};
pause_animation = action("pause_animation", () => {
if (this.animation) {
this.animation.action.paused = true;
this.animation_playing = false;
this.clock.stop();
}
});
toggle_animation_playing = action("toggle_animation_playing", () => {
if (this.animation) {
this.animation.action.paused = !this.animation.action.paused;
this.animation_playing = !this.animation.action.paused;
if (this.animation_playing) {
this.clock.start();
} else {
this.clock.stop();
}
}
});
update_animation_frame = action("update_animation_frame", () => {
if (this.animation && this.animation_playing) {
const time = this.animation.action.time;
this.animation_frame = Math.round(time * PSO_FRAME_RATE) + 1;
}
});
set_animation = action("set_animation", (clip: AnimationClip, animation?: PlayerAnimation) => {
2019-07-11 23:30:23 +08:00
if (!this.current_obj3d || !(this.current_obj3d instanceof SkinnedMesh)) return;
let mixer: AnimationMixer;
if (this.animation) {
this.animation.mixer.stopAllAction();
mixer = this.animation.mixer;
} else {
mixer = new AnimationMixer(this.current_obj3d);
}
this.animation = {
player_animation: animation,
mixer,
clip,
action: mixer.clipAction(clip),
};
this.clock.start();
this.animation.action.play();
this.animation_playing = true;
this.animation_frame_count = Math.round(PSO_FRAME_RATE * clip.duration) + 1;
});
private set_model = action(
"set_model",
2019-07-11 23:30:23 +08:00
(model: NjObject<NjModel>, skeleton: boolean, player_model?: PlayerModel) => {
if (this.current_obj3d && this.animation) {
this.animation.mixer.stopAllAction();
this.animation.mixer.uncacheRoot(this.current_obj3d);
this.animation = undefined;
}
this.current_player_model = player_model;
this.current_model = model;
this.current_bone_count = model.bone_count();
this.has_skeleton = skeleton;
this.set_obj3d();
}
);
private add_to_bone(
2019-07-07 03:55:02 +08:00
object: NjObject<NjModel>,
head_part: NjObject<NjModel>,
bone_id: number
2019-07-03 02:56:33 +08:00
): void {
2019-07-02 03:20:09 +08:00
const bone = object.get_bone(bone_id);
if (bone) {
bone.evaluation_flags.hidden = false;
bone.evaluation_flags.break_child_trace = false;
bone.children.push(head_part);
}
}
2019-07-07 03:55:02 +08:00
private async get_player_ninja_object(model: PlayerModel): Promise<NjObject<NjModel>> {
let ninja_object = nj_object_cache.get(model.name);
2019-07-01 14:53:16 +08:00
if (ninja_object) {
return ninja_object;
} else {
ninja_object = this.get_all_assets(model);
nj_object_cache.set(model.name, ninja_object);
2019-07-01 14:53:16 +08:00
return ninja_object;
}
}
2019-07-07 03:55:02 +08:00
private async get_all_assets(model: PlayerModel): Promise<NjObject<NjModel>> {
const body_data = await get_player_data(model.name, "Body");
const body = parse_nj(new ArrayBufferCursor(body_data, Endianness.Little))[0];
2019-07-01 14:53:16 +08:00
if (!body) {
throw new Error(`Couldn't parse body for player class ${model.name}.`);
}
const head_data = await get_player_data(model.name, "Head", 0);
const head = parse_nj(new ArrayBufferCursor(head_data, Endianness.Little))[0];
2019-07-01 14:53:16 +08:00
if (head) {
this.add_to_bone(body, head, 59);
}
if (model.hair_styles_count > 0) {
const hair_data = await get_player_data(model.name, "Hair", 0);
const hair = parse_nj(new ArrayBufferCursor(hair_data, Endianness.Little))[0];
2019-07-01 14:53:16 +08:00
if (hair) {
this.add_to_bone(body, hair, 59);
}
if (model.hair_styles_with_accessory.has(0)) {
const accessory_data = await get_player_data(model.name, "Accessory", 0);
const accessory = parse_nj(
new ArrayBufferCursor(accessory_data, Endianness.Little)
)[0];
2019-07-01 14:53:16 +08:00
if (accessory) {
this.add_to_bone(body, accessory, 59);
}
}
}
return body;
}
private async get_nj_motion(animation: PlayerAnimation): Promise<NjMotion> {
let nj_motion = nj_motion_cache.get(animation.id);
if (nj_motion) {
return nj_motion;
} else {
nj_motion = get_player_animation_data(animation.id).then(motion_data =>
parse_njm(
new ArrayBufferCursor(motion_data, Endianness.Little),
this.current_bone_count
)
);
nj_motion_cache.set(animation.id, nj_motion);
return nj_motion;
}
}
private set_textures = action("set_textures", (textures: Texture[]) => {
this.set_obj3d(textures);
});
private set_obj3d = (textures?: Texture[]) => {
if (this.current_model) {
let mesh: Mesh;
let bb_size = new Vector3();
if (this.has_skeleton) {
mesh = ninja_object_to_skinned_mesh(
this.current_model,
textures &&
textures.map(
tex =>
new MeshLambertMaterial({
skinning: true,
map: tex,
side: DoubleSide,
alphaTest: 0.5,
})
)
);
} else {
mesh = new Mesh(
ninja_object_to_buffer_geometry(this.current_model),
new MeshLambertMaterial({
color: 0xff00ff,
side: DoubleSide,
alphaTest: 0.5,
})
);
}
mesh.geometry.boundingBox.getSize(bb_size);
mesh.translateY(-bb_size.y / 2);
this.current_obj3d = mesh;
}
};
}
export const model_viewer_store = new ModelViewerStore();