UNPKG

playcanvas

Version:

PlayCanvas WebGL game engine

125 lines (122 loc) 5.54 kB
import { ShaderUtils } from '../../scene/shader-lib/shader-utils.js'; import { Texture } from '../../platform/graphics/texture.js'; import { BlendState } from '../../platform/graphics/blend-state.js'; import { drawQuadWithShader } from '../../scene/graphics/quad-render-utils.js'; import { RenderTarget } from '../../platform/graphics/render-target.js'; import { isCompressedPixelFormat, PIXELFORMAT_RGBA8, ADDRESS_CLAMP_TO_EDGE, FILTER_LINEAR, SEMANTIC_POSITION } from '../../platform/graphics/constants.js'; /** * @import { Color } from '../../core/math/color.js' */ /** * The base class for the exporters, implementing shared functionality. * * @category Exporter * @ignore */ class CoreExporter { /** * Create a new instance of the exporter. */ // eslint-disable-next-line no-useless-constructor constructor(){} /** * Converts a texture to a canvas. * * @param {Texture} texture - The source texture to be converted. * @param {object} options - Object for passing optional arguments. * @param {Color} [options.color] - The tint color to modify the texture with. * @param {number} [options.maxTextureSize] - Maximum texture size. Texture is resized if over the size. * @returns {Promise<HTMLCanvasElement>|Promise<undefined>} - The canvas element containing the image. * * @ignore */ textureToCanvas(texture, options = {}) { const image = texture.getSource(); if (typeof HTMLImageElement !== 'undefined' && image instanceof HTMLImageElement || typeof HTMLCanvasElement !== 'undefined' && image instanceof HTMLCanvasElement || typeof OffscreenCanvas !== 'undefined' && image instanceof OffscreenCanvas || typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap) { // texture dimensions const { width, height } = this.calcTextureSize(image.width, image.height, options.maxTextureSize); // convert to a canvas const canvas = document.createElement('canvas'); canvas.width = width; canvas.height = height; const context = canvas.getContext('2d'); if (context === null) { return Promise.resolve(undefined); } context.drawImage(image, 0, 0, canvas.width, canvas.height); // tint the texture by specified color if (options.color) { const { r, g, b } = options.color; const imagedata = context.getImageData(0, 0, width, height); const data = imagedata.data; for(let i = 0; i < data.length; i += 4){ data[i + 0] = data[i + 0] * r; data[i + 1] = data[i + 1] * g; data[i + 2] = data[i + 2] * b; } context.putImageData(imagedata, 0, 0); } return Promise.resolve(canvas); } // for other image sources, for example compressed textures, we extract the data by rendering the texture to a render target const device = texture.device; const { width, height } = this.calcTextureSize(texture.width, texture.height, options.maxTextureSize); const format = isCompressedPixelFormat(texture.format) ? PIXELFORMAT_RGBA8 : texture.format; const dstTexture = new Texture(device, { name: 'ExtractedTexture', width, height, format: format, cubemap: false, mipmaps: false, minFilter: FILTER_LINEAR, magFilter: FILTER_LINEAR, addressU: ADDRESS_CLAMP_TO_EDGE, addressV: ADDRESS_CLAMP_TO_EDGE }); const renderTarget = new RenderTarget({ colorBuffer: dstTexture, depth: false }); const shader = ShaderUtils.createShader(device, { uniqueName: 'ShaderCoreExporterBlit', attributes: { vertex_position: SEMANTIC_POSITION }, vertexChunk: 'fullscreenQuadVS', fragmentChunk: 'outputTex2DPS' }); device.scope.resolve('source').setValue(texture); device.setBlendState(BlendState.NOBLEND); drawQuadWithShader(device, renderTarget, shader); // async read back the pixels of the texture return dstTexture.read(0, 0, width, height, { renderTarget: renderTarget, immediate: true }).then((textureData)=>{ dstTexture.destroy(); renderTarget.destroy(); const pixels = new Uint8ClampedArray(width * height * 4); pixels.set(textureData); // copy pixels to a canvas const newImage = new ImageData(pixels, width, height); const canvas = document.createElement('canvas'); canvas.width = width; canvas.height = height; const newContext = canvas.getContext('2d'); if (!newContext) { return Promise.resolve(undefined); } newContext.putImageData(newImage, 0, 0); return Promise.resolve(canvas); }); } calcTextureSize(width, height, maxTextureSize) { if (maxTextureSize) { const scale = Math.min(maxTextureSize / Math.max(width, height), 1); width = Math.round(width * scale); height = Math.round(height * scale); } return { width, height }; } } export { CoreExporter };