2019-07-10 02:22:18 +08:00
|
|
|
import { Cursor } from "../../cursor/Cursor";
|
2019-07-12 19:10:51 +08:00
|
|
|
import { Vec3 } from "../../vector";
|
2019-07-13 23:09:28 +08:00
|
|
|
import { parse_iff } from "../iff";
|
2019-07-07 03:55:02 +08:00
|
|
|
import { NjcmModel, parse_njcm_model } from "./njcm";
|
2019-07-03 00:08:06 +08:00
|
|
|
import { parse_xj_model, XjModel } from "./xj";
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-07-10 02:22:18 +08:00
|
|
|
export const ANGLE_TO_RAD = (2 * Math.PI) / 0xffff;
|
2019-06-30 05:18:52 +08:00
|
|
|
|
2019-07-11 23:30:23 +08:00
|
|
|
const NJCM = 0x4d434a4e;
|
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
export type NjModel = NjcmModel | XjModel;
|
|
|
|
|
|
|
|
export function is_njcm_model(model: NjModel): model is NjcmModel {
|
|
|
|
return model.type === "njcm";
|
|
|
|
}
|
|
|
|
|
|
|
|
export function is_xj_model(model: NjModel): model is XjModel {
|
|
|
|
return model.type === "xj";
|
|
|
|
}
|
2019-06-30 05:18:52 +08:00
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
export class NjObject<M extends NjModel> {
|
|
|
|
evaluation_flags: NjEvaluationFlags;
|
2019-07-03 02:56:33 +08:00
|
|
|
model: M | undefined;
|
|
|
|
position: Vec3;
|
|
|
|
rotation: Vec3; // Euler angles in radians.
|
|
|
|
scale: Vec3;
|
2019-07-07 03:55:02 +08:00
|
|
|
children: NjObject<M>[];
|
2019-07-03 02:56:33 +08:00
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
private bone_cache = new Map<number, NjObject<M> | null>();
|
2019-07-02 03:20:09 +08:00
|
|
|
private _bone_count = -1;
|
2019-07-01 19:05:58 +08:00
|
|
|
|
|
|
|
constructor(
|
2019-07-07 03:55:02 +08:00
|
|
|
evaluation_flags: NjEvaluationFlags,
|
2019-07-03 02:56:33 +08:00
|
|
|
model: M | undefined,
|
|
|
|
position: Vec3,
|
|
|
|
rotation: Vec3, // Euler angles in radians.
|
|
|
|
scale: Vec3,
|
2019-07-07 03:55:02 +08:00
|
|
|
children: NjObject<M>[]
|
2019-07-03 02:56:33 +08:00
|
|
|
) {
|
|
|
|
this.evaluation_flags = evaluation_flags;
|
|
|
|
this.model = model;
|
|
|
|
this.position = position;
|
|
|
|
this.rotation = rotation;
|
|
|
|
this.scale = scale;
|
|
|
|
this.children = children;
|
|
|
|
}
|
2019-07-01 19:05:58 +08:00
|
|
|
|
2019-07-02 03:20:09 +08:00
|
|
|
bone_count(): number {
|
|
|
|
if (this._bone_count === -1) {
|
|
|
|
const id_ref: [number] = [0];
|
|
|
|
this.get_bone_internal(this, Infinity, id_ref);
|
|
|
|
this._bone_count = id_ref[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
return this._bone_count;
|
|
|
|
}
|
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
get_bone(bone_id: number): NjObject<M> | undefined {
|
2019-07-01 19:05:58 +08:00
|
|
|
let bone = this.bone_cache.get(bone_id);
|
|
|
|
|
|
|
|
// Strict check because null means there's no bone with this id.
|
|
|
|
if (bone === undefined) {
|
2019-07-02 03:20:09 +08:00
|
|
|
bone = this.get_bone_internal(this, bone_id, [0]);
|
2019-07-01 19:05:58 +08:00
|
|
|
this.bone_cache.set(bone_id, bone || null);
|
|
|
|
}
|
|
|
|
|
|
|
|
return bone || undefined;
|
|
|
|
}
|
|
|
|
|
2019-07-02 03:20:09 +08:00
|
|
|
private get_bone_internal(
|
2019-07-07 03:55:02 +08:00
|
|
|
object: NjObject<M>,
|
2019-07-01 19:05:58 +08:00
|
|
|
bone_id: number,
|
|
|
|
id_ref: [number]
|
2019-07-07 03:55:02 +08:00
|
|
|
): NjObject<M> | undefined {
|
2019-07-01 19:05:58 +08:00
|
|
|
if (!object.evaluation_flags.skip) {
|
|
|
|
const id = id_ref[0]++;
|
2019-07-02 03:20:09 +08:00
|
|
|
this.bone_cache.set(id, object);
|
2019-07-01 19:05:58 +08:00
|
|
|
|
|
|
|
if (id === bone_id) {
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-02 03:20:09 +08:00
|
|
|
if (!object.evaluation_flags.break_child_trace) {
|
|
|
|
for (const child of object.children) {
|
|
|
|
const bone = this.get_bone_internal(child, bone_id, id_ref);
|
|
|
|
if (bone) return bone;
|
|
|
|
}
|
2019-07-01 19:05:58 +08:00
|
|
|
}
|
|
|
|
}
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
export type NjEvaluationFlags = {
|
2019-07-03 02:56:33 +08:00
|
|
|
no_translate: boolean;
|
|
|
|
no_rotate: boolean;
|
|
|
|
no_scale: boolean;
|
|
|
|
hidden: boolean;
|
|
|
|
break_child_trace: boolean;
|
|
|
|
zxy_rotation_order: boolean;
|
|
|
|
skip: boolean;
|
|
|
|
shape_skip: boolean;
|
|
|
|
};
|
|
|
|
|
2019-07-20 01:34:48 +08:00
|
|
|
/**
|
|
|
|
* Parses an NJCM file.
|
|
|
|
*/
|
2019-07-09 05:56:05 +08:00
|
|
|
export function parse_nj(cursor: Cursor): NjObject<NjcmModel>[] {
|
2019-07-07 03:55:02 +08:00
|
|
|
return parse_ninja(cursor, parse_njcm_model, []);
|
2019-06-30 05:18:52 +08:00
|
|
|
}
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-07-20 01:34:48 +08:00
|
|
|
/**
|
|
|
|
* Parses an NJCM file.
|
|
|
|
*/
|
2019-07-09 05:56:05 +08:00
|
|
|
export function parse_xj(cursor: Cursor): NjObject<XjModel>[] {
|
2019-06-30 05:18:52 +08:00
|
|
|
return parse_ninja(cursor, parse_xj_model, undefined);
|
|
|
|
}
|
|
|
|
|
2019-07-20 01:34:48 +08:00
|
|
|
/**
|
|
|
|
* Parses a ninja object.
|
|
|
|
*/
|
|
|
|
export function parse_xj_object(cursor: Cursor): NjObject<XjModel>[] {
|
|
|
|
return parse_sibling_objects(cursor, parse_xj_model, undefined);
|
|
|
|
}
|
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
function parse_ninja<M extends NjModel>(
|
2019-07-09 05:56:05 +08:00
|
|
|
cursor: Cursor,
|
|
|
|
parse_model: (cursor: Cursor, context: any) => M,
|
2019-06-30 05:18:52 +08:00
|
|
|
context: any
|
2019-07-07 03:55:02 +08:00
|
|
|
): NjObject<M>[] {
|
2019-07-13 23:09:28 +08:00
|
|
|
// POF0 and other chunks types are ignored.
|
2019-07-28 03:47:49 +08:00
|
|
|
const njcm_chunks = parse_iff(cursor).filter(chunk => chunk.type === NJCM);
|
|
|
|
const objects: NjObject<M>[] = [];
|
|
|
|
|
|
|
|
for (const chunk of njcm_chunks) {
|
|
|
|
objects.push(...parse_sibling_objects(chunk.data, parse_model, context));
|
|
|
|
}
|
|
|
|
|
|
|
|
return objects;
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
// TODO: cache model and object offsets so we don't reparse the same data.
|
2019-07-07 03:55:02 +08:00
|
|
|
function parse_sibling_objects<M extends NjModel>(
|
2019-07-09 05:56:05 +08:00
|
|
|
cursor: Cursor,
|
|
|
|
parse_model: (cursor: Cursor, context: any) => M,
|
2019-06-30 05:18:52 +08:00
|
|
|
context: any
|
2019-07-07 03:55:02 +08:00
|
|
|
): NjObject<M>[] {
|
2019-06-28 00:50:22 +08:00
|
|
|
const eval_flags = cursor.u32();
|
|
|
|
const no_translate = (eval_flags & 0b1) !== 0;
|
|
|
|
const no_rotate = (eval_flags & 0b10) !== 0;
|
|
|
|
const no_scale = (eval_flags & 0b100) !== 0;
|
|
|
|
const hidden = (eval_flags & 0b1000) !== 0;
|
|
|
|
const break_child_trace = (eval_flags & 0b10000) !== 0;
|
|
|
|
const zxy_rotation_order = (eval_flags & 0b100000) !== 0;
|
2019-07-01 05:55:28 +08:00
|
|
|
const skip = (eval_flags & 0b1000000) !== 0;
|
|
|
|
const shape_skip = (eval_flags & 0b10000000) !== 0;
|
2019-06-28 00:50:22 +08:00
|
|
|
|
|
|
|
const model_offset = cursor.u32();
|
|
|
|
const pos_x = cursor.f32();
|
|
|
|
const pos_y = cursor.f32();
|
|
|
|
const pos_z = cursor.f32();
|
2019-06-30 05:18:52 +08:00
|
|
|
const rotation_x = cursor.i32() * ANGLE_TO_RAD;
|
|
|
|
const rotation_y = cursor.i32() * ANGLE_TO_RAD;
|
|
|
|
const rotation_z = cursor.i32() * ANGLE_TO_RAD;
|
2019-06-28 00:50:22 +08:00
|
|
|
const scale_x = cursor.f32();
|
|
|
|
const scale_y = cursor.f32();
|
|
|
|
const scale_z = cursor.f32();
|
|
|
|
const child_offset = cursor.u32();
|
|
|
|
const sibling_offset = cursor.u32();
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
let model: M | undefined;
|
2019-07-07 03:55:02 +08:00
|
|
|
let children: NjObject<M>[];
|
|
|
|
let siblings: NjObject<M>[];
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
if (model_offset) {
|
2019-06-28 00:50:22 +08:00
|
|
|
cursor.seek_start(model_offset);
|
2019-06-30 05:18:52 +08:00
|
|
|
model = parse_model(cursor, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
if (child_offset) {
|
2019-06-28 00:50:22 +08:00
|
|
|
cursor.seek_start(child_offset);
|
2019-06-30 05:18:52 +08:00
|
|
|
children = parse_sibling_objects(cursor, parse_model, context);
|
|
|
|
} else {
|
|
|
|
children = [];
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
if (sibling_offset) {
|
|
|
|
cursor.seek_start(sibling_offset);
|
2019-06-30 05:18:52 +08:00
|
|
|
siblings = parse_sibling_objects(cursor, parse_model, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
} else {
|
2019-06-30 05:18:52 +08:00
|
|
|
siblings = [];
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
2019-06-30 05:18:52 +08:00
|
|
|
|
2019-07-07 03:55:02 +08:00
|
|
|
const object = new NjObject<M>(
|
2019-07-01 19:05:58 +08:00
|
|
|
{
|
2019-06-30 05:18:52 +08:00
|
|
|
no_translate,
|
|
|
|
no_rotate,
|
|
|
|
no_scale,
|
|
|
|
hidden,
|
|
|
|
break_child_trace,
|
|
|
|
zxy_rotation_order,
|
2019-07-01 05:55:28 +08:00
|
|
|
skip,
|
|
|
|
shape_skip,
|
2019-06-30 05:18:52 +08:00
|
|
|
},
|
|
|
|
model,
|
2019-07-01 19:05:58 +08:00
|
|
|
new Vec3(pos_x, pos_y, pos_z),
|
|
|
|
new Vec3(rotation_x, rotation_y, rotation_z),
|
|
|
|
new Vec3(scale_x, scale_y, scale_z),
|
|
|
|
children
|
|
|
|
);
|
2019-06-30 05:18:52 +08:00
|
|
|
|
|
|
|
return [object, ...siblings];
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|