AFS archives with compressed XVM texture archives are now supported.

This commit is contained in:
Daan Vanden Bosch 2020-01-06 14:04:05 +01:00
parent 3edc9b857d
commit 7f5accf790
8 changed files with 31 additions and 11 deletions

View File

@ -26,7 +26,7 @@ export function parse_afs(cursor: Cursor): ArrayBuffer[] {
const file_count = cursor.u16(); const file_count = cursor.u16();
// Skip two unused bytes. // Skip two unused bytes (are these just part of the file count field?).
cursor.seek(2); cursor.seek(2);
const file_entries: AfsFileEntry[] = []; const file_entries: AfsFileEntry[] = [];

View File

@ -43,7 +43,7 @@ export function parse_xvr(cursor: Cursor): XvrTexture {
}; };
} }
export function parse_xvm(cursor: Cursor): Xvm { export function parse_xvm(cursor: Cursor): Xvm | undefined {
const chunks = parse_iff(cursor); const chunks = parse_iff(cursor);
const header_chunk = chunks.find(chunk => chunk.type === XVMH); const header_chunk = chunks.find(chunk => chunk.type === XVMH);
const header = header_chunk && parse_header(header_chunk.data); const header = header_chunk && parse_header(header_chunk.data);
@ -52,6 +52,10 @@ export function parse_xvm(cursor: Cursor): Xvm {
.filter(chunk => chunk.type === XVRT) .filter(chunk => chunk.type === XVRT)
.map(chunk => parse_xvr(chunk.data)); .map(chunk => parse_xvr(chunk.data));
if (!header && textures.length === 0) {
return undefined;
}
if (header && header.texture_count !== textures.length) { if (header && header.texture_count !== textures.length) {
logger.warn( logger.warn(
`Found ${textures.length} textures instead of ${header.texture_count} as defined in the header.`, `Found ${textures.length} textures instead of ${header.texture_count} as defined in the header.`,

View File

@ -1,7 +1,7 @@
import { input } from "./gui/dom"; import { input } from "./gui/dom";
export function open_files(options?: { accept?: string; multiple?: boolean }): Promise<File[]> { export function open_files(options?: { accept?: string; multiple?: boolean }): Promise<File[]> {
return new Promise((resolve) => { return new Promise(resolve => {
const el = input({ type: "file" }); const el = input({ type: "file" });
el.accept = options?.accept ?? ""; el.accept = options?.accept ?? "";
el.multiple = options?.multiple ?? false; el.multiple = options?.multiple ?? false;

View File

@ -79,7 +79,7 @@ export class EntityAssetLoader implements Disposable {
.then(({ data }) => { .then(({ data }) => {
const cursor = new ArrayBufferCursor(data, Endianness.Little); const cursor = new ArrayBufferCursor(data, Endianness.Little);
const xvm = parse_xvm(cursor); const xvm = parse_xvm(cursor);
return xvm_to_textures(xvm); return xvm === undefined ? [] : xvm_to_textures(xvm);
}) })
.catch(e => { .catch(e => {
logger.warn( logger.warn(

View File

@ -9,6 +9,7 @@ import { LogManager } from "../../core/Logger";
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty"; import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
import { list_property } from "../../core/observable"; import { list_property } from "../../core/observable";
import { ListProperty } from "../../core/observable/property/list/ListProperty"; import { ListProperty } from "../../core/observable/property/list/ListProperty";
import { prs_decompress } from "../../core/data_formats/compression/prs/decompress";
const logger = LogManager.get("viewer/controllers/TextureController"); const logger = LogManager.get("viewer/controllers/TextureController");
@ -24,14 +25,26 @@ export class TextureController extends Controller {
if (ext === "xvm") { if (ext === "xvm") {
const xvm = parse_xvm(new ArrayBufferCursor(buffer, Endianness.Little)); const xvm = parse_xvm(new ArrayBufferCursor(buffer, Endianness.Little));
this._textures.splice(0, Infinity, ...xvm.textures); if (xvm) {
this._textures.splice(0, Infinity, ...xvm.textures);
}
} else if (ext === "afs") { } else if (ext === "afs") {
const afs = parse_afs(new ArrayBufferCursor(buffer, Endianness.Little)); const afs = parse_afs(new ArrayBufferCursor(buffer, Endianness.Little));
const textures: XvrTexture[] = []; const textures: XvrTexture[] = [];
for (const buffer of afs) { for (const buffer of afs) {
const xvm = parse_xvm(new ArrayBufferCursor(buffer, Endianness.Little)); const cursor = new ArrayBufferCursor(buffer, Endianness.Little);
textures.push(...xvm.textures); const xvm = parse_xvm(cursor);
if (xvm) {
textures.push(...xvm.textures);
} else {
const xvm = parse_xvm(prs_decompress(cursor.seek_start(0)));
if (xvm) {
textures.push(...xvm.textures);
}
}
} }
this._textures.val = textures; this._textures.val = textures;

View File

@ -70,11 +70,12 @@ export class ModelToolBarController extends Controller {
this.store.set_current_nj_motion(parse_njm(cursor, nj_object.bone_count())); this.store.set_current_nj_motion(parse_njm(cursor, nj_object.bone_count()));
} }
} else if (file.name.endsWith(".xvm")) { } else if (file.name.endsWith(".xvm")) {
this.store.set_current_textures(parse_xvm(cursor).textures); this.store.set_current_textures(parse_xvm(cursor)?.textures ?? []);
} else if (file.name.endsWith(".afs")) { } else if (file.name.endsWith(".afs")) {
const files = parse_afs(cursor); const files = parse_afs(cursor);
const textures: XvrTexture[] = files.flatMap( const textures: XvrTexture[] = files.flatMap(
file => parse_xvm(new ArrayBufferCursor(file, Endianness.Little)).textures, file =>
parse_xvm(new ArrayBufferCursor(file, Endianness.Little))?.textures ?? [],
); );
this.store.set_current_textures(textures); this.store.set_current_textures(textures);

View File

@ -173,7 +173,10 @@ export class CharacterClassAssetLoader implements Disposable {
for (const file of afs) { for (const file of afs) {
const xvm = parse_xvm(new ArrayBufferCursor(file, Endianness.Little)); const xvm = parse_xvm(new ArrayBufferCursor(file, Endianness.Little));
textures.push(...xvm.textures);
if (xvm) {
textures.push(...xvm.textures);
}
} }
return textures; return textures;

View File

@ -38,7 +38,6 @@ export class TextureRenderer extends Renderer implements Disposable {
); );
this.init_camera_controls(); this.init_camera_controls();
this.controls.dollySpeed = -1;
this.controls.azimuthRotateSpeed = 0; this.controls.azimuthRotateSpeed = 0;
this.controls.polarRotateSpeed = 0; this.controls.polarRotateSpeed = 0;
} }