Added various improvements.

This commit is contained in:
Daan Vanden Bosch 2019-07-01 15:01:40 +02:00
parent e8226d94be
commit f19f2df2d2
4 changed files with 65 additions and 18 deletions

View File

@ -95,6 +95,10 @@ function parse_ninja<M extends NinjaModel>(
if (iff_type_id === 'NJCM') { if (iff_type_id === 'NJCM') {
return parse_sibling_objects(cursor.take(iff_chunk_size), parse_model, context); return parse_sibling_objects(cursor.take(iff_chunk_size), parse_model, context);
} else { } else {
if (iff_chunk_size > cursor.bytes_left) {
break;
}
cursor.seek(iff_chunk_size); cursor.seek(iff_chunk_size);
} }
} }

View File

@ -92,8 +92,20 @@ type NjVertex = {
} }
type NjTriangleStrip = { type NjTriangleStrip = {
ignore_light: boolean,
ignore_specular: boolean,
ignore_ambient: boolean,
use_alpha: boolean,
double_side: boolean,
flat_shading: boolean,
environment_mapping: boolean,
clockwise_winding: boolean, clockwise_winding: boolean,
indices: number[], vertices: NjMeshVertex[],
}
type NjMeshVertex = {
index: number,
normal?: Vec3,
} }
export function parse_nj_model(cursor: BufferCursor, cached_chunk_offsets: number[]): NjModel { export function parse_nj_model(cursor: BufferCursor, cached_chunk_offsets: number[]): NjModel {
@ -227,7 +239,7 @@ function parse_chunks(
chunks.push({ chunks.push({
type: NjChunkType.Strip, type: NjChunkType.Strip,
type_id, type_id,
triangle_strips: parse_triangle_strip_chunk(cursor, type_id) triangle_strips: parse_triangle_strip_chunk(cursor, type_id, flags)
}); });
} else if (type_id === 255) { } else if (type_id === 255) {
size = wide_end_chunks ? 2 : 0; size = wide_end_chunks ? 2 : 0;
@ -345,8 +357,18 @@ function parse_vertex_chunk(
function parse_triangle_strip_chunk( function parse_triangle_strip_chunk(
cursor: BufferCursor, cursor: BufferCursor,
chunk_type_id: number chunk_type_id: number,
flags: number
): NjTriangleStrip[] { ): NjTriangleStrip[] {
const render_flags = {
ignore_light: (flags & 0b1) !== 0,
ignore_specular: (flags & 0b10) !== 0,
ignore_ambient: (flags & 0b100) !== 0,
use_alpha: (flags & 0b1000) !== 0,
double_side: (flags & 0b10000) !== 0,
flat_shading: (flags & 0b100000) !== 0,
environment_mapping: (flags & 0b1000000) !== 0,
};
const user_offset_and_strip_count = cursor.u16(); const user_offset_and_strip_count = cursor.u16();
const user_flags_size = user_offset_and_strip_count >>> 14; const user_flags_size = user_offset_and_strip_count >>> 14;
const strip_count = user_offset_and_strip_count & 0x3FFF; const strip_count = user_offset_and_strip_count & 0x3FFF;
@ -382,10 +404,13 @@ function parse_triangle_strip_chunk(
const clockwise_winding = winding_flag_and_index_count < 1; const clockwise_winding = winding_flag_and_index_count < 1;
const index_count = Math.abs(winding_flag_and_index_count); const index_count = Math.abs(winding_flag_and_index_count);
const indices = []; const vertices: NjMeshVertex[] = [];
for (let j = 0; j < index_count; ++j) { for (let j = 0; j < index_count; ++j) {
indices.push(cursor.u16()); const vertex: NjMeshVertex = {
index: cursor.u16()
};
vertices.push(vertex);
if (parse_texture_coords) { if (parse_texture_coords) {
cursor.seek(4); cursor.seek(4);
@ -396,7 +421,11 @@ function parse_triangle_strip_chunk(
} }
if (parse_normal) { if (parse_normal) {
cursor.seek(6); vertex.normal = new Vec3(
cursor.u16(),
cursor.u16(),
cursor.u16()
);
} }
if (parse_texture_coords_hires) { if (parse_texture_coords_hires) {
@ -408,7 +437,11 @@ function parse_triangle_strip_chunk(
} }
} }
strips.push({ clockwise_winding, indices }); strips.push({
...render_flags,
clockwise_winding,
vertices,
});
} }
return strips; return strips;

View File

@ -188,13 +188,13 @@ class Object3DCreator {
this.vertices.put(new_vertices); this.vertices.put(new_vertices);
for (const mesh of model.meshes) { for (const mesh of model.meshes) {
for (let i = 0; i < mesh.indices.length; ++i) { for (let i = 0; i < mesh.vertices.length; ++i) {
const vertex_idx = mesh.indices[i]; const mesh_vertex = mesh.vertices[i];
const vertices = this.vertices.get(vertex_idx); const vertices = this.vertices.get(mesh_vertex.index);
if (vertices.length) { if (vertices.length) {
const vertex = vertices[0]; const vertex = vertices[0];
const normal = vertex.normal || DEFAULT_NORMAL; const normal = vertex.normal || mesh_vertex.normal || DEFAULT_NORMAL;
const index = this.positions.length / 3; const index = this.positions.length / 3;
this.positions.push(vertex.position.x, vertex.position.y, vertex.position.z); this.positions.push(vertex.position.x, vertex.position.y, vertex.position.z);

View File

@ -1,16 +1,26 @@
import { BufferGeometry } from "three"; import { BufferGeometry, CylinderBufferGeometry } from "three";
import { NpcType, ObjectType } from "../domain"; import { NpcType, ObjectType } from "../domain";
import { BufferCursor } from "../data_formats/BufferCursor"; import { BufferCursor } from "../data_formats/BufferCursor";
import { get_npc_data, get_object_data } from "./binary_assets"; import { get_npc_data, get_object_data } from "./binary_assets";
import { ninja_object_to_buffer_geometry } from "../rendering/models"; import { ninja_object_to_buffer_geometry } from "../rendering/models";
import { parse_nj, parse_xj } from "../data_formats/parsing/ninja"; import { parse_nj, parse_xj } from "../data_formats/parsing/ninja";
const npc_cache: Map<number, Promise<BufferGeometry>> = new Map(); const DEFAULT_ENTITY = new CylinderBufferGeometry(3, 3, 20);
const object_cache: Map<number, Promise<BufferGeometry>> = new Map(); DEFAULT_ENTITY.translate(0, 10, 0);
const DEFAULT_ENTITY_PROMISE: Promise<BufferGeometry> = new Promise(resolve =>
resolve(DEFAULT_ENTITY)
);
const npc_cache: Map<NpcType, Promise<BufferGeometry>> = new Map();
npc_cache.set(NpcType.Unknown, DEFAULT_ENTITY_PROMISE);
const object_cache: Map<ObjectType, Promise<BufferGeometry>> = new Map();
object_cache.set(ObjectType.Unknown, DEFAULT_ENTITY_PROMISE);
class EntityStore { class EntityStore {
async get_npc_geometry(npc_type: NpcType): Promise<BufferGeometry> { async get_npc_geometry(npc_type: NpcType): Promise<BufferGeometry> {
let mesh = npc_cache.get(npc_type.id); let mesh = npc_cache.get(npc_type);
if (mesh) { if (mesh) {
return mesh; return mesh;
@ -26,13 +36,13 @@ class EntityStore {
} }
}); });
npc_cache.set(npc_type.id, mesh); npc_cache.set(npc_type, mesh);
return mesh; return mesh;
} }
} }
async get_object_geometry(object_type: ObjectType): Promise<BufferGeometry> { async get_object_geometry(object_type: ObjectType): Promise<BufferGeometry> {
let geometry = object_cache.get(object_type.id); let geometry = object_cache.get(object_type);
if (geometry) { if (geometry) {
return geometry; return geometry;
@ -48,7 +58,7 @@ class EntityStore {
} }
}); });
object_cache.set(object_type.id, geometry); object_cache.set(object_type, geometry);
return geometry; return geometry;
} }
} }