mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
AFS archives with compressed XVM texture archives are now supported.
This commit is contained in:
parent
3edc9b857d
commit
7f5accf790
@ -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[] = [];
|
||||||
|
@ -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.`,
|
||||||
|
@ -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;
|
||||||
|
@ -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(
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user