Extracting more data from njm files.

This commit is contained in:
Daan Vanden Bosch 2019-06-29 19:09:38 +02:00
parent de3de9256b
commit a36e177eef

View File

@ -1,7 +1,7 @@
import Logger from 'js-logger';
import { BufferCursor } from '../../BufferCursor'; import { BufferCursor } from '../../BufferCursor';
import { Vec3 } from '../../../domain';
const logger = Logger.get('bin_data/parsing/ninja/njm2'); const ANGLE_TO_RAD = 2 * Math.PI / 65536;
export type NjAction = { export type NjAction = {
object_offset: number, object_offset: number,
@ -17,7 +17,7 @@ export type NjMotion = {
} }
export enum NjInterpolation { export enum NjInterpolation {
Linear, Spline, UserFunction, SamplingMask Linear, Spline, UserFunction
} }
export type NjMotionData = { export type NjMotionData = {
@ -53,7 +53,7 @@ export type NjKeyframe = NjKeyframeF | NjKeyframeA
*/ */
export type NjKeyframeF = { export type NjKeyframeF = {
frame: number, frame: number,
value: [number, number, number], value: Vec3,
} }
/** /**
@ -61,7 +61,7 @@ export type NjKeyframeF = {
*/ */
export type NjKeyframeA = { export type NjKeyframeA = {
frame: number, frame: number,
value: [number, number, number], value: Vec3, // Euler angles in radians.
} }
/** /**
@ -70,10 +70,8 @@ export type NjKeyframeA = {
export function parse_njm_4(cursor: BufferCursor): NjAction { export function parse_njm_4(cursor: BufferCursor): NjAction {
cursor.seek_end(16); cursor.seek_end(16);
const offset1 = cursor.u32(); const offset1 = cursor.u32();
log_offset('offset1', offset1);
cursor.seek_start(offset1); cursor.seek_start(offset1);
const action_offset = cursor.u32(); const action_offset = cursor.u32();
log_offset('action_offset', action_offset);
cursor.seek_start(action_offset); cursor.seek_start(action_offset);
return parse_action(cursor); return parse_action(cursor);
} }
@ -81,8 +79,6 @@ export function parse_njm_4(cursor: BufferCursor): NjAction {
function parse_action(cursor: BufferCursor): NjAction { function parse_action(cursor: BufferCursor): NjAction {
const object_offset = cursor.u32(); const object_offset = cursor.u32();
const motion_offset = cursor.u32(); const motion_offset = cursor.u32();
log_offset('object offset', object_offset);
log_offset('motion offset', motion_offset);
cursor.seek_start(motion_offset); cursor.seek_start(motion_offset);
const motion = parse_motion(cursor); const motion = parse_motion(cursor);
@ -92,31 +88,41 @@ function parse_action(cursor: BufferCursor): NjAction {
}; };
} }
// TODO: parse data for all objects.
function parse_motion(cursor: BufferCursor): NjMotion { function parse_motion(cursor: BufferCursor): NjMotion {
const motion_offset = cursor.position;
// Points to an array the size of the total amount of objects in the object tree. // Points to an array the size of the total amount of objects in the object tree.
const mdata_offset = cursor.u32(); let mdata_offset = cursor.u32();
const frame_count = cursor.u32(); const frame_count = cursor.u32();
const type = cursor.u16(); const type = cursor.u16();
const inp_fn = cursor.u16(); const inp_fn = cursor.u16();
// Linear, spline, user function or sampling mask. // Linear, spline or user function.
const interpolation: NjInterpolation = (inp_fn & 0b11000000) >> 6; const interpolation: NjInterpolation = (inp_fn & 0b11000000) >> 6;
const element_count = inp_fn & 0b1111; const element_count = inp_fn & 0b1111;
const motion_data_list = [];
// The mdata array stops where the motion structure starts.
while (true) {
cursor.seek_start(mdata_offset);
if (cursor.position >= motion_offset) {
break;
}
mdata_offset = mdata_offset += 8 * element_count;
let motion_data: NjMotionData = { let motion_data: NjMotionData = {
tracks: [], tracks: [],
}; };
motion_data_list.push(motion_data);
const size = count_set_bits(type);
cursor.seek_start(mdata_offset);
const keyframe_offsets: number[] = []; const keyframe_offsets: number[] = [];
const keyframe_counts: number[] = []; const keyframe_counts: number[] = [];
for (let i = 0; i < size; i++) { for (let i = 0; i < element_count; i++) {
keyframe_offsets.push(cursor.u32()); keyframe_offsets.push(cursor.u32());
} }
for (let i = 0; i < size; i++) { for (let i = 0; i < element_count; i++) {
const count = cursor.u32(); const count = cursor.u32();
keyframe_counts.push(count); keyframe_counts.push(count);
} }
@ -124,29 +130,41 @@ function parse_motion(cursor: BufferCursor): NjMotion {
// NJD_MTYPE_POS_0 // NJD_MTYPE_POS_0
if ((type & (1 << 0)) !== 0) { if ((type & (1 << 0)) !== 0) {
cursor.seek_start(keyframe_offsets.shift()!); cursor.seek_start(keyframe_offsets.shift()!);
const count = keyframe_counts.shift();
if (count) {
motion_data.tracks.push({ motion_data.tracks.push({
type: NjKeyframeTrackType.Position, type: NjKeyframeTrackType.Position,
keyframes: parse_motion_data_f(cursor, keyframe_counts.shift()!) keyframes: parse_motion_data_f(cursor, count)
}); });
} }
}
// NJD_MTYPE_ANG_1 // NJD_MTYPE_ANG_1
if ((type & (1 << 1)) !== 0) { if ((type & (1 << 1)) !== 0) {
cursor.seek_start(keyframe_offsets.shift()!); cursor.seek_start(keyframe_offsets.shift()!);
const count = keyframe_counts.shift();
if (count) {
motion_data.tracks.push({ motion_data.tracks.push({
type: NjKeyframeTrackType.Rotation, type: NjKeyframeTrackType.Rotation,
keyframes: parse_motion_data_a(cursor, keyframe_counts.shift()!) keyframes: parse_motion_data_a(cursor, count)
}); });
} }
}
// NJD_MTYPE_SCL_2 // NJD_MTYPE_SCL_2
if ((type & (1 << 2)) !== 0) { if ((type & (1 << 2)) !== 0) {
cursor.seek_start(keyframe_offsets.shift()!); cursor.seek_start(keyframe_offsets.shift()!);
const count = keyframe_counts.shift();
if (count) {
motion_data.tracks.push({ motion_data.tracks.push({
type: NjKeyframeTrackType.Scale, type: NjKeyframeTrackType.Scale,
keyframes: parse_motion_data_f(cursor, keyframe_counts.shift()!) keyframes: parse_motion_data_f(cursor, count)
}); });
} }
}
// // NJD_MTYPE_VEC_3 // // NJD_MTYPE_VEC_3
// if ((type & (1 << 3)) !== 0) { // if ((type & (1 << 3)) !== 0) {
@ -165,9 +183,10 @@ function parse_motion(cursor: BufferCursor): NjMotion {
// } // }
// TODO: all NJD_MTYPE's // TODO: all NJD_MTYPE's
}
return { return {
motion_data: [motion_data], motion_data: motion_data_list,
frame_count, frame_count,
type, type,
interpolation, interpolation,
@ -181,7 +200,7 @@ function parse_motion_data_f(cursor: BufferCursor, count: number): NjKeyframeF[]
for (let i = 0; i < count; ++i) { for (let i = 0; i < count; ++i) {
frames.push({ frames.push({
frame: cursor.u32(), frame: cursor.u32(),
value: [cursor.f32(), cursor.f32(), cursor.f32()], value: new Vec3(cursor.f32(), cursor.f32(), cursor.f32()),
}); });
} }
@ -194,24 +213,13 @@ function parse_motion_data_a(cursor: BufferCursor, count: number): NjKeyframeA[]
for (let i = 0; i < count; ++i) { for (let i = 0; i < count; ++i) {
frames.push({ frames.push({
frame: cursor.u16(), frame: cursor.u16(),
value: [cursor.i16(), cursor.i16(), cursor.i16()], value: new Vec3(
cursor.u16() * ANGLE_TO_RAD,
cursor.u16() * ANGLE_TO_RAD,
cursor.u16() * ANGLE_TO_RAD
),
}); });
} }
return frames; return frames;
} }
function log_offset(name: string, offset: number) {
logger.debug(`${name}: 0x${offset.toString(16).toUpperCase()}`);
}
function count_set_bits(n: number): number {
let count = 0;
while (n) {
count += n & 1;
n >>= 1;
}
return count;
}