mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
119 lines
3.9 KiB
TypeScript
119 lines
3.9 KiB
TypeScript
import { Disposer } from "../../core/observable/Disposer";
|
|
import { LogManager } from "../../core/Logger";
|
|
import { TextureController } from "../controllers/texture/TextureController";
|
|
import { XvrTexture } from "../../core/data_formats/parsing/ninja/texture";
|
|
import { xvr_texture_to_three_texture } from "../../core/rendering/conversion/ninja_textures";
|
|
import {
|
|
Mesh,
|
|
MeshBasicMaterial,
|
|
OrthographicCamera,
|
|
PlaneGeometry,
|
|
Texture,
|
|
Vector2,
|
|
Vector3,
|
|
} from "three";
|
|
import { DisposableThreeRenderer, Renderer } from "../../core/rendering/Renderer";
|
|
import { Disposable } from "../../core/observable/Disposable";
|
|
|
|
const logger = LogManager.get("viewer/rendering/TextureRenderer");
|
|
|
|
const CAMERA_POSITION = Object.freeze(new Vector3(0, 0, 5));
|
|
const CAMERA_LOOK_AT = Object.freeze(new Vector3(0, 0, 0));
|
|
|
|
export class TextureRenderer extends Renderer implements Disposable {
|
|
private readonly disposer = new Disposer();
|
|
private readonly quad_meshes: Mesh[] = [];
|
|
|
|
readonly camera = new OrthographicCamera(-400, 400, 300, -300, 1, 10);
|
|
|
|
constructor(ctrl: TextureController, three_renderer: DisposableThreeRenderer) {
|
|
super(three_renderer);
|
|
|
|
this.disposer.add_all(
|
|
ctrl.textures.observe(({ value: textures }) => {
|
|
this.scene.remove(...this.quad_meshes);
|
|
|
|
this.create_quads(textures);
|
|
|
|
this.reset_camera(CAMERA_POSITION, CAMERA_LOOK_AT);
|
|
this.schedule_render();
|
|
}),
|
|
);
|
|
|
|
this.init_camera_controls();
|
|
this.controls.azimuthRotateSpeed = 0;
|
|
this.controls.polarRotateSpeed = 0;
|
|
}
|
|
|
|
set_size(width: number, height: number): void {
|
|
this.camera.left = -Math.floor(width / 2);
|
|
this.camera.right = Math.ceil(width / 2);
|
|
this.camera.top = Math.floor(height / 2);
|
|
this.camera.bottom = -Math.ceil(height / 2);
|
|
this.camera.updateProjectionMatrix();
|
|
super.set_size(width, height);
|
|
}
|
|
|
|
dispose(): void {
|
|
super.dispose();
|
|
this.disposer.dispose();
|
|
}
|
|
|
|
private create_quads(textures: readonly XvrTexture[]): void {
|
|
let total_width = 10 * (textures.length - 1); // 10px spacing between textures.
|
|
let total_height = 0;
|
|
|
|
for (const tex of textures) {
|
|
total_width += tex.width;
|
|
total_height = Math.max(total_height, tex.height);
|
|
}
|
|
|
|
let x = -Math.floor(total_width / 2);
|
|
const y = -Math.floor(total_height / 2);
|
|
|
|
for (const tex of textures) {
|
|
let texture: Texture | undefined = undefined;
|
|
|
|
try {
|
|
texture = xvr_texture_to_three_texture(tex);
|
|
} catch (e) {
|
|
logger.error("Couldn't convert XVR texture.", e);
|
|
}
|
|
|
|
const quad_mesh = new Mesh(
|
|
this.create_quad(
|
|
x,
|
|
y + Math.floor((total_height - tex.height) / 2),
|
|
tex.width,
|
|
tex.height,
|
|
),
|
|
texture
|
|
? new MeshBasicMaterial({
|
|
map: texture,
|
|
transparent: true,
|
|
})
|
|
: new MeshBasicMaterial({
|
|
color: 0xff00ff,
|
|
}),
|
|
);
|
|
|
|
this.quad_meshes.push(quad_mesh);
|
|
this.scene.add(quad_mesh);
|
|
|
|
x += 10 + tex.width;
|
|
}
|
|
}
|
|
|
|
private create_quad(x: number, y: number, width: number, height: number): PlaneGeometry {
|
|
const quad = new PlaneGeometry(width, height, 1, 1);
|
|
quad.faceVertexUvs = [
|
|
[
|
|
[new Vector2(0, 0), new Vector2(0, 1), new Vector2(1, 0)],
|
|
[new Vector2(0, 1), new Vector2(1, 1), new Vector2(1, 0)],
|
|
],
|
|
];
|
|
quad.translate(x + width / 2, y + height / 2, -5);
|
|
return quad;
|
|
}
|
|
}
|