Objects are now rendered with textures in the quest editor.

This commit is contained in:
Daan Vanden Bosch 2019-07-13 18:12:09 +02:00
parent 60dcbae9b4
commit 7c85081bb0
4 changed files with 62 additions and 27 deletions

View File

@ -16,8 +16,12 @@ export const NPC_COLOR = 0xff0000;
export const NPC_HOVER_COLOR = 0xff3f5f; export const NPC_HOVER_COLOR = 0xff3f5f;
export const NPC_SELECTED_COLOR = 0xff0054; export const NPC_SELECTED_COLOR = 0xff0054;
export function create_object_mesh(object: QuestObject, geometry: BufferGeometry): Mesh { export function create_object_mesh(
return create_mesh(object, geometry, [], OBJECT_COLOR, "Object"); object: QuestObject,
geometry: BufferGeometry,
textures: Texture[]
): Mesh {
return create_mesh(object, geometry, textures, OBJECT_COLOR, "Object");
} }
export function create_npc_mesh( export function create_npc_mesh(

View File

@ -33,6 +33,9 @@ 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); object_cache.set(ObjectType.Unknown, DEFAULT_ENTITY_PROMISE);
const object_tex_cache: Map<ObjectType, Promise<Texture[]>> = new Map();
object_tex_cache.set(ObjectType.Unknown, 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> {
let mesh = npc_cache.get(npc_type); let mesh = npc_cache.get(npc_type);
@ -90,7 +93,7 @@ class EntityStore {
if (geometry) { if (geometry) {
return geometry; return geometry;
} else { } else {
geometry = get_object_data(object_type) geometry = get_object_data(object_type, AssetType.Geometry)
.then(({ url, data }) => { .then(({ url, data }) => {
const cursor = new ArrayBufferCursor(data, Endianness.Little); const cursor = new ArrayBufferCursor(data, Endianness.Little);
const nj_objects = url.endsWith(".nj") ? parse_nj(cursor) : parse_xj(cursor); const nj_objects = url.endsWith(".nj") ? parse_nj(cursor) : parse_xj(cursor);
@ -111,6 +114,28 @@ class EntityStore {
return geometry; return geometry;
} }
} }
async get_object_tex(object_type: ObjectType): Promise<Texture[]> {
let tex = object_tex_cache.get(object_type);
if (tex) {
return tex;
} else {
tex = get_object_data(object_type, AssetType.Texture)
.then(({ data }) => {
const cursor = new ArrayBufferCursor(data, Endianness.Little);
const xvm = parse_xvm(cursor);
return xvm_to_textures(xvm);
})
.catch(e => {
logger.warn(`Could load texture file for ${object_type.name}.`, e);
return DEFAULT_ENTITY_TEX;
});
object_tex_cache.set(object_type, tex);
return tex;
}
}
} }
export const entity_store = new EntityStore(); export const entity_store = new EntityStore();

View File

@ -70,8 +70,9 @@ class QuestEditorStore {
for (const object of quest.objects.filter(o => o.area_id === variant.area.id)) { for (const object of quest.objects.filter(o => o.area_id === variant.area.id)) {
try { try {
const object_geom = await entity_store.get_object_geometry(object.type); const object_geom = await entity_store.get_object_geometry(object.type);
const object_tex = await entity_store.get_object_tex(object.type);
this.set_section_on_visible_quest_entity(object, sections); this.set_section_on_visible_quest_entity(object, sections);
object.object_3d = create_object_mesh(object, object_geom); object.object_3d = create_object_mesh(object, object_geom, object_tex);
} catch (e) { } catch (e) {
logger.error(e); logger.error(e);
} }

View File

@ -31,9 +31,10 @@ export async function get_npc_data(
} }
export async function get_object_data( export async function get_object_data(
object_type: ObjectType object_type: ObjectType,
type: AssetType
): Promise<{ url: string; data: ArrayBuffer }> { ): Promise<{ url: string; data: ArrayBuffer }> {
const url = object_type_to_url(object_type); const url = object_type_to_url(object_type, type);
const data = await get_asset(url); const data = await get_asset(url);
return { url, data }; return { url, data };
} }
@ -208,7 +209,8 @@ function npc_type_to_url(npc_type: NpcType, type: AssetType): string {
} }
} }
function object_type_to_url(object_type: ObjectType): string { function object_type_to_url(object_type: ObjectType, type: AssetType): string {
if (type === AssetType.Geometry) {
switch (object_type) { switch (object_type) {
case ObjectType.EasterEgg: case ObjectType.EasterEgg:
case ObjectType.ChristmasTree: case ObjectType.ChristmasTree:
@ -231,6 +233,9 @@ function object_type_to_url(object_type: ObjectType): string {
default: default:
return `/objects/${object_type.pso_id}.xj`; return `/objects/${object_type.pso_id}.xj`;
} }
} else {
return `/objects/${object_type.pso_id}.xvm`;
}
} }
function player_class_to_url(player_class: string, body_part: string, no?: number): string { function player_class_to_url(player_class: string, body_part: string, no?: number): string {