mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Ninja render geometry is now parsed correctly.
This commit is contained in:
parent
8c21ea59c9
commit
a181847647
@ -1,7 +1,7 @@
|
|||||||
import { Cursor } from "../cursor/Cursor";
|
import { Cursor } from "../cursor/Cursor";
|
||||||
import { Vec3 } from "../vector";
|
import { Vec3 } from "../vector";
|
||||||
import { ANGLE_TO_RAD } from "./ninja";
|
import { ANGLE_TO_RAD, NjObject, parse_xj_object } from "./ninja";
|
||||||
import { parse_xj_model, XjModel } from "./ninja/xj";
|
import { XjModel } from "./ninja/xj";
|
||||||
import { parse_rel } from "./rel";
|
import { parse_rel } from "./rel";
|
||||||
|
|
||||||
export type RenderObject = {
|
export type RenderObject = {
|
||||||
@ -12,7 +12,7 @@ export type RenderSection = {
|
|||||||
id: number;
|
id: number;
|
||||||
position: Vec3;
|
position: Vec3;
|
||||||
rotation: Vec3;
|
rotation: Vec3;
|
||||||
models: XjModel[];
|
objects: NjObject<XjModel>[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Vertex = {
|
export type Vertex = {
|
||||||
@ -53,7 +53,7 @@ export function parse_area_geometry(cursor: Cursor): RenderObject {
|
|||||||
// const animated_geometry_offset_count = cursor.u32();
|
// const animated_geometry_offset_count = cursor.u32();
|
||||||
// Ignore animated_geometry_offset_count and the last 4 bytes.
|
// Ignore animated_geometry_offset_count and the last 4 bytes.
|
||||||
|
|
||||||
const models = parse_geometry_table(
|
const objects = parse_geometry_table(
|
||||||
cursor,
|
cursor,
|
||||||
simple_geometry_offset_table_offset,
|
simple_geometry_offset_table_offset,
|
||||||
simple_geometry_offset_count
|
simple_geometry_offset_count
|
||||||
@ -63,19 +63,20 @@ export function parse_area_geometry(cursor: Cursor): RenderObject {
|
|||||||
id: section_id,
|
id: section_id,
|
||||||
position: section_position,
|
position: section_position,
|
||||||
rotation: section_rotation,
|
rotation: section_rotation,
|
||||||
models,
|
objects,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return { sections };
|
return { sections };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: don't reparse the same objects multiple times. Create DAG instead of tree.
|
||||||
function parse_geometry_table(
|
function parse_geometry_table(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
table_offset: number,
|
table_offset: number,
|
||||||
table_entry_count: number
|
table_entry_count: number
|
||||||
): XjModel[] {
|
): NjObject<XjModel>[] {
|
||||||
const models: XjModel[] = [];
|
const objects: NjObject<XjModel>[] = [];
|
||||||
|
|
||||||
for (let i = 0; i < table_entry_count; i++) {
|
for (let i = 0; i < table_entry_count; i++) {
|
||||||
cursor.seek_start(table_offset + 16 * i);
|
cursor.seek_start(table_offset + 16 * i);
|
||||||
@ -88,14 +89,9 @@ function parse_geometry_table(
|
|||||||
offset = cursor.seek_start(offset).u32();
|
offset = cursor.seek_start(offset).u32();
|
||||||
}
|
}
|
||||||
|
|
||||||
cursor.seek_start(offset + 4);
|
cursor.seek_start(offset);
|
||||||
const geometry_offset = cursor.u32();
|
objects.push(...parse_xj_object(cursor));
|
||||||
|
|
||||||
if (geometry_offset > 0) {
|
|
||||||
cursor.seek_start(geometry_offset);
|
|
||||||
models.push(parse_xj_model(cursor));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return models;
|
return objects;
|
||||||
}
|
}
|
||||||
|
@ -101,14 +101,27 @@ export type NjEvaluationFlags = {
|
|||||||
shape_skip: boolean;
|
shape_skip: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses an NJCM file.
|
||||||
|
*/
|
||||||
export function parse_nj(cursor: Cursor): NjObject<NjcmModel>[] {
|
export function parse_nj(cursor: Cursor): NjObject<NjcmModel>[] {
|
||||||
return parse_ninja(cursor, parse_njcm_model, []);
|
return parse_ninja(cursor, parse_njcm_model, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses an NJCM file.
|
||||||
|
*/
|
||||||
export function parse_xj(cursor: Cursor): NjObject<XjModel>[] {
|
export function parse_xj(cursor: Cursor): NjObject<XjModel>[] {
|
||||||
return parse_ninja(cursor, parse_xj_model, undefined);
|
return parse_ninja(cursor, parse_xj_model, undefined);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a ninja object.
|
||||||
|
*/
|
||||||
|
export function parse_xj_object(cursor: Cursor): NjObject<XjModel>[] {
|
||||||
|
return parse_sibling_objects(cursor, parse_xj_model, undefined);
|
||||||
|
}
|
||||||
|
|
||||||
function parse_ninja<M extends NjModel>(
|
function parse_ninja<M extends NjModel>(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
parse_model: (cursor: Cursor, context: any) => M,
|
parse_model: (cursor: Cursor, context: any) => M,
|
||||||
|
115
src/rendering/conversion/GeometryBuilder.ts
Normal file
115
src/rendering/conversion/GeometryBuilder.ts
Normal file
@ -0,0 +1,115 @@
|
|||||||
|
import { BufferGeometry, Float32BufferAttribute, Uint16BufferAttribute, Vector3 } from "three";
|
||||||
|
|
||||||
|
export type BuilderVec2 = {
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type BuilderVec3 = {
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
z: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
type VertexGroup = {
|
||||||
|
offset: number;
|
||||||
|
size: number;
|
||||||
|
material_index: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export class GeometryBuilder {
|
||||||
|
private positions: number[] = [];
|
||||||
|
private normals: number[] = [];
|
||||||
|
private uvs: number[] = [];
|
||||||
|
private indices: number[] = [];
|
||||||
|
private bone_indices: number[] = [];
|
||||||
|
private bone_weights: number[] = [];
|
||||||
|
private groups: VertexGroup[] = [];
|
||||||
|
private _max_material_index?: number;
|
||||||
|
|
||||||
|
get vertex_count(): number {
|
||||||
|
return this.positions.length / 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
get index_count(): number {
|
||||||
|
return this.indices.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
get max_material_index(): number | undefined {
|
||||||
|
return this._max_material_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
get_position(index: number): Vector3 {
|
||||||
|
return new Vector3(
|
||||||
|
this.positions[3 * index],
|
||||||
|
this.positions[3 * index + 1],
|
||||||
|
this.positions[3 * index + 2]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
get_normal(index: number): Vector3 {
|
||||||
|
return new Vector3(
|
||||||
|
this.normals[3 * index],
|
||||||
|
this.normals[3 * index + 1],
|
||||||
|
this.normals[3 * index + 2]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
add_vertex(position: BuilderVec3, normal: BuilderVec3, uv: BuilderVec2): void {
|
||||||
|
this.positions.push(position.x, position.y, position.z);
|
||||||
|
this.normals.push(normal.x, normal.y, normal.z);
|
||||||
|
this.uvs.push(uv.x, uv.y);
|
||||||
|
}
|
||||||
|
|
||||||
|
add_index(index: number): void {
|
||||||
|
this.indices.push(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
add_bone(index: number, weight: number): void {
|
||||||
|
this.bone_indices.push(index);
|
||||||
|
this.bone_weights.push(weight);
|
||||||
|
}
|
||||||
|
|
||||||
|
add_group(offset: number, size: number, material_id?: number): void {
|
||||||
|
const last_group = this.groups[this.groups.length - 1];
|
||||||
|
const mat_idx = material_id == null ? 0 : material_id + 1;
|
||||||
|
|
||||||
|
if (last_group && last_group.material_index === mat_idx) {
|
||||||
|
last_group.size += size;
|
||||||
|
} else {
|
||||||
|
this.groups.push({
|
||||||
|
offset,
|
||||||
|
size,
|
||||||
|
material_index: mat_idx,
|
||||||
|
});
|
||||||
|
|
||||||
|
this._max_material_index = this._max_material_index
|
||||||
|
? Math.max(this._max_material_index, mat_idx)
|
||||||
|
: mat_idx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
build(): BufferGeometry {
|
||||||
|
const geom = new BufferGeometry();
|
||||||
|
|
||||||
|
geom.addAttribute("position", new Float32BufferAttribute(this.positions, 3));
|
||||||
|
geom.addAttribute("normal", new Float32BufferAttribute(this.normals, 3));
|
||||||
|
geom.addAttribute("uv", new Float32BufferAttribute(this.uvs, 2));
|
||||||
|
|
||||||
|
geom.setIndex(new Uint16BufferAttribute(this.indices, 1));
|
||||||
|
|
||||||
|
for (const group of this.groups) {
|
||||||
|
geom.addGroup(group.offset, group.size, group.material_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.bone_indices.length) {
|
||||||
|
geom.addAttribute("skinIndex", new Uint16BufferAttribute(this.bone_indices, 4));
|
||||||
|
geom.addAttribute("skinWeight", new Float32BufferAttribute(this.bone_weights, 4));
|
||||||
|
}
|
||||||
|
|
||||||
|
geom.computeBoundingSphere();
|
||||||
|
geom.computeBoundingBox();
|
||||||
|
|
||||||
|
return geom;
|
||||||
|
}
|
||||||
|
}
|
@ -1,22 +1,19 @@
|
|||||||
import {
|
import {
|
||||||
BufferGeometry,
|
|
||||||
DoubleSide,
|
DoubleSide,
|
||||||
Face3,
|
Face3,
|
||||||
Float32BufferAttribute,
|
|
||||||
Geometry,
|
Geometry,
|
||||||
Group,
|
Group,
|
||||||
Matrix4,
|
|
||||||
Mesh,
|
Mesh,
|
||||||
MeshBasicMaterial,
|
MeshBasicMaterial,
|
||||||
MeshLambertMaterial,
|
MeshLambertMaterial,
|
||||||
Object3D,
|
Object3D,
|
||||||
Uint16BufferAttribute,
|
|
||||||
Vector3,
|
Vector3,
|
||||||
} from "three";
|
} from "three";
|
||||||
import { CollisionObject } from "../../data_formats/parsing/area_collision_geometry";
|
import { CollisionObject } from "../../data_formats/parsing/area_collision_geometry";
|
||||||
import { RenderObject } from "../../data_formats/parsing/area_geometry";
|
import { RenderObject } from "../../data_formats/parsing/area_geometry";
|
||||||
import { Section } from "../../domain";
|
import { Section } from "../../domain";
|
||||||
import { xj_model_to_geometry } from "./xj_models";
|
import { ninja_object_to_mesh, ninja_object_to_geometry_builder } from "./ninja_geometry";
|
||||||
|
import { GeometryBuilder } from "./GeometryBuilder";
|
||||||
|
|
||||||
const materials = [
|
const materials = [
|
||||||
// Wall
|
// Wall
|
||||||
@ -120,21 +117,17 @@ export function area_geometry_to_sections_and_object_3d(
|
|||||||
const group = new Group();
|
const group = new Group();
|
||||||
|
|
||||||
for (const section of object.sections) {
|
for (const section of object.sections) {
|
||||||
const positions: number[] = [];
|
const sec = new Section(section.id, section.position, section.rotation.y);
|
||||||
const normals: number[] = [];
|
sections.push(sec);
|
||||||
const indices: number[] = [];
|
|
||||||
|
|
||||||
for (const model of section.models) {
|
const builder = new GeometryBuilder();
|
||||||
xj_model_to_geometry(model, new Matrix4(), positions, normals, [], indices, []);
|
|
||||||
|
for (const object of section.objects) {
|
||||||
|
ninja_object_to_geometry_builder(object, builder);
|
||||||
}
|
}
|
||||||
|
|
||||||
const geometry = new BufferGeometry();
|
|
||||||
geometry.addAttribute("position", new Float32BufferAttribute(positions, 3));
|
|
||||||
geometry.addAttribute("normal", new Float32BufferAttribute(normals, 3));
|
|
||||||
geometry.setIndex(new Uint16BufferAttribute(indices, 1));
|
|
||||||
|
|
||||||
const mesh = new Mesh(
|
const mesh = new Mesh(
|
||||||
geometry,
|
builder.build(),
|
||||||
new MeshLambertMaterial({
|
new MeshLambertMaterial({
|
||||||
color: 0x44aaff,
|
color: 0x44aaff,
|
||||||
transparent: true,
|
transparent: true,
|
||||||
@ -142,13 +135,9 @@ export function area_geometry_to_sections_and_object_3d(
|
|||||||
side: DoubleSide,
|
side: DoubleSide,
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
mesh.position.set(section.position.x, section.position.y, section.position.z);
|
|
||||||
mesh.rotation.set(section.rotation.x, section.rotation.y, section.rotation.z);
|
|
||||||
group.add(mesh);
|
|
||||||
|
|
||||||
const sec = new Section(section.id, section.position, section.rotation.y);
|
|
||||||
(mesh.userData as AreaUserData).section = sec;
|
(mesh.userData as AreaUserData).section = sec;
|
||||||
sections.push(sec);
|
group.add(mesh);
|
||||||
}
|
}
|
||||||
|
|
||||||
return [sections, group];
|
return [sections, group];
|
||||||
|
@ -3,23 +3,23 @@ import {
|
|||||||
BufferGeometry,
|
BufferGeometry,
|
||||||
DoubleSide,
|
DoubleSide,
|
||||||
Euler,
|
Euler,
|
||||||
Float32BufferAttribute,
|
|
||||||
Material,
|
Material,
|
||||||
Matrix3,
|
Matrix3,
|
||||||
Matrix4,
|
Matrix4,
|
||||||
|
Mesh,
|
||||||
|
MeshBasicMaterial,
|
||||||
MeshLambertMaterial,
|
MeshLambertMaterial,
|
||||||
Quaternion,
|
Quaternion,
|
||||||
Skeleton,
|
Skeleton,
|
||||||
SkinnedMesh,
|
SkinnedMesh,
|
||||||
Uint16BufferAttribute,
|
Vector2,
|
||||||
Vector3,
|
Vector3,
|
||||||
MeshBasicMaterial,
|
|
||||||
Mesh,
|
|
||||||
} from "three";
|
} from "three";
|
||||||
import { vec3_to_threejs } from ".";
|
import { vec3_to_threejs } from ".";
|
||||||
import { is_njcm_model, NjModel, NjObject } from "../../data_formats/parsing/ninja";
|
import { is_njcm_model, NjModel, NjObject } from "../../data_formats/parsing/ninja";
|
||||||
import { NjcmModel } from "../../data_formats/parsing/ninja/njcm";
|
import { NjcmModel } from "../../data_formats/parsing/ninja/njcm";
|
||||||
import { xj_model_to_geometry } from "./xj_models";
|
import { XjModel } from "../../data_formats/parsing/ninja/xj";
|
||||||
|
import { GeometryBuilder } from "./GeometryBuilder";
|
||||||
|
|
||||||
const DUMMY_MATERIAL = new MeshBasicMaterial({
|
const DUMMY_MATERIAL = new MeshBasicMaterial({
|
||||||
color: 0x00ff00,
|
color: 0x00ff00,
|
||||||
@ -35,34 +35,42 @@ const DEFAULT_SKINNED_MATERIAL = new MeshLambertMaterial({
|
|||||||
side: DoubleSide,
|
side: DoubleSide,
|
||||||
});
|
});
|
||||||
const DEFAULT_NORMAL = new Vector3(0, 1, 0);
|
const DEFAULT_NORMAL = new Vector3(0, 1, 0);
|
||||||
|
const DEFAULT_UV = new Vector2(0, 0);
|
||||||
const NO_TRANSLATION = new Vector3(0, 0, 0);
|
const NO_TRANSLATION = new Vector3(0, 0, 0);
|
||||||
const NO_ROTATION = new Quaternion(0, 0, 0, 1);
|
const NO_ROTATION = new Quaternion(0, 0, 0, 1);
|
||||||
const NO_SCALE = new Vector3(1, 1, 1);
|
const NO_SCALE = new Vector3(1, 1, 1);
|
||||||
|
|
||||||
export function ninja_object_to_buffer_geometry(object: NjObject<NjModel>): BufferGeometry {
|
export function ninja_object_to_geometry_builder(
|
||||||
return new Object3DCreator([]).create_buffer_geometry(object);
|
object: NjObject<NjModel>,
|
||||||
|
builder: GeometryBuilder
|
||||||
|
) {
|
||||||
|
new ModelCreator(builder).to_geometry_builder(object);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ninja_object_to_mesh(object: NjObject<NjModel>, materials: Material[] = []): Mesh {
|
export function ninja_object_to_buffer_geometry(object: NjObject<NjModel>): BufferGeometry {
|
||||||
return new Object3DCreator(materials).create_mesh(object);
|
return new ModelCreator(new GeometryBuilder()).create_buffer_geometry(object);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ninja_object_to_mesh(
|
||||||
|
object: NjObject<NjModel>,
|
||||||
|
materials: Material[] = [],
|
||||||
|
default_material: Material = DEFAULT_MATERIAL
|
||||||
|
): Mesh {
|
||||||
|
return new ModelCreator(new GeometryBuilder()).create_mesh(object, materials, default_material);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ninja_object_to_skinned_mesh(
|
export function ninja_object_to_skinned_mesh(
|
||||||
object: NjObject<NjModel>,
|
object: NjObject<NjModel>,
|
||||||
materials: Material[] = []
|
materials: Material[] = [],
|
||||||
|
default_material: Material = DEFAULT_SKINNED_MATERIAL
|
||||||
): SkinnedMesh {
|
): SkinnedMesh {
|
||||||
return new Object3DCreator(materials).create_skinned_mesh(object);
|
return new ModelCreator(new GeometryBuilder()).create_skinned_mesh(
|
||||||
|
object,
|
||||||
|
materials,
|
||||||
|
default_material
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export type VertexGroup = {
|
|
||||||
/**
|
|
||||||
* Start index.
|
|
||||||
*/
|
|
||||||
start: number;
|
|
||||||
count: number;
|
|
||||||
material_index: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Vertex = {
|
type Vertex = {
|
||||||
bone_id: number;
|
bone_id: number;
|
||||||
position: Vector3;
|
position: Vector3;
|
||||||
@ -94,70 +102,57 @@ class VerticesHolder {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class Object3DCreator {
|
class ModelCreator {
|
||||||
private materials: Material[];
|
|
||||||
private vertices = new VerticesHolder();
|
private vertices = new VerticesHolder();
|
||||||
private bone_id: number = 0;
|
private bone_id: number = 0;
|
||||||
private bones: Bone[] = [];
|
private bones: Bone[] = [];
|
||||||
|
private builder: GeometryBuilder;
|
||||||
|
|
||||||
private positions: number[] = [];
|
constructor(builder: GeometryBuilder) {
|
||||||
private normals: number[] = [];
|
this.builder = builder;
|
||||||
private uvs: number[] = [];
|
}
|
||||||
private indices: number[] = [];
|
|
||||||
private bone_indices: number[] = [];
|
|
||||||
private bone_weights: number[] = [];
|
|
||||||
private groups: VertexGroup[] = [];
|
|
||||||
|
|
||||||
constructor(materials: Material[]) {
|
to_geometry_builder(object: NjObject<NjModel>) {
|
||||||
this.materials = [DUMMY_MATERIAL, ...materials];
|
this.object_to_geometry(object, undefined, new Matrix4());
|
||||||
}
|
}
|
||||||
|
|
||||||
create_buffer_geometry(object: NjObject<NjModel>): BufferGeometry {
|
create_buffer_geometry(object: NjObject<NjModel>): BufferGeometry {
|
||||||
this.object_to_geometry(object, undefined, new Matrix4());
|
this.to_geometry_builder(object);
|
||||||
|
return this.builder.build();
|
||||||
const geom = new BufferGeometry();
|
|
||||||
|
|
||||||
geom.addAttribute("position", new Float32BufferAttribute(this.positions, 3));
|
|
||||||
geom.addAttribute("normal", new Float32BufferAttribute(this.normals, 3));
|
|
||||||
geom.addAttribute("uv", new Float32BufferAttribute(this.uvs, 2));
|
|
||||||
|
|
||||||
geom.setIndex(new Uint16BufferAttribute(this.indices, 1));
|
|
||||||
|
|
||||||
for (const group of this.groups) {
|
|
||||||
geom.addGroup(group.start, group.count, group.material_index);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
geom.computeBoundingSphere();
|
create_mesh(
|
||||||
geom.computeBoundingBox();
|
object: NjObject<NjModel>,
|
||||||
|
materials: Material[],
|
||||||
return geom;
|
default_material: Material
|
||||||
}
|
): Mesh {
|
||||||
|
|
||||||
create_mesh(object: NjObject<NjModel>): Mesh {
|
|
||||||
const geom = this.create_buffer_geometry(object);
|
const geom = this.create_buffer_geometry(object);
|
||||||
|
|
||||||
const max_mat_idx = this.groups.reduce((max, g) => Math.max(max, g.material_index), 0);
|
materials = [DUMMY_MATERIAL, ...materials];
|
||||||
|
const max_mat_idx = this.builder.max_material_index || 0;
|
||||||
|
|
||||||
for (let i = this.materials.length - 1; i < max_mat_idx; ++i) {
|
for (let i = materials.length - 1; i < max_mat_idx; ++i) {
|
||||||
this.materials.push(DEFAULT_MATERIAL);
|
materials.push(default_material);
|
||||||
}
|
}
|
||||||
|
|
||||||
return new Mesh(geom, this.materials);
|
return new Mesh(geom, materials);
|
||||||
}
|
}
|
||||||
|
|
||||||
create_skinned_mesh(object: NjObject<NjModel>): SkinnedMesh {
|
create_skinned_mesh(
|
||||||
|
object: NjObject<NjModel>,
|
||||||
|
materials: Material[],
|
||||||
|
default_material: Material
|
||||||
|
): SkinnedMesh {
|
||||||
const geom = this.create_buffer_geometry(object);
|
const geom = this.create_buffer_geometry(object);
|
||||||
|
|
||||||
geom.addAttribute("skinIndex", new Uint16BufferAttribute(this.bone_indices, 4));
|
materials = [DUMMY_MATERIAL, ...materials];
|
||||||
geom.addAttribute("skinWeight", new Float32BufferAttribute(this.bone_weights, 4));
|
const max_mat_idx = this.builder.max_material_index || 0;
|
||||||
|
|
||||||
const max_mat_idx = this.groups.reduce((max, g) => Math.max(max, g.material_index), 0);
|
for (let i = materials.length - 1; i < max_mat_idx; ++i) {
|
||||||
|
materials.push(default_material);
|
||||||
for (let i = this.materials.length - 1; i < max_mat_idx; ++i) {
|
|
||||||
this.materials.push(DEFAULT_SKINNED_MATERIAL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const mesh = new SkinnedMesh(geom, this.materials);
|
const mesh = new SkinnedMesh(geom, materials);
|
||||||
mesh.add(this.bones[0]);
|
mesh.add(this.bones[0]);
|
||||||
mesh.bind(new Skeleton(this.bones));
|
mesh.bind(new Skeleton(this.bones));
|
||||||
|
|
||||||
@ -230,15 +225,7 @@ class Object3DCreator {
|
|||||||
if (is_njcm_model(model)) {
|
if (is_njcm_model(model)) {
|
||||||
this.njcm_model_to_geometry(model, matrix);
|
this.njcm_model_to_geometry(model, matrix);
|
||||||
} else {
|
} else {
|
||||||
xj_model_to_geometry(
|
this.xj_model_to_geometry(model, matrix);
|
||||||
model,
|
|
||||||
matrix,
|
|
||||||
this.positions,
|
|
||||||
this.normals,
|
|
||||||
this.uvs,
|
|
||||||
this.indices,
|
|
||||||
this.groups
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -247,7 +234,7 @@ class Object3DCreator {
|
|||||||
|
|
||||||
const new_vertices = model.vertices.map(vertex => {
|
const new_vertices = model.vertices.map(vertex => {
|
||||||
const position = vec3_to_threejs(vertex.position);
|
const position = vec3_to_threejs(vertex.position);
|
||||||
const normal = vertex.normal ? vec3_to_threejs(vertex.normal) : DEFAULT_NORMAL;
|
const normal = vertex.normal ? vec3_to_threejs(vertex.normal) : new Vector3(0, 1, 0);
|
||||||
|
|
||||||
position.applyMatrix4(matrix);
|
position.applyMatrix4(matrix);
|
||||||
normal.applyMatrix3(normal_matrix);
|
normal.applyMatrix3(normal_matrix);
|
||||||
@ -265,7 +252,7 @@ class Object3DCreator {
|
|||||||
this.vertices.put(new_vertices);
|
this.vertices.put(new_vertices);
|
||||||
|
|
||||||
for (const mesh of model.meshes) {
|
for (const mesh of model.meshes) {
|
||||||
const start_index_count = this.indices.length;
|
const start_index_count = this.builder.index_count;
|
||||||
|
|
||||||
for (let i = 0; i < mesh.vertices.length; ++i) {
|
for (let i = 0; i < mesh.vertices.length; ++i) {
|
||||||
const mesh_vertex = mesh.vertices[i];
|
const mesh_vertex = mesh.vertices[i];
|
||||||
@ -274,55 +261,121 @@ class Object3DCreator {
|
|||||||
if (vertices.length) {
|
if (vertices.length) {
|
||||||
const vertex = vertices[0];
|
const vertex = vertices[0];
|
||||||
const normal = vertex.normal || mesh_vertex.normal || DEFAULT_NORMAL;
|
const normal = vertex.normal || mesh_vertex.normal || DEFAULT_NORMAL;
|
||||||
const index = this.positions.length / 3;
|
const index = this.builder.vertex_count;
|
||||||
|
|
||||||
this.positions.push(vertex.position.x, vertex.position.y, vertex.position.z);
|
this.builder.add_vertex(
|
||||||
this.normals.push(normal.x, normal.y, normal.z);
|
vertex.position,
|
||||||
|
normal,
|
||||||
if (mesh.has_tex_coords) {
|
mesh.has_tex_coords ? mesh_vertex.tex_coords! : DEFAULT_UV
|
||||||
this.uvs.push(mesh_vertex.tex_coords!.x, mesh_vertex.tex_coords!.y);
|
);
|
||||||
} else {
|
|
||||||
this.uvs.push(0, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (i >= 2) {
|
if (i >= 2) {
|
||||||
if (i % 2 === (mesh.clockwise_winding ? 1 : 0)) {
|
if (i % 2 === (mesh.clockwise_winding ? 1 : 0)) {
|
||||||
this.indices.push(index - 2);
|
this.builder.add_index(index - 2);
|
||||||
this.indices.push(index - 1);
|
this.builder.add_index(index - 1);
|
||||||
this.indices.push(index);
|
this.builder.add_index(index);
|
||||||
} else {
|
} else {
|
||||||
this.indices.push(index - 2);
|
this.builder.add_index(index - 2);
|
||||||
this.indices.push(index);
|
this.builder.add_index(index);
|
||||||
this.indices.push(index - 1);
|
this.builder.add_index(index - 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const bone_indices = [0, 0, 0, 0];
|
const bones = [[0, 0], [0, 0], [0, 0], [0, 0]];
|
||||||
const bone_weights = [0, 0, 0, 0];
|
|
||||||
|
|
||||||
for (let j = vertices.length - 1; j >= 0; j--) {
|
for (let j = vertices.length - 1; j >= 0; j--) {
|
||||||
const vertex = vertices[j];
|
const vertex = vertices[j];
|
||||||
bone_indices[vertex.bone_weight_status] = vertex.bone_id;
|
bones[vertex.bone_weight_status] = [vertex.bone_id, vertex.bone_weight];
|
||||||
bone_weights[vertex.bone_weight_status] = vertex.bone_weight;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
this.bone_indices.push(...bone_indices);
|
for (const [bone_index, bone_weight] of bones) {
|
||||||
this.bone_weights.push(...bone_weights);
|
this.builder.add_bone(bone_index, bone_weight);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const last_group = this.groups[this.groups.length - 1];
|
this.builder.add_group(
|
||||||
const mat_idx = mesh.texture_id == null ? 0 : mesh.texture_id + 1;
|
start_index_count,
|
||||||
|
this.builder.index_count - start_index_count,
|
||||||
|
mesh.texture_id
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (last_group && last_group.material_index === mat_idx) {
|
private xj_model_to_geometry(model: XjModel, matrix: Matrix4): void {
|
||||||
last_group.count += this.indices.length - start_index_count;
|
const index_offset = this.builder.vertex_count;
|
||||||
|
const normal_matrix = new Matrix3().getNormalMatrix(matrix);
|
||||||
|
|
||||||
|
for (let { position, normal, uv } of model.vertices) {
|
||||||
|
const p = vec3_to_threejs(position).applyMatrix4(matrix);
|
||||||
|
|
||||||
|
const local_n = normal ? vec3_to_threejs(normal) : new Vector3(0, 1, 0);
|
||||||
|
const n = local_n.applyMatrix3(normal_matrix);
|
||||||
|
|
||||||
|
const tuv = uv || DEFAULT_UV;
|
||||||
|
|
||||||
|
this.builder.add_vertex(p, n, tuv);
|
||||||
|
}
|
||||||
|
|
||||||
|
let current_mat_idx: number | undefined;
|
||||||
|
|
||||||
|
for (const mesh of model.meshes) {
|
||||||
|
const start_index_count = this.builder.index_count;
|
||||||
|
let clockwise = false;
|
||||||
|
|
||||||
|
for (let j = 2; j < mesh.indices.length; ++j) {
|
||||||
|
const a = index_offset + mesh.indices[j - 2];
|
||||||
|
const b = index_offset + mesh.indices[j - 1];
|
||||||
|
const c = index_offset + mesh.indices[j];
|
||||||
|
const pa = this.builder.get_position(a);
|
||||||
|
const pb = this.builder.get_position(b);
|
||||||
|
const pc = this.builder.get_position(c);
|
||||||
|
const na = this.builder.get_normal(a);
|
||||||
|
const nb = this.builder.get_normal(b);
|
||||||
|
const nc = this.builder.get_normal(c);
|
||||||
|
|
||||||
|
// 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();
|
||||||
|
}
|
||||||
|
|
||||||
|
const opposite_count =
|
||||||
|
(normal.dot(na) < 0 ? 1 : 0) +
|
||||||
|
(normal.dot(nb) < 0 ? 1 : 0) +
|
||||||
|
(normal.dot(nc) < 0 ? 1 : 0);
|
||||||
|
|
||||||
|
if (opposite_count >= 2) {
|
||||||
|
clockwise = !clockwise;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (clockwise) {
|
||||||
|
this.builder.add_index(b);
|
||||||
|
this.builder.add_index(a);
|
||||||
|
this.builder.add_index(c);
|
||||||
} else {
|
} else {
|
||||||
this.groups.push({
|
this.builder.add_index(a);
|
||||||
start: start_index_count,
|
this.builder.add_index(b);
|
||||||
count: this.indices.length - start_index_count,
|
this.builder.add_index(c);
|
||||||
material_index: mat_idx,
|
}
|
||||||
});
|
|
||||||
}
|
clockwise = !clockwise;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mesh.material_properties.texture_id != null) {
|
||||||
|
current_mat_idx = mesh.material_properties.texture_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.builder.add_group(
|
||||||
|
start_index_count,
|
||||||
|
this.builder.index_count - start_index_count,
|
||||||
|
current_mat_idx
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,100 +0,0 @@
|
|||||||
import { Matrix3, Matrix4, Vector3 } from "three";
|
|
||||||
import { vec3_to_threejs } from ".";
|
|
||||||
import { XjModel } from "../../data_formats/parsing/ninja/xj";
|
|
||||||
import { Vec2 } from "../../data_formats/vector";
|
|
||||||
import { VertexGroup } from "./ninja_geometry";
|
|
||||||
|
|
||||||
const DEFAULT_NORMAL = new Vector3(0, 1, 0);
|
|
||||||
const DEFAULT_UV = new Vec2(0, 0);
|
|
||||||
|
|
||||||
export function xj_model_to_geometry(
|
|
||||||
model: XjModel,
|
|
||||||
matrix: Matrix4,
|
|
||||||
positions: number[],
|
|
||||||
normals: number[],
|
|
||||||
uvs: number[],
|
|
||||||
indices: number[],
|
|
||||||
groups: VertexGroup[]
|
|
||||||
): void {
|
|
||||||
const index_offset = positions.length / 3;
|
|
||||||
const normal_matrix = new Matrix3().getNormalMatrix(matrix);
|
|
||||||
|
|
||||||
for (let { position, normal, uv } of model.vertices) {
|
|
||||||
const p = vec3_to_threejs(position).applyMatrix4(matrix);
|
|
||||||
positions.push(p.x, p.y, p.z);
|
|
||||||
|
|
||||||
const local_n = normal ? vec3_to_threejs(normal) : DEFAULT_NORMAL;
|
|
||||||
const n = local_n.applyMatrix3(normal_matrix);
|
|
||||||
normals.push(n.x, n.y, n.z);
|
|
||||||
|
|
||||||
const tuv = uv || DEFAULT_UV;
|
|
||||||
uvs.push(tuv.x, tuv.y);
|
|
||||||
}
|
|
||||||
|
|
||||||
let current_mat_idx = 0;
|
|
||||||
|
|
||||||
for (const mesh of model.meshes) {
|
|
||||||
const start_index_count = indices.length;
|
|
||||||
let clockwise = true;
|
|
||||||
|
|
||||||
for (let j = 2; j < mesh.indices.length; ++j) {
|
|
||||||
const a = index_offset + mesh.indices[j - 2];
|
|
||||||
const b = index_offset + mesh.indices[j - 1];
|
|
||||||
const c = index_offset + mesh.indices[j];
|
|
||||||
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();
|
|
||||||
}
|
|
||||||
|
|
||||||
const opposite_count =
|
|
||||||
(normal.dot(na) < 0 ? 1 : 0) +
|
|
||||||
(normal.dot(nb) < 0 ? 1 : 0) +
|
|
||||||
(normal.dot(nc) < 0 ? 1 : 0);
|
|
||||||
|
|
||||||
if (opposite_count >= 2) {
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
const last_group = groups[groups.length - 1];
|
|
||||||
|
|
||||||
if (mesh.material_properties.texture_id != null) {
|
|
||||||
current_mat_idx = mesh.material_properties.texture_id + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (last_group && last_group.material_index === current_mat_idx) {
|
|
||||||
last_group.count += indices.length - start_index_count;
|
|
||||||
} else {
|
|
||||||
groups.push({
|
|
||||||
start: start_index_count,
|
|
||||||
count: indices.length - start_index_count,
|
|
||||||
material_index: current_mat_idx,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user