mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Added "additive" blending for the materials that need it.
This commit is contained in:
parent
5f7547dae9
commit
06e1a8e60b
@ -39,6 +39,8 @@ export type NjcmTriangleStrip = {
|
|||||||
has_tex_coords: boolean;
|
has_tex_coords: boolean;
|
||||||
has_normal: boolean;
|
has_normal: boolean;
|
||||||
texture_id?: number;
|
texture_id?: number;
|
||||||
|
src_alpha?: number;
|
||||||
|
dst_alpha?: number;
|
||||||
vertices: NjcmMeshVertex[];
|
vertices: NjcmMeshVertex[];
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -89,6 +91,8 @@ type NjcmNullChunk = {
|
|||||||
|
|
||||||
type NjcmBitsChunk = {
|
type NjcmBitsChunk = {
|
||||||
type: NjcmChunkType.Bits;
|
type: NjcmChunkType.Bits;
|
||||||
|
src_alpha: number;
|
||||||
|
dst_alpha: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
type NjcmCachePolygonListChunk = {
|
type NjcmCachePolygonListChunk = {
|
||||||
@ -116,6 +120,11 @@ type NjcmTinyChunk = {
|
|||||||
|
|
||||||
type NjcmMaterialChunk = {
|
type NjcmMaterialChunk = {
|
||||||
type: NjcmChunkType.Material;
|
type: NjcmChunkType.Material;
|
||||||
|
src_alpha: number;
|
||||||
|
dst_alpha: number;
|
||||||
|
diffuse?: NjcmArgb;
|
||||||
|
ambient?: NjcmArgb;
|
||||||
|
specular?: NjcmErgb;
|
||||||
};
|
};
|
||||||
|
|
||||||
type NjcmVertexChunk = {
|
type NjcmVertexChunk = {
|
||||||
@ -145,6 +154,26 @@ type NjcmChunkVertex = {
|
|||||||
calc_continue: boolean;
|
calc_continue: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Channels are in range [0, 1].
|
||||||
|
*/
|
||||||
|
type NjcmArgb = {
|
||||||
|
a: number;
|
||||||
|
r: number;
|
||||||
|
g: number;
|
||||||
|
b: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Channels are not normalized.
|
||||||
|
*/
|
||||||
|
type NjcmErgb = {
|
||||||
|
e: number;
|
||||||
|
r: number;
|
||||||
|
g: number;
|
||||||
|
b: number;
|
||||||
|
};
|
||||||
|
|
||||||
export function parse_njcm_model(cursor: Cursor, cached_chunk_offsets: number[]): NjcmModel {
|
export function parse_njcm_model(cursor: Cursor, cached_chunk_offsets: number[]): NjcmModel {
|
||||||
const vlist_offset = cursor.u32(); // Vertex list
|
const vlist_offset = cursor.u32(); // Vertex list
|
||||||
const plist_offset = cursor.u32(); // Triangle strip index list
|
const plist_offset = cursor.u32(); // Triangle strip index list
|
||||||
@ -175,16 +204,34 @@ export function parse_njcm_model(cursor: Cursor, cached_chunk_offsets: number[])
|
|||||||
cursor.seek_start(plist_offset);
|
cursor.seek_start(plist_offset);
|
||||||
|
|
||||||
let texture_id: number | undefined = undefined;
|
let texture_id: number | undefined = undefined;
|
||||||
|
let src_alpha: number | undefined = undefined;
|
||||||
|
let dst_alpha: number | undefined = undefined;
|
||||||
|
|
||||||
for (const chunk of parse_chunks(cursor, cached_chunk_offsets, false)) {
|
for (const chunk of parse_chunks(cursor, cached_chunk_offsets, false)) {
|
||||||
if (chunk.type === NjcmChunkType.Tiny) {
|
switch (chunk.type) {
|
||||||
texture_id = chunk.texture_id;
|
case NjcmChunkType.Bits:
|
||||||
} else if (chunk.type === NjcmChunkType.Strip) {
|
src_alpha = chunk.src_alpha;
|
||||||
for (const strip of chunk.triangle_strips) {
|
dst_alpha = chunk.dst_alpha;
|
||||||
strip.texture_id = texture_id;
|
break;
|
||||||
}
|
|
||||||
|
|
||||||
meshes.push(...chunk.triangle_strips);
|
case NjcmChunkType.Tiny:
|
||||||
|
texture_id = chunk.texture_id;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case NjcmChunkType.Material:
|
||||||
|
src_alpha = chunk.src_alpha;
|
||||||
|
dst_alpha = chunk.dst_alpha;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case NjcmChunkType.Strip:
|
||||||
|
for (const strip of chunk.triangle_strips) {
|
||||||
|
strip.texture_id = texture_id;
|
||||||
|
strip.src_alpha = src_alpha;
|
||||||
|
strip.dst_alpha = dst_alpha;
|
||||||
|
}
|
||||||
|
|
||||||
|
meshes.push(...chunk.triangle_strips);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -222,6 +269,8 @@ function parse_chunks(
|
|||||||
chunks.push({
|
chunks.push({
|
||||||
type: NjcmChunkType.Bits,
|
type: NjcmChunkType.Bits,
|
||||||
type_id,
|
type_id,
|
||||||
|
src_alpha: (flags >>> 3) & 0b111,
|
||||||
|
dst_alpha: flags & 0b111,
|
||||||
});
|
});
|
||||||
} else if (type_id === 4) {
|
} else if (type_id === 4) {
|
||||||
const cache_index = flags;
|
const cache_index = flags;
|
||||||
@ -265,9 +314,46 @@ function parse_chunks(
|
|||||||
});
|
});
|
||||||
} else if (17 <= type_id && type_id <= 31) {
|
} else if (17 <= type_id && type_id <= 31) {
|
||||||
size = 2 + 2 * cursor.u16();
|
size = 2 + 2 * cursor.u16();
|
||||||
|
|
||||||
|
let diffuse: NjcmArgb | undefined;
|
||||||
|
let ambient: NjcmArgb | undefined;
|
||||||
|
let specular: NjcmErgb | undefined;
|
||||||
|
|
||||||
|
if ((flags & 0b1) !== 0) {
|
||||||
|
diffuse = {
|
||||||
|
b: cursor.u8() / 255,
|
||||||
|
g: cursor.u8() / 255,
|
||||||
|
r: cursor.u8() / 255,
|
||||||
|
a: cursor.u8() / 255,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((flags & 0b10) !== 0) {
|
||||||
|
ambient = {
|
||||||
|
b: cursor.u8() / 255,
|
||||||
|
g: cursor.u8() / 255,
|
||||||
|
r: cursor.u8() / 255,
|
||||||
|
a: cursor.u8() / 255,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((flags & 0b100) !== 0) {
|
||||||
|
specular = {
|
||||||
|
b: cursor.u8(),
|
||||||
|
g: cursor.u8(),
|
||||||
|
r: cursor.u8(),
|
||||||
|
e: cursor.u8(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
chunks.push({
|
chunks.push({
|
||||||
type: NjcmChunkType.Material,
|
type: NjcmChunkType.Material,
|
||||||
type_id,
|
type_id,
|
||||||
|
src_alpha: (flags >>> 3) & 0b111,
|
||||||
|
dst_alpha: flags & 0b111,
|
||||||
|
diffuse,
|
||||||
|
ambient,
|
||||||
|
specular,
|
||||||
});
|
});
|
||||||
} else if (32 <= type_id && type_id <= 50) {
|
} else if (32 <= type_id && type_id <= 50) {
|
||||||
size = 2 + 4 * cursor.u16();
|
size = 2 + 4 * cursor.u16();
|
||||||
|
@ -1,31 +1,67 @@
|
|||||||
import {
|
import {
|
||||||
|
Bone,
|
||||||
BufferGeometry,
|
BufferGeometry,
|
||||||
Float32BufferAttribute,
|
Float32BufferAttribute,
|
||||||
Uint16BufferAttribute,
|
Uint16BufferAttribute,
|
||||||
Vector3,
|
Vector3,
|
||||||
Bone,
|
|
||||||
} from "three";
|
} from "three";
|
||||||
|
import { map_get_or_put } from "../../util";
|
||||||
|
|
||||||
export type BuilderData = {
|
export type BuilderData = {
|
||||||
created_by_geometry_builder: boolean;
|
readonly created_by_geometry_builder: boolean;
|
||||||
/**
|
readonly materials: BuilderMaterial[];
|
||||||
* Maps material indices to normalized material indices.
|
readonly bones: Bone[];
|
||||||
*/
|
|
||||||
normalized_material_indices: Map<number, number>;
|
|
||||||
bones: Bone[];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export type BuilderVec2 = {
|
export type BuilderVec2 = {
|
||||||
x: number;
|
readonly x: number;
|
||||||
y: number;
|
readonly y: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type BuilderVec3 = {
|
export type BuilderVec3 = {
|
||||||
x: number;
|
readonly x: number;
|
||||||
y: number;
|
readonly y: number;
|
||||||
z: number;
|
readonly z: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type BuilderMaterial = {
|
||||||
|
readonly texture_id?: number;
|
||||||
|
readonly alpha: boolean;
|
||||||
|
readonly additive_blending: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps various material properties to material IDs.
|
||||||
|
*/
|
||||||
|
export class MaterialMap {
|
||||||
|
private readonly materials: BuilderMaterial[] = [{ alpha: false, additive_blending: false }];
|
||||||
|
private readonly map = new Map<number, number>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an index to an existing material if one exists for the given arguments. Otherwise
|
||||||
|
* adds a new material and returns its index.
|
||||||
|
*/
|
||||||
|
add_material(
|
||||||
|
texture_id?: number,
|
||||||
|
alpha: boolean = false,
|
||||||
|
additive_blending: boolean = false,
|
||||||
|
): number {
|
||||||
|
if (texture_id == undefined) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
const key = (texture_id << 2) | (alpha ? 0b10 : 0) | (additive_blending ? 1 : 0);
|
||||||
|
return map_get_or_put(this.map, key, () => {
|
||||||
|
this.materials.push({ texture_id, alpha, additive_blending });
|
||||||
|
return this.materials.length - 1;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get_materials(): BuilderMaterial[] {
|
||||||
|
return this.materials;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
type VertexGroup = {
|
type VertexGroup = {
|
||||||
offset: number;
|
offset: number;
|
||||||
size: number;
|
size: number;
|
||||||
@ -33,18 +69,18 @@ type VertexGroup = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export class GeometryBuilder {
|
export class GeometryBuilder {
|
||||||
private positions: number[] = [];
|
private readonly positions: number[] = [];
|
||||||
private normals: number[] = [];
|
private readonly normals: number[] = [];
|
||||||
private uvs: number[] = [];
|
private readonly uvs: number[] = [];
|
||||||
private indices: number[] = [];
|
private readonly indices: number[] = [];
|
||||||
private bones: Bone[] = [];
|
private readonly bones: Bone[] = [];
|
||||||
private bone_indices: number[] = [];
|
private readonly bone_indices: number[] = [];
|
||||||
private bone_weights: number[] = [];
|
private readonly bone_weights: number[] = [];
|
||||||
private groups: VertexGroup[] = [];
|
private readonly groups: VertexGroup[] = [];
|
||||||
/**
|
/**
|
||||||
* Will contain all material indices used in {@link this.groups} and -1 for the dummy material.
|
* Will contain all material indices used in {@link this.groups} and -1 for the dummy material.
|
||||||
*/
|
*/
|
||||||
private material_indices = new Set<number>([-1]);
|
private readonly material_map = new MaterialMap();
|
||||||
|
|
||||||
get vertex_count(): number {
|
get vertex_count(): number {
|
||||||
return this.positions.length / 3;
|
return this.positions.length / 3;
|
||||||
@ -89,26 +125,29 @@ export class GeometryBuilder {
|
|||||||
this.bone_weights.push(weight);
|
this.bone_weights.push(weight);
|
||||||
}
|
}
|
||||||
|
|
||||||
add_group(offset: number, size: number, material_index?: number): void {
|
add_group(
|
||||||
|
offset: number,
|
||||||
|
size: number,
|
||||||
|
texture_id?: number,
|
||||||
|
alpha: boolean = false,
|
||||||
|
additive_blending: boolean = false,
|
||||||
|
): void {
|
||||||
const last_group = this.groups[this.groups.length - 1];
|
const last_group = this.groups[this.groups.length - 1];
|
||||||
const mat_idx = material_index == null ? -1 : material_index;
|
const material_index = this.material_map.add_material(texture_id, alpha, additive_blending);
|
||||||
|
|
||||||
if (last_group && last_group.material_index === mat_idx) {
|
if (last_group && last_group.material_index === material_index) {
|
||||||
last_group.size += size;
|
last_group.size += size;
|
||||||
} else {
|
} else {
|
||||||
this.groups.push({
|
this.groups.push({
|
||||||
offset,
|
offset,
|
||||||
size,
|
size,
|
||||||
material_index: mat_idx,
|
material_index,
|
||||||
});
|
});
|
||||||
this.material_indices.add(mat_idx);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
build(): BufferGeometry {
|
build(): BufferGeometry {
|
||||||
const geom = new BufferGeometry();
|
const geom = new BufferGeometry();
|
||||||
const data = geom.userData as BuilderData;
|
|
||||||
data.created_by_geometry_builder = true;
|
|
||||||
|
|
||||||
geom.setAttribute("position", new Float32BufferAttribute(this.positions, 3));
|
geom.setAttribute("position", new Float32BufferAttribute(this.positions, 3));
|
||||||
geom.setAttribute("normal", new Float32BufferAttribute(this.normals, 3));
|
geom.setAttribute("normal", new Float32BufferAttribute(this.normals, 3));
|
||||||
@ -116,28 +155,28 @@ export class GeometryBuilder {
|
|||||||
|
|
||||||
geom.setIndex(new Uint16BufferAttribute(this.indices, 1));
|
geom.setIndex(new Uint16BufferAttribute(this.indices, 1));
|
||||||
|
|
||||||
|
let bones: Bone[];
|
||||||
|
|
||||||
if (this.bone_indices.length && this.bones.length) {
|
if (this.bone_indices.length && this.bones.length) {
|
||||||
geom.setAttribute("skinIndex", new Uint16BufferAttribute(this.bone_indices, 4));
|
geom.setAttribute("skinIndex", new Uint16BufferAttribute(this.bone_indices, 4));
|
||||||
geom.setAttribute("skinWeight", new Float32BufferAttribute(this.bone_weights, 4));
|
geom.setAttribute("skinWeight", new Float32BufferAttribute(this.bone_weights, 4));
|
||||||
data.bones = this.bones;
|
bones = this.bones;
|
||||||
} else {
|
} else {
|
||||||
data.bones = [];
|
bones = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Normalize material indices.
|
|
||||||
const normalized_mat_idxs = new Map<number, number>();
|
|
||||||
let i = 0;
|
|
||||||
|
|
||||||
for (const mat_idx of [...this.material_indices].sort((a, b) => a - b)) {
|
|
||||||
normalized_mat_idxs.set(mat_idx, i++);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use normalized material indices in Three.js groups.
|
|
||||||
for (const group of this.groups) {
|
for (const group of this.groups) {
|
||||||
geom.addGroup(group.offset, group.size, normalized_mat_idxs.get(group.material_index));
|
geom.addGroup(group.offset, group.size, group.material_index);
|
||||||
}
|
}
|
||||||
|
|
||||||
data.normalized_material_indices = normalized_mat_idxs;
|
// noinspection UnnecessaryLocalVariableJS
|
||||||
|
const data: BuilderData = {
|
||||||
|
created_by_geometry_builder: true,
|
||||||
|
materials: this.material_map.get_materials(),
|
||||||
|
bones,
|
||||||
|
};
|
||||||
|
|
||||||
|
geom.userData = data;
|
||||||
|
|
||||||
geom.computeBoundingSphere();
|
geom.computeBoundingSphere();
|
||||||
geom.computeBoundingBox();
|
geom.computeBoundingBox();
|
||||||
|
@ -1,82 +1,78 @@
|
|||||||
import {
|
import {
|
||||||
|
AdditiveBlending,
|
||||||
BufferGeometry,
|
BufferGeometry,
|
||||||
DoubleSide,
|
DoubleSide,
|
||||||
Material,
|
Material,
|
||||||
Mesh,
|
Mesh,
|
||||||
|
MeshBasicMaterial,
|
||||||
MeshLambertMaterial,
|
MeshLambertMaterial,
|
||||||
Skeleton,
|
Skeleton,
|
||||||
SkinnedMesh,
|
SkinnedMesh,
|
||||||
|
Texture,
|
||||||
} from "three";
|
} from "three";
|
||||||
import { BuilderData } from "./GeometryBuilder";
|
import { BuilderData } from "./GeometryBuilder";
|
||||||
|
import { MeshBasicMaterialParameters } from "three/src/materials/MeshBasicMaterial";
|
||||||
|
|
||||||
const DUMMY_MATERIAL = new MeshLambertMaterial({
|
const DUMMY_MATERIAL = new MeshLambertMaterial({
|
||||||
color: 0x00ff00,
|
color: 0x00ff00,
|
||||||
side: DoubleSide,
|
side: DoubleSide,
|
||||||
});
|
});
|
||||||
const DEFAULT_MATERIAL = new MeshLambertMaterial({
|
|
||||||
color: 0xff00ff,
|
|
||||||
side: DoubleSide,
|
|
||||||
});
|
|
||||||
const DEFAULT_SKINNED_MATERIAL = new MeshLambertMaterial({
|
|
||||||
skinning: true,
|
|
||||||
color: 0xff00ff,
|
|
||||||
side: DoubleSide,
|
|
||||||
});
|
|
||||||
|
|
||||||
export function create_mesh(
|
export function create_mesh(
|
||||||
geometry: BufferGeometry,
|
geometry: BufferGeometry,
|
||||||
material?: Material | Material[],
|
textures: (Texture | undefined)[],
|
||||||
default_material: Material = DEFAULT_MATERIAL,
|
|
||||||
): Mesh {
|
|
||||||
return create(geometry, material, default_material, Mesh);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function create_skinned_mesh(
|
|
||||||
geometry: BufferGeometry,
|
|
||||||
material?: Material | Material[],
|
|
||||||
default_material: Material = DEFAULT_SKINNED_MATERIAL,
|
|
||||||
): SkinnedMesh {
|
|
||||||
return create(geometry, material, default_material, SkinnedMesh);
|
|
||||||
}
|
|
||||||
|
|
||||||
function create<M extends Mesh>(
|
|
||||||
geometry: BufferGeometry,
|
|
||||||
material: Material | Material[] | undefined,
|
|
||||||
default_material: Material,
|
default_material: Material,
|
||||||
mesh_constructor: new (geometry: BufferGeometry, material: Material | Material[]) => M,
|
skinning: boolean,
|
||||||
): M {
|
): Mesh {
|
||||||
const {
|
const { created_by_geometry_builder, materials, bones } = geometry.userData as BuilderData;
|
||||||
created_by_geometry_builder,
|
|
||||||
normalized_material_indices: mat_idxs,
|
|
||||||
bones,
|
|
||||||
} = geometry.userData as BuilderData;
|
|
||||||
|
|
||||||
let mat: Material | Material[];
|
let mat: Material | Material[];
|
||||||
|
|
||||||
if (Array.isArray(material)) {
|
if (textures.length && created_by_geometry_builder) {
|
||||||
if (created_by_geometry_builder) {
|
mat = [DUMMY_MATERIAL];
|
||||||
mat = [DUMMY_MATERIAL];
|
|
||||||
|
|
||||||
for (const [idx, normalized_idx] of mat_idxs.entries()) {
|
for (let i = 1; i < materials.length; i++) {
|
||||||
if (normalized_idx > 0) {
|
const { texture_id, alpha, additive_blending } = materials[i];
|
||||||
mat[normalized_idx] = material[idx] || default_material;
|
const tex = texture_id == undefined ? undefined : textures[texture_id];
|
||||||
|
|
||||||
|
if (tex) {
|
||||||
|
const mat_params: MeshBasicMaterialParameters = {
|
||||||
|
skinning,
|
||||||
|
map: tex,
|
||||||
|
side: DoubleSide,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (alpha) {
|
||||||
|
mat_params.transparent = true;
|
||||||
|
mat_params.alphaTest = 0.01;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (additive_blending) {
|
||||||
|
mat_params.transparent = true;
|
||||||
|
mat_params.alphaTest = 0.01;
|
||||||
|
mat_params.blending = AdditiveBlending;
|
||||||
|
}
|
||||||
|
|
||||||
|
mat.push(new MeshBasicMaterial(mat_params));
|
||||||
|
} else {
|
||||||
|
mat.push(
|
||||||
|
new MeshLambertMaterial({
|
||||||
|
skinning,
|
||||||
|
side: DoubleSide,
|
||||||
|
}),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
mat = material;
|
|
||||||
}
|
}
|
||||||
} else if (material) {
|
|
||||||
mat = material;
|
|
||||||
} else {
|
} else {
|
||||||
mat = default_material;
|
mat = default_material;
|
||||||
}
|
}
|
||||||
|
|
||||||
const mesh = new mesh_constructor(geometry, mat);
|
if (created_by_geometry_builder && bones.length && skinning) {
|
||||||
|
const mesh = new SkinnedMesh(geometry, mat);
|
||||||
if (created_by_geometry_builder && bones.length && mesh instanceof SkinnedMesh) {
|
|
||||||
mesh.add(bones[0]);
|
mesh.add(bones[0]);
|
||||||
mesh.bind(new Skeleton(bones));
|
mesh.bind(new Skeleton(bones));
|
||||||
|
return mesh;
|
||||||
|
} else {
|
||||||
|
return new Mesh(geometry, mat);
|
||||||
}
|
}
|
||||||
|
|
||||||
return mesh;
|
|
||||||
}
|
}
|
||||||
|
@ -217,6 +217,8 @@ class GeometryCreator {
|
|||||||
start_index_count,
|
start_index_count,
|
||||||
this.builder.index_count - start_index_count,
|
this.builder.index_count - start_index_count,
|
||||||
mesh.texture_id,
|
mesh.texture_id,
|
||||||
|
mesh.use_alpha,
|
||||||
|
mesh.src_alpha !== 4 || mesh.dst_alpha !== 5,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,19 +1,11 @@
|
|||||||
import { QuestEntityModel } from "../../model/QuestEntityModel";
|
import { QuestEntityModel } from "../../model/QuestEntityModel";
|
||||||
import {
|
import { BufferGeometry, DoubleSide, Mesh, MeshLambertMaterial, Texture } from "three";
|
||||||
BufferGeometry,
|
|
||||||
DoubleSide,
|
|
||||||
Mesh,
|
|
||||||
MeshBasicMaterial,
|
|
||||||
MeshLambertMaterial,
|
|
||||||
Texture,
|
|
||||||
} from "three";
|
|
||||||
import { create_mesh } from "../../../core/rendering/conversion/create_mesh";
|
import { create_mesh } from "../../../core/rendering/conversion/create_mesh";
|
||||||
import {
|
import {
|
||||||
entity_type_to_string,
|
entity_type_to_string,
|
||||||
EntityType,
|
EntityType,
|
||||||
is_npc_type,
|
is_npc_type,
|
||||||
} from "../../../core/data_formats/parsing/quest/entities";
|
} from "../../../core/data_formats/parsing/quest/entities";
|
||||||
import { NpcType } from "../../../core/data_formats/parsing/quest/npc_types";
|
|
||||||
|
|
||||||
export enum ColorType {
|
export enum ColorType {
|
||||||
Normal,
|
Normal,
|
||||||
@ -45,27 +37,8 @@ export function create_entity_type_mesh(
|
|||||||
side: DoubleSide,
|
side: DoubleSide,
|
||||||
});
|
});
|
||||||
|
|
||||||
const mesh = create_mesh(
|
const mesh = create_mesh(geometry, textures, default_material, false);
|
||||||
geometry,
|
|
||||||
textures.length
|
|
||||||
? textures.map(
|
|
||||||
tex =>
|
|
||||||
new MeshBasicMaterial({
|
|
||||||
map: tex,
|
|
||||||
side: DoubleSide,
|
|
||||||
// TODO: figure out why these NPC types don't render correctly when
|
|
||||||
// transparency is turned on.
|
|
||||||
transparent:
|
|
||||||
type !== NpcType.PofuillySlime && type !== NpcType.PouillySlime,
|
|
||||||
alphaTest: 0.01,
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
: default_material,
|
|
||||||
default_material,
|
|
||||||
);
|
|
||||||
|
|
||||||
mesh.name = entity_type_to_string(type);
|
mesh.name = entity_type_to_string(type);
|
||||||
|
|
||||||
return mesh;
|
return mesh;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -263,8 +263,6 @@ function texture_ids(
|
|||||||
return {
|
return {
|
||||||
section_id: section_id + 275,
|
section_id: section_id + 275,
|
||||||
body: [body_idx, body_idx + 1, body_idx + 2, body + 250],
|
body: [body_idx, body_idx + 1, body_idx + 2, body + 250],
|
||||||
// Eyes don't look correct because NJCM material chunks (which contain alpha blending
|
|
||||||
// details) aren't parsed yet. Material.blending should be AdditiveBlending.
|
|
||||||
head: [body_idx + 3, body_idx + 4],
|
head: [body_idx + 3, body_idx + 4],
|
||||||
hair: [],
|
hair: [],
|
||||||
accessories: [],
|
accessories: [],
|
||||||
|
@ -3,7 +3,6 @@ import {
|
|||||||
AnimationMixer,
|
AnimationMixer,
|
||||||
Clock,
|
Clock,
|
||||||
DoubleSide,
|
DoubleSide,
|
||||||
MeshBasicMaterial,
|
|
||||||
MeshLambertMaterial,
|
MeshLambertMaterial,
|
||||||
Object3D,
|
Object3D,
|
||||||
PerspectiveCamera,
|
PerspectiveCamera,
|
||||||
@ -14,7 +13,7 @@ import {
|
|||||||
import { Disposable } from "../../core/observable/Disposable";
|
import { Disposable } from "../../core/observable/Disposable";
|
||||||
import { NjMotion } from "../../core/data_formats/parsing/ninja/motion";
|
import { NjMotion } from "../../core/data_formats/parsing/ninja/motion";
|
||||||
import { xvr_texture_to_texture } from "../../core/rendering/conversion/ninja_textures";
|
import { xvr_texture_to_texture } from "../../core/rendering/conversion/ninja_textures";
|
||||||
import { create_mesh, create_skinned_mesh } from "../../core/rendering/conversion/create_mesh";
|
import { create_mesh } from "../../core/rendering/conversion/create_mesh";
|
||||||
import { ninja_object_to_buffer_geometry } from "../../core/rendering/conversion/ninja_geometry";
|
import { ninja_object_to_buffer_geometry } from "../../core/rendering/conversion/ninja_geometry";
|
||||||
import {
|
import {
|
||||||
create_animation_clip,
|
create_animation_clip,
|
||||||
@ -150,25 +149,13 @@ export class ModelRenderer extends Renderer implements Disposable {
|
|||||||
const geometry = ninja_object_to_buffer_geometry(nj_object);
|
const geometry = ninja_object_to_buffer_geometry(nj_object);
|
||||||
const has_skeleton = geometry.getAttribute("skinIndex") != undefined;
|
const has_skeleton = geometry.getAttribute("skinIndex") != undefined;
|
||||||
|
|
||||||
const materials = textures.map(tex =>
|
this.mesh = create_mesh(
|
||||||
tex
|
geometry,
|
||||||
? new MeshBasicMaterial({
|
textures,
|
||||||
skinning: has_skeleton,
|
has_skeleton ? DEFAULT_SKINNED_MATERIAL : DEFAULT_MATERIAL,
|
||||||
map: tex,
|
has_skeleton,
|
||||||
side: DoubleSide,
|
|
||||||
alphaTest: 0.1,
|
|
||||||
transparent: true,
|
|
||||||
})
|
|
||||||
: new MeshLambertMaterial({
|
|
||||||
skinning: has_skeleton,
|
|
||||||
side: DoubleSide,
|
|
||||||
}),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
this.mesh = has_skeleton
|
|
||||||
? create_skinned_mesh(geometry, materials, DEFAULT_SKINNED_MATERIAL)
|
|
||||||
: create_mesh(geometry, materials, DEFAULT_MATERIAL);
|
|
||||||
|
|
||||||
// Make sure we rotate around the center of the model instead of its origin.
|
// Make sure we rotate around the center of the model instead of its origin.
|
||||||
const bb = geometry.boundingBox;
|
const bb = geometry.boundingBox;
|
||||||
const height = bb.max.y - bb.min.y;
|
const height = bb.max.y - bb.min.y;
|
||||||
|
Loading…
Reference in New Issue
Block a user