-
-
Notifications
You must be signed in to change notification settings - Fork 151
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[@thi.ng/webgl] Best practice to save screenshot #204
Comments
Hi @nkint - instead of having to resize the mounted onscreen canvas and re-initialize the whole scene, you could just use an FBO, bind it, render to it, then read its pixels (via Also, I don't understand the need for using canvas.toBlob((blob) => download("image.png", blob), "image/png"); |
Hi @postspectacular ! Thank you for the answer. So I made a quick trial... It seems to work well... but: I'm posting the code here as a note for future investigations: import {
TextureOpts,
RBO,
texture,
fbo,
RboOpts,
FboOpts,
rbo,
readPixels,
TextureFormat,
} from '@thi.ng/webgl';
import { isBoolean, isNil } from '@thi.ng/checks';
import { download } from '../download';
export const createWebglScreenShot = (
gl: WebGLRenderingContext,
width: number,
height: number,
fileName: string = 'image.png',
optsColorTexture?: Partial<TextureOpts>,
depthBuffer: RBO | RboOpts | boolean = true,
) => {
const optsTexture: Partial<TextureOpts> = {
width,
height,
image: null,
filter: gl.LINEAR,
wrap: gl.CLAMP_TO_EDGE,
...optsColorTexture,
};
const colorTexture = texture(gl, optsTexture);
const optsFbo: Partial<FboOpts> = { tex: [colorTexture] };
if (isNil(depthBuffer)) {
// doing nothing
} else if (depthBuffer instanceof RBO) {
optsFbo.depth = depthBuffer;
} else if (isBoolean(depthBuffer)) {
optsFbo.depth = rbo(gl, { width, height });
} else if (depthBuffer.width && depthBuffer.height) {
optsFbo.depth = rbo(gl, depthBuffer);
}
const frameBuffer = fbo(gl, optsFbo);
frameBuffer.unbind();
return {
capture(draw: () => void) {
const canvas = document.createElement('canvas');
canvas.width = width;
canvas.height = height;
const context = canvas.getContext('2d');
const imageData = context.getImageData(0, 0, width, height);
frameBuffer.bind();
const prevViewport = gl.getParameter(gl.VIEWPORT);
gl.viewport(0, 0, width, height);
draw();
const buffer = new Uint8Array(width * height * 4);
readPixels(gl, 0, 0, width, height, TextureFormat.RGBA, colorTexture.type, buffer);
imageData.data.set(buffer);
context.putImageData(imageData, 0, 0);
canvas.toBlob(blob => download(fileName, blob), 'image/png');
// frameBuffer.release();
// colorTexture.release();
frameBuffer.unbind();
gl.viewport(...(prevViewport as [number, number, number, number]));
},
};
};
export type WebglScreenShot = ReturnType<typeof createWebglScreenShot>; |
Ciao @nkint - haven't tried this myself yet, but if you're using WebGL2 you could try adding this manual RBO multi-sample init/config step before initializing the actual FBO: gl.renderbufferStorageMultisample(gl.RENDERBUFFER, gl.getParameter(gl.MAX_SAMPLES), gl.RGBA8, width, height); |
Hi! I'm looking for some best practice to save screenshot of any desired size.
Right now I'm relying on a global variable that during the update push a new canvas dimension/aspect ratio/viewport, save the screenshot, pop back the dimensions.
Honestly I don't like that approach, or at least I'm looking for something more modular with less boilerplate to easily use it in other project.
Any help/ideas are super usefull! 💡
The text was updated successfully, but these errors were encountered: