2019-05-29 00:40:29 +08:00
|
|
|
import { Matrix3, Matrix4, Vector3 } from 'three';
|
|
|
|
import { ArrayBufferCursor } from '../../ArrayBufferCursor';
|
|
|
|
|
|
|
|
// TODO:
|
|
|
|
// - textures
|
|
|
|
// - colors
|
|
|
|
// - bump maps
|
|
|
|
// - animation
|
|
|
|
|
|
|
|
export interface XjContext {
|
|
|
|
format: 'xj';
|
|
|
|
positions: number[];
|
|
|
|
normals: number[];
|
|
|
|
indices: number[];
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
export function parseXjModel(cursor: ArrayBufferCursor, matrix: Matrix4, context: XjContext): void {
|
2019-05-29 00:40:29 +08:00
|
|
|
const { positions, normals, indices } = context;
|
|
|
|
|
|
|
|
cursor.seek(4); // Flags according to QEdit, seemingly always 0.
|
2019-05-29 07:37:00 +08:00
|
|
|
const vertexInfoListOffset = cursor.u32();
|
|
|
|
cursor.seek(4); // Seems to be the vertexInfoCount, always 1.
|
|
|
|
const triangleStripListAOffset = cursor.u32();
|
|
|
|
const triangleStripACount = cursor.u32();
|
|
|
|
const triangleStripListBOffset = cursor.u32();
|
|
|
|
const triangleStripBCount = cursor.u32();
|
2019-05-29 00:40:29 +08:00
|
|
|
cursor.seek(16); // Bounding sphere position and radius in floats.
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
const normalMatrix = new Matrix3().getNormalMatrix(matrix);
|
|
|
|
const indexOffset = positions.length / 3;
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (vertexInfoListOffset) {
|
|
|
|
cursor.seekStart(vertexInfoListOffset);
|
2019-05-29 00:40:29 +08:00
|
|
|
cursor.seek(4); // Possibly the vertex type.
|
2019-05-29 07:37:00 +08:00
|
|
|
const vertexListOffset = cursor.u32();
|
|
|
|
const vertexSize = cursor.u32();
|
|
|
|
const vertexCount = cursor.u32();
|
2019-05-29 00:40:29 +08:00
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
for (let i = 0; i < vertexCount; ++i) {
|
|
|
|
cursor.seekStart(vertexListOffset + i * vertexSize);
|
2019-05-29 00:40:29 +08:00
|
|
|
const position = new Vector3(
|
|
|
|
cursor.f32(),
|
|
|
|
cursor.f32(),
|
|
|
|
cursor.f32()
|
|
|
|
).applyMatrix4(matrix);
|
|
|
|
let normal;
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (vertexSize === 28 || vertexSize === 32 || vertexSize === 36) {
|
2019-05-29 00:40:29 +08:00
|
|
|
normal = new Vector3(
|
|
|
|
cursor.f32(),
|
|
|
|
cursor.f32(),
|
|
|
|
cursor.f32()
|
2019-05-29 07:37:00 +08:00
|
|
|
).applyMatrix3(normalMatrix);
|
2019-05-29 00:40:29 +08:00
|
|
|
} else {
|
|
|
|
normal = new Vector3(0, 1, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
positions.push(position.x);
|
|
|
|
positions.push(position.y);
|
|
|
|
positions.push(position.z);
|
|
|
|
normals.push(normal.x);
|
|
|
|
normals.push(normal.y);
|
|
|
|
normals.push(normal.z);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (triangleStripListAOffset) {
|
|
|
|
parseTriangleStripList(
|
2019-05-29 00:40:29 +08:00
|
|
|
cursor,
|
2019-05-29 07:37:00 +08:00
|
|
|
triangleStripListAOffset,
|
|
|
|
triangleStripACount,
|
2019-05-29 00:40:29 +08:00
|
|
|
positions,
|
|
|
|
normals,
|
|
|
|
indices,
|
2019-05-29 07:37:00 +08:00
|
|
|
indexOffset
|
|
|
|
);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (triangleStripListBOffset) {
|
|
|
|
parseTriangleStripList(
|
2019-05-29 00:40:29 +08:00
|
|
|
cursor,
|
2019-05-29 07:37:00 +08:00
|
|
|
triangleStripListBOffset,
|
|
|
|
triangleStripBCount,
|
2019-05-29 00:40:29 +08:00
|
|
|
positions,
|
|
|
|
normals,
|
|
|
|
indices,
|
2019-05-29 07:37:00 +08:00
|
|
|
indexOffset
|
|
|
|
);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
function parseTriangleStripList(
|
2019-05-29 00:40:29 +08:00
|
|
|
cursor: ArrayBufferCursor,
|
2019-05-29 07:37:00 +08:00
|
|
|
triangleStripListOffset: number,
|
|
|
|
triangleStripCount: number,
|
2019-05-29 00:40:29 +08:00
|
|
|
positions: number[],
|
|
|
|
normals: number[],
|
|
|
|
indices: number[],
|
2019-05-29 07:37:00 +08:00
|
|
|
indexOffset: number
|
2019-05-29 00:40:29 +08:00
|
|
|
): void {
|
2019-05-29 07:37:00 +08:00
|
|
|
for (let i = 0; i < triangleStripCount; ++i) {
|
|
|
|
cursor.seekStart(triangleStripListOffset + i * 20);
|
2019-05-29 00:40:29 +08:00
|
|
|
cursor.seek(8); // Skip material information.
|
2019-05-29 07:37:00 +08:00
|
|
|
const indexListOffset = cursor.u32();
|
|
|
|
const indexCount = cursor.u32();
|
2019-05-29 00:40:29 +08:00
|
|
|
// Ignoring 4 bytes.
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
cursor.seekStart(indexListOffset);
|
|
|
|
const stripIndices = cursor.u16Array(indexCount);
|
2019-05-29 00:40:29 +08:00
|
|
|
let clockwise = true;
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
for (let j = 2; j < stripIndices.length; ++j) {
|
|
|
|
const a = indexOffset + stripIndices[j - 2];
|
|
|
|
const b = indexOffset + stripIndices[j - 1];
|
|
|
|
const c = indexOffset + stripIndices[j];
|
2019-05-29 00:40:29 +08:00
|
|
|
const pa = new Vector3(positions[3 * a], positions[3 * a + 1], positions[3 * a + 2]);
|
|
|
|
const pb = new Vector3(positions[3 * b], positions[3 * b + 1], positions[3 * b + 2]);
|
|
|
|
const pc = new Vector3(positions[3 * c], positions[3 * c + 1], positions[3 * c + 2]);
|
|
|
|
const na = new Vector3(normals[3 * a], normals[3 * a + 1], normals[3 * a + 2]);
|
|
|
|
const nb = new Vector3(normals[3 * a], normals[3 * a + 1], normals[3 * a + 2]);
|
|
|
|
const nc = new Vector3(normals[3 * a], normals[3 * a + 1], normals[3 * a + 2]);
|
|
|
|
|
|
|
|
// Calculate a surface normal and reverse the vertex winding if at least 2 of the vertex normals point in the opposite direction.
|
|
|
|
// This hack fixes the winding for most models.
|
|
|
|
const normal = pb.clone().sub(pa).cross(pc.clone().sub(pa));
|
|
|
|
|
|
|
|
if (clockwise) {
|
|
|
|
normal.negate();
|
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
const oppositeCount =
|
2019-05-29 00:40:29 +08:00
|
|
|
(normal.dot(na) < 0 ? 1 : 0) +
|
|
|
|
(normal.dot(nb) < 0 ? 1 : 0) +
|
|
|
|
(normal.dot(nc) < 0 ? 1 : 0);
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
if (oppositeCount >= 2) {
|
2019-05-29 00:40:29 +08:00
|
|
|
clockwise = !clockwise;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clockwise) {
|
|
|
|
indices.push(b);
|
|
|
|
indices.push(a);
|
|
|
|
indices.push(c);
|
|
|
|
} else {
|
|
|
|
indices.push(a);
|
|
|
|
indices.push(b);
|
|
|
|
indices.push(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
clockwise = !clockwise;
|
|
|
|
|
|
|
|
// The following switch statement fixes model 180.xj (zanba).
|
|
|
|
// switch (j) {
|
|
|
|
// case 17:
|
|
|
|
// case 52:
|
|
|
|
// case 70:
|
|
|
|
// case 92:
|
|
|
|
// case 97:
|
|
|
|
// case 126:
|
|
|
|
// case 140:
|
|
|
|
// case 148:
|
|
|
|
// case 187:
|
|
|
|
// case 200:
|
|
|
|
// console.warn(`swapping winding at: ${j}, (${a}, ${b}, ${c})`);
|
|
|
|
// break;
|
|
|
|
// default:
|
|
|
|
// ccw = !ccw;
|
|
|
|
// break;
|
|
|
|
// }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|