phantasmal-world/src/data_formats/parsing/ninja/index.ts

219 lines
6.2 KiB
TypeScript
Raw Normal View History

2019-07-10 02:22:18 +08:00
import { Cursor } from "../../cursor/Cursor";
2019-07-07 03:55:02 +08:00
import { Vec3 } from "../../Vec3";
import { NjcmModel, parse_njcm_model } from "./njcm";
import { parse_xj_model, XjModel } from "./xj";
// TODO:
// - deal with multiple NJCM chunks
// - deal with other types of chunks
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-07 03:55:02 +08:00
export type NjVertex = {
position: Vec3;
normal?: Vec3;
bone_weight: number;
bone_weight_status: number;
calc_continue: boolean;
};
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;
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-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 {
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]);
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>,
bone_id: number,
id_ref: [number]
2019-07-07 03:55:02 +08:00
): NjObject<M> | undefined {
if (!object.evaluation_flags.skip) {
const id = id_ref[0]++;
2019-07-02 03:20:09 +08:00
this.bone_cache.set(id, object);
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-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;
};
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
}
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-07 03:55:02 +08:00
function parse_ninja<M extends NjModel>(
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>[] {
while (cursor.bytes_left) {
// Ninja uses a little endian variant of the IFF format.
// IFF files contain chunks preceded by an 8-byte header.
// The header consists of 4 ASCII characters for the "Type ID" and a 32-bit integer specifying the chunk size.
const iff_type_id = cursor.string_ascii(4, false, false);
const iff_chunk_size = cursor.u32();
if (iff_type_id === "NJCM") {
2019-06-30 05:18:52 +08:00
return parse_sibling_objects(cursor.take(iff_chunk_size), parse_model, context);
} else {
2019-07-01 21:01:40 +08:00
if (iff_chunk_size > cursor.bytes_left) {
break;
}
cursor.seek(iff_chunk_size);
}
}
2019-06-30 05:18:52 +08:00
return [];
}
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>(
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>[] {
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;
const skip = (eval_flags & 0b1000000) !== 0;
const shape_skip = (eval_flags & 0b10000000) !== 0;
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;
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-06-30 05:18:52 +08:00
if (model_offset) {
cursor.seek_start(model_offset);
2019-06-30 05:18:52 +08:00
model = parse_model(cursor, context);
}
2019-06-30 05:18:52 +08:00
if (child_offset) {
cursor.seek_start(child_offset);
2019-06-30 05:18:52 +08:00
children = parse_sibling_objects(cursor, parse_model, context);
} else {
children = [];
}
if (sibling_offset) {
cursor.seek_start(sibling_offset);
2019-06-30 05:18:52 +08:00
siblings = parse_sibling_objects(cursor, parse_model, context);
} else {
2019-06-30 05:18:52 +08:00
siblings = [];
}
2019-06-30 05:18:52 +08:00
2019-07-07 03:55:02 +08:00
const object = new NjObject<M>(
{
2019-06-30 05:18:52 +08:00
no_translate,
no_rotate,
no_scale,
hidden,
break_child_trace,
zxy_rotation_order,
skip,
shape_skip,
2019-06-30 05:18:52 +08:00
},
model,
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];
}