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();
// Skip two unused bytes.
// Skip two unused bytes (are these just part of the file count field?).
cursor.seek(2);
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 header_chunk = chunks.find(chunk => chunk.type === XVMH);
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)
.map(chunk => parse_xvr(chunk.data));
if (!header && textures.length === 0) {
return undefined;
}
if (header && header.texture_count !== textures.length) {
logger.warn(
`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";
export function open_files(options?: { accept?: string; multiple?: boolean }): Promise<File[]> {
return new Promise((resolve) => {
return new Promise(resolve => {
const el = input({ type: "file" });
el.accept = options?.accept ?? "";
el.multiple = options?.multiple ?? false;

View File

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

View File

@ -9,6 +9,7 @@ import { LogManager } from "../../core/Logger";
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
import { list_property } from "../../core/observable";
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");
@ -24,14 +25,26 @@ export class TextureController extends Controller {
if (ext === "xvm") {
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") {
const afs = parse_afs(new ArrayBufferCursor(buffer, Endianness.Little));
const textures: XvrTexture[] = [];
for (const buffer of afs) {
const xvm = parse_xvm(new ArrayBufferCursor(buffer, Endianness.Little));
textures.push(...xvm.textures);
const cursor = new ArrayBufferCursor(buffer, Endianness.Little);
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;

View File

@ -70,11 +70,12 @@ export class ModelToolBarController extends Controller {
this.store.set_current_nj_motion(parse_njm(cursor, nj_object.bone_count()));
}
} 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")) {
const files = parse_afs(cursor);
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);

View File

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

View File

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