Fixed issues with textures and tweaked lighting and ground color in quest renderer.

This commit is contained in:
Daan Vanden Bosch 2019-07-13 21:48:18 +02:00
parent f23fe0091b
commit 769e0e8377
10 changed files with 86 additions and 35 deletions

View File

@ -62,9 +62,8 @@ export function parse_xj_model(cursor: Cursor): XjModel {
} }
cursor.seek_start(vertex_info_table_offset); cursor.seek_start(vertex_info_table_offset);
cursor.seek(4); // Vertex type. const vertex_type = cursor.u16();
// Vertex Types cursor.seek(2); // Flags?
// 3) size: 32, has position, normal, uv
const vertex_table_offset = cursor.u32(); const vertex_table_offset = cursor.u32();
const vertex_size = cursor.u32(); const vertex_size = cursor.u32();
const vertex_count = cursor.u32(); const vertex_count = cursor.u32();
@ -76,13 +75,38 @@ export function parse_xj_model(cursor: Cursor): XjModel {
let normal: Vec3 | undefined; let normal: Vec3 | undefined;
let uv: Vec2 | undefined; let uv: Vec2 | undefined;
if (vertex_size === 28 || vertex_size === 32 || vertex_size === 36) { switch (vertex_type) {
case 3:
normal = cursor.vec3_f32(); normal = cursor.vec3_f32();
uv = cursor.vec2_f32();
break;
case 4:
// Skip 4 bytes.
break;
case 5:
cursor.seek(4);
uv = cursor.vec2_f32();
break;
case 6:
normal = cursor.vec3_f32();
// Skip 4 bytes.
break;
case 7:
normal = cursor.vec3_f32();
uv = cursor.vec2_f32();
break;
default:
logger.warn(`Unknown vertex type ${vertex_type} with size ${vertex_size}.`);
break;
} }
if (vertex_size === 24 || vertex_size === 32 || vertex_size === 36) { // if (vertex_size === 28 || vertex_size === 32 || vertex_size === 36) {
uv = cursor.vec2_f32(); // normal = cursor.vec3_f32();
} // }
// if (vertex_size === 24 || vertex_size === 32 || vertex_size === 36) {
// uv = cursor.vec2_f32();
// }
model.vertices.push({ model.vertices.push({
position, position,

View File

@ -22,7 +22,7 @@ export class Renderer<C extends Camera> {
private renderer = new WebGLRenderer({ antialias: true }); private renderer = new WebGLRenderer({ antialias: true });
private render_scheduled = false; private render_scheduled = false;
private light = new HemisphereLight(0xffffff, 0x505050, 1); private light = new HemisphereLight(0xffffff, 0x505050, 1.2);
constructor(camera: C) { constructor(camera: C) {
this.renderer.setPixelRatio(window.devicePixelRatio); this.renderer.setPixelRatio(window.devicePixelRatio);

View File

@ -28,7 +28,7 @@ const materials = [
}), }),
// Ground // Ground
new MeshLambertMaterial({ new MeshLambertMaterial({
color: 0x50d0d0, color: 0xa0a0a0,
side: DoubleSide, side: DoubleSide,
}), }),
// Vegetation // Vegetation
@ -53,17 +53,17 @@ const wireframe_materials = [
}), }),
// Ground // Ground
new MeshBasicMaterial({ new MeshBasicMaterial({
color: 0x60f0f0, color: 0xd0d0d0,
wireframe: true, wireframe: true,
}), }),
// Vegetation // Vegetation
new MeshBasicMaterial({ new MeshBasicMaterial({
color: 0x60c080, color: 0x80e0a0,
wireframe: true, wireframe: true,
}), }),
// Section transition zone // Section transition zone
new MeshBasicMaterial({ new MeshBasicMaterial({
color: 0x705090, color: 0x9070b0,
wireframe: true, wireframe: true,
}), }),
]; ];

View File

