2019-05-29 00:40:29 +08:00
|
|
|
import {
|
|
|
|
BufferAttribute,
|
|
|
|
BufferGeometry,
|
|
|
|
Euler,
|
|
|
|
Matrix4,
|
|
|
|
Quaternion,
|
|
|
|
Vector3
|
|
|
|
} from 'three';
|
2019-06-26 23:21:05 +08:00
|
|
|
import { BufferCursor } from '../../BufferCursor';
|
2019-05-29 07:37:00 +08:00
|
|
|
import { parseNjModel, NjContext } from './nj';
|
|
|
|
import { parseXjModel, XjContext } from './xj';
|
2019-05-29 00:40:29 +08:00
|
|
|
|
|
|
|
// TODO:
|
|
|
|
// - deal with multiple NJCM chunks
|
|
|
|
// - deal with other types of chunks
|
|
|
|
|
2019-06-26 23:21:05 +08:00
|
|
|
export function parseNj(cursor: BufferCursor): BufferGeometry | undefined {
|
2019-05-29 07:37:00 +08:00
|
|
|
return parseNinja(cursor, 'nj');
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-26 23:21:05 +08:00
|
|
|
export function parseXj(cursor: BufferCursor): BufferGeometry | undefined {
|
2019-05-29 07:37:00 +08:00
|
|
|
return parseNinja(cursor, 'xj');
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type Format = 'nj' | 'xj';
|
|
|
|
type Context = NjContext | XjContext;
|
|
|
|
|
2019-06-26 23:21:05 +08:00
|
|
|
function parseNinja(cursor: BufferCursor, format: Format): BufferGeometry | undefined {
|
|
|
|
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-26 23:21:05 +08:00
|
|
|
const iffTypeId = cursor.string_ascii(4, false, false);
|
2019-05-29 07:37:00 +08:00
|
|
|
const iffChunkSize = cursor.u32();
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (iffTypeId === 'NJCM') {
|
|
|
|
return parseNjcm(cursor.take(iffChunkSize), format);
|
2019-05-29 00:40:29 +08:00
|
|
|
} else {
|
2019-05-29 07:37:00 +08:00
|
|
|
cursor.seek(iffChunkSize);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-26 23:21:05 +08:00
|
|
|
function parseNjcm(cursor: BufferCursor, format: Format): BufferGeometry | undefined {
|
|
|
|
if (cursor.bytes_left) {
|
2019-05-29 00:40:29 +08:00
|
|
|
let context: Context;
|
|
|
|
|
|
|
|
if (format === 'nj') {
|
|
|
|
context = {
|
|
|
|
format,
|
|
|
|
positions: [],
|
|
|
|
normals: [],
|
2019-05-29 07:37:00 +08:00
|
|
|
cachedChunkOffsets: [],
|
2019-05-29 00:40:29 +08:00
|
|
|
vertices: []
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
context = {
|
|
|
|
format,
|
|
|
|
positions: [],
|
|
|
|
normals: [],
|
|
|
|
indices: []
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
parseSiblingObjects(cursor, new Matrix4(), context);
|
|
|
|
return createBufferGeometry(context);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
function parseSiblingObjects(
|
2019-06-26 23:21:05 +08:00
|
|
|
cursor: BufferCursor,
|
2019-05-29 07:37:00 +08:00
|
|
|
parentMatrix: Matrix4,
|
2019-05-29 00:40:29 +08:00
|
|
|
context: Context
|
|
|
|
): void {
|
2019-05-29 07:37:00 +08:00
|
|
|
const evalFlags = cursor.u32();
|
|
|
|
const noTranslate = (evalFlags & 0b1) !== 0;
|
|
|
|
const noRotate = (evalFlags & 0b10) !== 0;
|
|
|
|
const noScale = (evalFlags & 0b100) !== 0;
|
|
|
|
const hidden = (evalFlags & 0b1000) !== 0;
|
|
|
|
const breakChildTrace = (evalFlags & 0b10000) !== 0;
|
|
|
|
const zxyRotationOrder = (evalFlags & 0b100000) !== 0;
|
|
|
|
|
|
|
|
const modelOffset = cursor.u32();
|
|
|
|
const posX = cursor.f32();
|
|
|
|
const posY = cursor.f32();
|
|
|
|
const posZ = cursor.f32();
|
|
|
|
const rotationX = cursor.i32() * (2 * Math.PI / 0xFFFF);
|
|
|
|
const rotationY = cursor.i32() * (2 * Math.PI / 0xFFFF);
|
|
|
|
const rotationZ = cursor.i32() * (2 * Math.PI / 0xFFFF);
|
|
|
|
const scaleX = cursor.f32();
|
|
|
|
const scaleY = cursor.f32();
|
|
|
|
const scaleZ = cursor.f32();
|
|
|
|
const childOffset = cursor.u32();
|
|
|
|
const siblingOffset = cursor.u32();
|
|
|
|
|
|
|
|
const rotation = new Euler(rotationX, rotationY, rotationZ, zxyRotationOrder ? 'ZXY' : 'ZYX');
|
2019-05-29 00:40:29 +08:00
|
|
|
const matrix = new Matrix4()
|
|
|
|
.compose(
|
2019-05-29 07:37:00 +08:00
|
|
|
noTranslate ? new Vector3() : new Vector3(posX, posY, posZ),
|
|
|
|
noRotate ? new Quaternion(0, 0, 0, 1) : new Quaternion().setFromEuler(rotation),
|
|
|
|
noScale ? new Vector3(1, 1, 1) : new Vector3(scaleX, scaleY, scaleZ)
|
2019-05-29 00:40:29 +08:00
|
|
|
)
|
2019-05-29 07:37:00 +08:00
|
|
|
.premultiply(parentMatrix);
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (modelOffset && !hidden) {
|
2019-06-26 23:21:05 +08:00
|
|
|
cursor.seek_start(modelOffset);
|
2019-05-29 07:37:00 +08:00
|
|
|
parseModel(cursor, matrix, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (childOffset && !breakChildTrace) {
|
2019-06-26 23:21:05 +08:00
|
|
|
cursor.seek_start(childOffset);
|
2019-05-29 07:37:00 +08:00
|
|
|
parseSiblingObjects(cursor, matrix, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (siblingOffset) {
|
2019-06-26 23:21:05 +08:00
|
|
|
cursor.seek_start(siblingOffset);
|
2019-05-29 07:37:00 +08:00
|
|
|
parseSiblingObjects(cursor, parentMatrix, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
function createBufferGeometry(context: Context): BufferGeometry {
|
2019-05-29 00:40:29 +08:00
|
|
|
const geometry = new BufferGeometry();
|
|
|
|
geometry.addAttribute('position', new BufferAttribute(new Float32Array(context.positions), 3));
|
|
|
|
geometry.addAttribute('normal', new BufferAttribute(new Float32Array(context.normals), 3));
|
|
|
|
|
|
|
|
if ('indices' in context) {
|
|
|
|
geometry.setIndex(new BufferAttribute(new Uint16Array(context.indices), 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
return geometry;
|
|
|
|
}
|
|
|
|
|
2019-06-26 23:21:05 +08:00
|
|
|
function parseModel(cursor: BufferCursor, matrix: Matrix4, context: Context): void {
|
2019-05-29 00:40:29 +08:00
|
|
|
if (context.format === 'nj') {
|
2019-05-29 07:37:00 +08:00
|
|
|
parseNjModel(cursor, matrix, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
} else {
|
2019-05-29 07:37:00 +08:00
|
|
|
parseXjModel(cursor, matrix, context);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|