Files
GDevelop/GDJS/Runtime/pixi-renderers/pixi-image-manager.ts

630 lines
23 KiB
TypeScript

/*
* GDevelop JS Platform
* Copyright 2013-2016 Florian Rival (Florian.Rival@gmail.com). All rights reserved.
* This project is released under the MIT License.
*/
namespace gdjs {
const logger = new gdjs.Logger('PIXI Image manager');
const logFileLoadingError = (file: string, error: Error | undefined) => {
logger.error(
'Unable to load file ' + file + ' with error:',
error ? error : '(unknown error)'
);
};
const applyTextureSettings = (
texture: PIXI.Texture | undefined,
resourceData: ResourceData
) => {
if (!texture) return;
if (!resourceData.smoothed) {
texture.baseTexture.scaleMode = PIXI.SCALE_MODES.NEAREST;
}
};
const applyThreeTextureSettings = (
threeTexture: THREE.Texture,
resourceData: ResourceData | null
) => {
if (resourceData && !resourceData.smoothed) {
threeTexture.magFilter = THREE.NearestFilter;
threeTexture.minFilter = THREE.NearestFilter;
}
};
const resourceKinds: Array<ResourceKind> = ['image', 'video'];
/**
* PixiImageManager loads and stores textures that can be used by the Pixi.js renderers.
*/
export class PixiImageManager implements gdjs.ResourceManager {
/**
* The invalid texture is a 8x8 PNG file filled with magenta (#ff00ff), to be
* easily spotted if rendered on screen.
*/
private _invalidTexture: PIXI.Texture;
/**
* Map associating a resource name to the loaded PixiJS texture.
*/
private _loadedTextures = new gdjs.ResourceCache<PIXI.Texture>();
/**
* Map associating a resource name to the loaded Three.js texture.
*/
private _loadedThreeTextures: Hashtable<THREE.Texture>;
private _loadedThreeMaterials = new ThreeMaterialCache();
private _diskTextures = new Map<float, PIXI.Texture>();
private _rectangleTextures = new Map<string, PIXI.Texture>();
private _scaledTextures = new Map<string, PIXI.Texture>();
private _resourceLoader: gdjs.ResourceLoader;
/**
* @param resourceLoader The resources loader of the game.
*/
constructor(resourceLoader: gdjs.ResourceLoader) {
this._resourceLoader = resourceLoader;
this._invalidTexture = PIXI.Texture.from(
'',
{ width: 192, height: 192 }
);
this._loadedThreeTextures = new Hashtable();
}
getResourceKinds(): ResourceKind[] {
return resourceKinds;
}
/**
* Return the PIXI texture associated to the specified resource name.
* Returns a placeholder texture if not found.
* @param resourceName The name of the resource
* @returns The requested texture, or a placeholder if not found.
*/
getPIXITexture(resourceName: string): PIXI.Texture {
const resource = this._getImageResource(resourceName);
if (!resource) {
logger.warn(
'Unable to find texture for resource "' + resourceName + '".'
);
return this._invalidTexture;
}
const existingTexture = this._loadedTextures.get(resource);
if (!existingTexture) {
return this._invalidTexture;
}
if (!existingTexture.valid) {
logger.error(
'Texture for ' +
resourceName +
' is not valid anymore (or never was).'
);
return this._invalidTexture;
}
return existingTexture;
}
/**
* Return the PIXI texture associated to the specified resource name.
* If not found in the loaded textures, this method will try to load it.
* Warning: this method should only be used in specific cases that cannot rely on
* the initial resources loading of the game, such as the splashscreen.
* @param resourceName The name of the resource
* @returns The requested texture, or a placeholder if not valid.
*/
getOrLoadPIXITexture(resourceName: string): PIXI.Texture {
const resource = this._getImageResource(resourceName);
if (!resource) {
logger.warn(
'Unable to find texture for resource "' + resourceName + '".'
);
return this._invalidTexture;
}
const existingTexture = this._loadedTextures.get(resource);
if (existingTexture) {
if (existingTexture.valid) {
return existingTexture;
} else {
logger.error(
'Texture for ' +
resourceName +
' is not valid anymore (or never was).'
);
return this._invalidTexture;
}
}
logger.log('Loading texture for resource "' + resourceName + '"...');
const file = resource.file;
const url = this._resourceLoader.getFullUrl(file);
const texture = PIXI.Texture.from(url, {
resourceOptions: {
// Note that using `false`
// to not having `crossorigin` at all would NOT work because the browser would taint the
// loaded resource so that it can't be read/used in a canvas (it's only working for display `<img>` on screen).
crossorigin: this._resourceLoader.checkIfCredentialsRequired(file)
? 'use-credentials'
: 'anonymous',
},
}).on('error', (error) => {
logFileLoadingError(file, error);
});
if (!texture) {
throw new Error(
'Texture loading by PIXI returned nothing for file ' +
file +
' behind url ' +
url
);
}
applyTextureSettings(texture, resource);
this._loadedTextures.set(resource, texture);
return texture;
}
/**
* Return the three.js texture associated to the specified resource name.
* Returns a placeholder texture if not found.
* @param resourceName The name of the resource
* @returns The requested texture, or a placeholder if not found.
*/
getThreeTexture(resourceName: string): THREE.Texture {
const loadedThreeTexture = this._loadedThreeTextures.get(resourceName);
if (loadedThreeTexture) {
return loadedThreeTexture;
}
// Texture is not loaded, load it now from the PixiJS texture.
// TODO (3D) - optimization: don't load the PixiJS Texture if not used by PixiJS.
// TODO (3D) - optimization: Ideally we could even share the same WebGL texture.
const pixiTexture = this.getPIXITexture(resourceName);
const pixiRenderer = this._resourceLoader._runtimeGame
.getRenderer()
.getPIXIRenderer();
if (!pixiRenderer) throw new Error('No PIXI renderer was found.');
// @ts-ignore - source does exist on resource.
const image = pixiTexture.baseTexture.resource.source;
if (!(image instanceof HTMLImageElement)) {
throw new Error(
`Can't load texture for resource "${resourceName}" as it's not an image.`
);
}
const threeTexture = new THREE.Texture(image);
threeTexture.magFilter = THREE.LinearFilter;
threeTexture.minFilter = THREE.LinearFilter;
threeTexture.wrapS = THREE.RepeatWrapping;
threeTexture.wrapT = THREE.RepeatWrapping;
threeTexture.colorSpace = THREE.SRGBColorSpace;
threeTexture.needsUpdate = true;
const resource = this._getImageResource(resourceName);
applyThreeTextureSettings(threeTexture, resource);
this._loadedThreeTextures.put(resourceName, threeTexture);
return threeTexture;
}
/**
* Return the three.js material associated to the specified resource name.
* @param resourceName The name of the resource
* @param options
* @returns The requested material.
*/
getThreeMaterial(
resourceName: string,
options: {
useTransparentTexture: boolean;
forceBasicMaterial: boolean;
vertexColors: boolean;
}
): THREE.Material {
const loadedThreeMaterial = this._loadedThreeMaterials.get(
resourceName,
options
);
if (loadedThreeMaterial) return loadedThreeMaterial;
const material = options.forceBasicMaterial
? new THREE.MeshBasicMaterial({
map: this.getThreeTexture(resourceName),
side: options.useTransparentTexture
? THREE.DoubleSide
: THREE.FrontSide,
transparent: options.useTransparentTexture,
vertexColors: options.vertexColors,
})
: new THREE.MeshStandardMaterial({
map: this.getThreeTexture(resourceName),
side: options.useTransparentTexture
? THREE.DoubleSide
: THREE.FrontSide,
transparent: options.useTransparentTexture,
metalness: 0,
vertexColors: options.vertexColors,
});
this._loadedThreeMaterials.set(resourceName, options, material);
return material;
}
/**
* Return the PIXI video texture associated to the specified resource name.
* Returns a placeholder texture if not found.
* @param resourceName The name of the resource to get.
*/
getPIXIVideoTexture(resourceName: string) {
if (resourceName === '') {
return this._invalidTexture;
}
const resource = this._getImageResource(resourceName);
if (!resource) {
logger.warn(
'Unable to find video texture for resource "' + resourceName + '".'
);
return this._invalidTexture;
}
const texture = this._loadedTextures.get(resource);
if (!texture) {
return this._invalidTexture;
}
return texture;
}
private _getImageResource = (resourceName: string): ResourceData | null => {
const resource = this._resourceLoader.getResource(resourceName);
return resource && this.getResourceKinds().includes(resource.kind)
? resource
: null;
};
/**
* Return a PIXI texture which can be used as a placeholder when no
* suitable texture can be found.
*/
getInvalidPIXITexture() {
return this._invalidTexture;
}
/**
* Load the specified resources, so that textures are loaded and can then be
* used by calling `getPIXITexture`.
*/
async loadResource(resourceName: string): Promise<void> {
const resource = this._resourceLoader.getResource(resourceName);
if (!resource) {
logger.warn(
'Unable to find texture for resource "' + resourceName + '".'
);
return;
}
await this._loadTexture(resource);
}
async processResource(resourceName: string): Promise<void> {
// Do nothing because images are light enough to be parsed in background.
}
/**
* Load the specified resources, so that textures are loaded and can then be
* used by calling `getPIXITexture`.
* @param onProgress Callback called each time a new file is loaded.
*/
async _loadTexture(resource: ResourceData): Promise<void> {
if (this._loadedTextures.get(resource)) {
return;
}
try {
if (resource.kind === 'video') {
// For videos, we want to preload them so they are available as soon as we want to use them.
// We cannot use Pixi.assets.load() as it does not allow passing options (autoplay) to the resource loader.
// Pixi.Texture.from() does not return a promise, so we need to ensure we look at the 'loaded' event of the baseTexture,
// to continue, otherwise if we try to play the video too soon (at the beginning of scene for instance),
// it will fail.
await new Promise<void>((resolve, reject) => {
const texture = PIXI.Texture.from(
this._resourceLoader.getFullUrl(resource.file),
{
resourceOptions: {
crossorigin: this._resourceLoader.checkIfCredentialsRequired(
resource.file
)
? 'use-credentials'
: 'anonymous',
autoPlay: false,
},
}
).on('error', (error) => {
reject(error);
});
const baseTexture = texture.baseTexture;
baseTexture
.on('loaded', () => {
this._loadedTextures.set(resource, texture);
applyTextureSettings(texture, resource);
resolve();
})
.on('error', (error) => {
reject(error);
});
});
} else {
// If the file has no extension, PIXI.assets.load cannot find
// an adequate load parser and does not load the file although
// we would like to force it to load (we are confident it's an image).
// TODO: When PIXI v8+ is used, PIXI.Assets.load can be used because
// loadParser can be forced in PIXI.Assets.load
// (see https://github.com/pixijs/pixijs/blob/71ed56c569ebc6b53da19e3c49258a0a84892101/packages/assets/src/loader/Loader.ts#L68)
const loadedTexture = PIXI.Texture.from(
this._resourceLoader.getFullUrl(resource.file),
{
resourceOptions: {
autoLoad: false,
crossorigin: this._resourceLoader.checkIfCredentialsRequired(
resource.file
)
? 'use-credentials'
: 'anonymous',
},
}
);
await loadedTexture.baseTexture.resource.load();
this._loadedTextures.set(resource, loadedTexture);
// TODO What if 2 assets share the same file with different settings?
applyTextureSettings(loadedTexture, resource);
}
} catch (error) {
logFileLoadingError(resource.file, error);
}
}
/**
* Return a texture containing a circle filled with white.
* @param radius The circle radius
* @param pixiRenderer The renderer used to generate the texture
*/
getOrCreateDiskTexture(
radius: float,
pixiRenderer: PIXI.Renderer
): PIXI.Texture {
let particleTexture = this._diskTextures.get(radius);
if (!particleTexture) {
const graphics = new PIXI.Graphics();
graphics.lineStyle(0, 0, 0);
graphics.beginFill(gdjs.rgbToHexNumber(255, 255, 255), 1);
graphics.drawCircle(0, 0, radius);
graphics.endFill();
particleTexture = pixiRenderer.generateTexture(graphics);
graphics.destroy();
this._diskTextures.set(radius, particleTexture);
}
return particleTexture;
}
/**
* Return a texture filled with white.
* @param width The texture width
* @param height The texture height
* @param pixiRenderer The renderer used to generate the texture
*/
getOrCreateRectangleTexture(
width: float,
height: float,
pixiRenderer: PIXI.Renderer
): PIXI.Texture {
const key = `${width}_${height}`;
let particleTexture = this._rectangleTextures.get(key);
if (!particleTexture) {
const graphics = new PIXI.Graphics();
graphics.lineStyle(0, 0, 0);
graphics.beginFill(gdjs.rgbToHexNumber(255, 255, 255), 1);
graphics.drawRect(0, 0, width, height);
graphics.endFill();
particleTexture = pixiRenderer.generateTexture(graphics);
graphics.destroy();
this._rectangleTextures.set(key, particleTexture);
}
return particleTexture;
}
/**
* Return a texture rescaled according to given dimensions.
* @param width The texture width
* @param height The texture height
* @param pixiRenderer The renderer used to generate the texture
*/
getOrCreateScaledTexture(
imageResourceName: string,
width: float,
height: float,
pixiRenderer: PIXI.Renderer
): PIXI.Texture {
const key = `${imageResourceName}_${width}_${height}`;
let particleTexture = this._scaledTextures.get(key);
if (!particleTexture) {
const graphics = new PIXI.Graphics();
const sprite = new PIXI.Sprite(this.getPIXITexture(imageResourceName));
sprite.width = width;
sprite.height = height;
graphics.addChild(sprite);
particleTexture = pixiRenderer.generateTexture(graphics);
graphics.destroy();
this._scaledTextures.set(key, particleTexture);
}
return particleTexture;
}
/**
* To be called when the game is disposed.
* Clear caches of loaded textures and materials.
*/
dispose(): void {
this._loadedTextures.clear();
const threeTextures: THREE.Texture[] = [];
this._loadedThreeTextures.values(threeTextures);
this._loadedThreeTextures.clear();
for (const threeTexture of threeTextures) {
threeTexture.dispose();
}
this._loadedThreeMaterials.disposeAll();
for (const pixiTexture of this._diskTextures.values()) {
if (pixiTexture.destroyed) {
continue;
}
pixiTexture.destroy();
}
this._diskTextures.clear();
for (const pixiTexture of this._rectangleTextures.values()) {
if (pixiTexture.destroyed) {
continue;
}
pixiTexture.destroy();
}
this._rectangleTextures.clear();
for (const pixiTexture of this._scaledTextures.values()) {
if (pixiTexture.destroyed) {
continue;
}
pixiTexture.destroy();
}
this._scaledTextures.clear();
}
unloadResource(resourceData: ResourceData): void {
const resourceName = resourceData.name;
const texture = this._loadedTextures.getFromName(resourceName);
if (texture) {
texture.destroy(true);
this._loadedTextures.delete(resourceData);
}
const threeTexture = this._loadedThreeTextures.get(resourceName);
if (threeTexture) {
threeTexture.dispose();
this._loadedThreeTextures.remove(resourceName);
}
this._loadedThreeMaterials.dispose(resourceName);
}
}
class ThreeMaterialCache {
private _flaggedMaterials = new Map<string, THREE.Material>();
private _materialFlaggedKeys = new Map<string, Array<string>>();
/**
* Return the three.js material associated to the specified resource name
* and options.
* @param resourceName The name of the resource
* @param options
* @returns The requested material.
*/
get(
resourceName: string,
{
useTransparentTexture,
forceBasicMaterial,
vertexColors,
}: {
useTransparentTexture: boolean;
forceBasicMaterial: boolean;
vertexColors: boolean;
}
): THREE.Material | null {
const flaggedKey = `${resourceName}|${useTransparentTexture ? 1 : 0}|${
forceBasicMaterial ? 1 : 0
}|${vertexColors ? 1 : 0}`;
return this._flaggedMaterials.get(flaggedKey) || null;
}
/**
* Set the three.js material associated to the specified resource name
* and options.
* @param resourceName The name of the resource
* @param options
* @param material The material to add to the cache
*/
set(
resourceName: string,
{
useTransparentTexture,
forceBasicMaterial,
vertexColors,
}: {
useTransparentTexture: boolean;
forceBasicMaterial: boolean;
vertexColors: boolean;
},
material: THREE.Material
): void {
const cacheKey = `${resourceName}|${useTransparentTexture ? 1 : 0}|${
forceBasicMaterial ? 1 : 0
}|${vertexColors ? 1 : 0}`;
this._flaggedMaterials.set(cacheKey, material);
let flaggedKeys = this._materialFlaggedKeys.get(resourceName);
if (!flaggedKeys) {
flaggedKeys = [];
this._materialFlaggedKeys.set(resourceName, flaggedKeys);
}
flaggedKeys.push(cacheKey);
}
/**
* Delete and dispose all the three.js material associated to the specified
* resource name.
* @param resourceName The name of the resource
*/
dispose(resourceName: string): void {
const flaggedKeys = this._materialFlaggedKeys.get(resourceName);
if (flaggedKeys) {
for (const flaggedKey of flaggedKeys) {
const threeMaterial = this._flaggedMaterials.get(flaggedKey);
if (threeMaterial) {
threeMaterial.dispose();
}
this._flaggedMaterials.delete(flaggedKey);
}
}
this._materialFlaggedKeys.delete(resourceName);
}
/**
* Delete and dispose all the three.js material in the cache.
*/
disposeAll(): void {
for (const material of this._flaggedMaterials.values()) {
material.dispose();
}
this._flaggedMaterials.clear();
this._materialFlaggedKeys.clear();
}
}
//Register the class to let the engine use it.
export const ImageManager = gdjs.PixiImageManager;
export type ImageManager = gdjs.PixiImageManager;
}