@ -44,6 +44,7 @@ function create_mesh(
const materials: Material[] = [ const materials: Material[] = [
new MeshBasicMaterial({ new MeshBasicMaterial({
color, color,
side: DoubleSide,
}), }),
]; ];
@ -51,7 +52,6 @@ function create_mesh(
...textures.map( ...textures.map(
tex => tex =>
new MeshLambertMaterial({ new MeshLambertMaterial({
skinning: true,
map: tex, map: tex,
side: DoubleSide, side: DoubleSide,
alphaTest: 0.5, alphaTest: 0.5,

View File

@ -127,6 +127,7 @@ class Object3DCreator {
geom.addGroup(group.start, group.count, group.material_index); geom.addGroup(group.start, group.count, group.material_index);
} }
geom.computeBoundingSphere();
geom.computeBoundingBox(); geom.computeBoundingBox();
return geom; return geom;

View File

@ -5,6 +5,7 @@ import {
RGBA_S3TC_DXT1_Format, RGBA_S3TC_DXT1_Format,
RGBA_S3TC_DXT3_Format, RGBA_S3TC_DXT3_Format,
Texture, Texture,
CompressedPixelFormat,
} from "three"; } from "three";
import { Xvm, XvmTexture } from "../data_formats/parsing/ninja/texture"; import { Xvm, XvmTexture } from "../data_formats/parsing/ninja/texture";
@ -13,33 +14,36 @@ export function xvm_to_textures(xvm: Xvm): Texture[] {
} }
export function xvm_texture_to_texture(tex: XvmTexture): Texture { export function xvm_texture_to_texture(tex: XvmTexture): Texture {
let format: CompressedPixelFormat;
let data_size: number;
// Ignore mipmaps.
switch (tex.format[1]) {
case 6:
format = RGBA_S3TC_DXT1_Format;
data_size = (tex.width * tex.height) / 2;
break;
case 7:
format = RGBA_S3TC_DXT3_Format;
data_size = tex.width * tex.height;
break;
default:
throw new Error(`Format ${tex.format.join(", ")} not supported.`);
}
const texture_3js = new CompressedTexture( const texture_3js = new CompressedTexture(
[ [
{ {
data: new Uint8Array(tex.data) as any, data: new Uint8Array(tex.data, 0, data_size) as any,
width: tex.width, width: tex.width,
height: tex.height, height: tex.height,
}, },
], ],
tex.width, tex.width,
tex.height tex.height,
format
); );
switch (tex.format[1]) {
case 6:
texture_3js.format = RGBA_S3TC_DXT1_Format as any;
break;
case 7:
if (tex.format[0] === 2) {
texture_3js.format = RGBA_S3TC_DXT3_Format as any;
} else {
throw new Error(`Format[0] ${tex.format[0]} not supported.`);
}
break;
default:
throw new Error(`Format[1] ${tex.format[1]} not supported.`);
}
texture_3js.minFilter = LinearFilter; texture_3js.minFilter = LinearFilter;
texture_3js.wrapS = MirroredRepeatWrapping; texture_3js.wrapS = MirroredRepeatWrapping;
texture_3js.wrapT = MirroredRepeatWrapping; texture_3js.wrapT = MirroredRepeatWrapping;

View File

@ -31,10 +31,32 @@ const npc_tex_cache: Map<NpcType, Promise<Texture[]>> = new Map();
npc_tex_cache.set(NpcType.Unknown, DEFAULT_ENTITY_TEX_PROMISE); npc_tex_cache.set(NpcType.Unknown, DEFAULT_ENTITY_TEX_PROMISE);
const object_cache: Map<ObjectType, Promise<BufferGeometry>> = new Map(); const object_cache: Map<ObjectType, Promise<BufferGeometry>> = new Map();
object_cache.set(ObjectType.Unknown, DEFAULT_ENTITY_PROMISE);
const object_tex_cache: Map<ObjectType, Promise<Texture[]>> = new Map(); const object_tex_cache: Map<ObjectType, Promise<Texture[]>> = new Map();
object_tex_cache.set(ObjectType.Unknown, DEFAULT_ENTITY_TEX_PROMISE);
for (const type of [
ObjectType.Unknown,
ObjectType.PlayerSet,
ObjectType.FogCollision,
ObjectType.EventCollision,
ObjectType.ObjRoomID,
ObjectType.ScriptCollision,
ObjectType.ItemLight,
ObjectType.FogCollisionSW,
ObjectType.MenuActivation,
ObjectType.BoxDetectObject,
ObjectType.SymbolChatObject,
ObjectType.TouchPlateObject,
ObjectType.TargetableObject,
ObjectType.EffectObject,
ObjectType.CountDownObject,
ObjectType.TelepipeLocation,
ObjectType.Pioneer2InvisibleTouchplate,
ObjectType.TempleMapDetect,
ObjectType.LabInvisibleObject,
]) {
object_cache.set(type, DEFAULT_ENTITY_PROMISE);
object_tex_cache.set(type, DEFAULT_ENTITY_TEX_PROMISE);
}
class EntityStore { class EntityStore {
async get_npc_geometry(npc_type: NpcType): Promise<BufferGeometry> { async get_npc_geometry(npc_type: NpcType): Promise<BufferGeometry> {

View File

@ -297,7 +297,7 @@ class ModelViewerStore {
textures.map( textures.map(
tex => tex =>
new MeshLambertMaterial({ new MeshLambertMaterial({
skinning: true, skinning: this.has_skeleton,
map: tex, map: tex,
side: DoubleSide, side: DoubleSide,
alphaTest: 0.5, alphaTest: 0.5,