2019-06-26 23:21:05 +08:00
|
|
|
import { BufferCursor } from '../../BufferCursor';
|
2019-06-30 05:18:52 +08:00
|
|
|
import { parse_nj_model, NjModel } from './nj';
|
|
|
|
import { parse_xj_model, XjModel } from './xj';
|
|
|
|
import { Vec3 } from '../../../domain';
|
2019-05-29 00:40:29 +08:00
|
|
|
|
|
|
|
// TODO:
|
|
|
|
// - deal with multiple NJCM chunks
|
|
|
|
// - deal with other types of chunks
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
const ANGLE_TO_RAD = 2 * Math.PI / 65536;
|
|
|
|
|
|
|
|
export type NinjaVertex = {
|
|
|
|
position: Vec3,
|
|
|
|
normal?: Vec3,
|
2019-06-30 08:07:59 +08:00
|
|
|
bone_weight: number,
|
2019-07-01 01:55:30 +08:00
|
|
|
bone_weight_status: number,
|
|
|
|
calc_continue: boolean
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
export type NinjaModel = NjModel | XjModel;
|
|
|
|
|
|
|
|
export type NinjaObject<M extends NinjaModel> = {
|
|
|
|
evaluation_flags: {
|
|
|
|
no_translate: boolean,
|
|
|
|
no_rotate: boolean,
|
|
|
|
no_scale: boolean,
|
|
|
|
hidden: boolean,
|
|
|
|
break_child_trace: boolean,
|
|
|
|
zxy_rotation_order: boolean,
|
2019-07-01 05:55:28 +08:00
|
|
|
skip: boolean,
|
|
|
|
shape_skip: boolean,
|
2019-06-30 05:18:52 +08:00
|
|
|
},
|
|
|
|
model?: M,
|
|
|
|
position: Vec3,
|
|
|
|
rotation: Vec3, // Euler angles in radians.
|
|
|
|
scale: Vec3,
|
|
|
|
children: NinjaObject<M>[],
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
export function parse_nj(cursor: BufferCursor): NinjaObject<NjModel>[] {
|
|
|
|
return parse_ninja(cursor, parse_nj_model, []);
|
|
|
|
}
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
export function parse_xj(cursor: BufferCursor): NinjaObject<XjModel>[] {
|
|
|
|
return parse_ninja(cursor, parse_xj_model, undefined);
|
|
|
|
}
|
|
|
|
|
|
|
|
function parse_ninja<M extends NinjaModel>(
|
|
|
|
cursor: BufferCursor,
|
|
|
|
parse_model: (cursor: BufferCursor, context: any) => M,
|
|
|
|
context: any
|
|
|
|
): NinjaObject<M>[] {
|
2019-06-26 23:21:05 +08:00
|
|
|
while (cursor.bytes_left) {
|
2019-05-29 00:40:29 +08:00
|
|
|
// 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.
|
2019-06-28 00:50:22 +08:00
|
|
|
const iff_type_id = cursor.string_ascii(4, false, false);
|
|
|
|
const iff_chunk_size = cursor.u32();
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
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);
|
2019-05-29 00:40:29 +08:00
|
|
|
} else {
|
2019-06-28 00:50:22 +08:00
|
|
|
cursor.seek(iff_chunk_size);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-30 05:18:52 +08:00
|
|
|
return [];
|
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.
|
|
|
|
function parse_sibling_objects<M extends NinjaModel>(
|
2019-06-26 23:21:05 +08:00
|
|
|
cursor: BufferCursor,
|
2019-06-30 05:18:52 +08:00
|
|
|
parse_model: (cursor: BufferCursor, context: any) => M,
|
|
|
|
context: any
|
|
|
|
): NinjaObject<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;
|
|
|
|
let children: NinjaObject<M>[];
|
|
|
|
let siblings: NinjaObject<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
|
|
|
|
|
|
|
const object: NinjaObject<M> = {
|
|
|
|
evaluation_flags: {
|
|
|
|
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,
|
|
|
|
position: new Vec3(pos_x, pos_y, pos_z),
|
|
|
|
rotation: new Vec3(rotation_x, rotation_y, rotation_z),
|
|
|
|
scale: new Vec3(scale_x, scale_y, scale_z),
|
|
|
|
children,
|
|
|
|
};
|
|
|
|
|
|
|
|
return [object, ...siblings];
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|