/**
* LittleJS Post Processing Plugin
* - Supports shadertoy style post processing shaders
* - call new new PostProcessPlugin() to setup post processing
* - can be enabled to pass other canvases through a final shader
*/
'use strict';
///////////////////////////////////////////////////////////////////////////////
/** Global Post Process plugin object
* @type {PostProcessPlugin} */
let postProcess;
/////////////////////////////////////////////////////////////////////////
/**
* UI System Global Object
*/
class PostProcessPlugin
{
/** Create global post processing shader
* @param {string} shaderCode
* @param {boolean} [includeOverlay]
* @example
* // create the post process plugin object
* new PostProcessPlugin(shaderCode);
*/
constructor(shaderCode, includeOverlay=false)
{
ASSERT(!postProcess, 'Post process already initialized');
postProcess = this;
if (headlessMode) return;
if (!glEnable)
{
console.warn('PostProcessPlugin: WebGL not enabled!');
return;
}
if (!shaderCode) // default shader pass through
shaderCode = 'void mainImage(out vec4 c,vec2 p){c=texture(iChannel0,p/iResolution.xy);}';
/** @property {WebGLProgram} - Shader for post processing */
this.shader = glCreateProgram(
'#version 300 es\n' + // specify GLSL ES version
'precision highp float;'+ // use highp for better accuracy
'in vec2 p;'+ // position
'void main(){'+ // shader entry point
'gl_Position=vec4(p+p-1.,1,1);'+ // set position
'}' // end of shader
,
'#version 300 es\n' + // specify GLSL ES version
'precision highp float;'+ // use highp for better accuracy
'uniform sampler2D iChannel0;'+ // input texture
'uniform vec3 iResolution;'+ // size of output texture
'uniform float iTime;'+ // time
'out vec4 c;'+ // out color
'\n' + shaderCode + '\n'+ // insert custom shader code
'void main(){'+ // shader entry point
'mainImage(c,gl_FragCoord.xy);'+ // call post process function
'c.a=1.;'+ // always use full alpha
'}' // end of shader
);
/** @property {WebGLTexture} - Texture for post processing */
this.texture = glCreateTexture();
/** @property {boolean} - Should overlay canvas be included in post processing */
this.includeOverlay = includeOverlay;
// Render the post processing shader, called automatically by the engine
engineAddPlugin(undefined, postProcessRender);
function postProcessRender()
{
if (headlessMode) return;
// prepare to render post process shader
if (glEnable)
{
glFlush(); // clear out the buffer
mainContext.drawImage(glCanvas, 0, 0); // copy to the main canvas
}
else
{
// set the viewport
glContext.viewport(0, 0, glCanvas.width = drawCanvas.width, glCanvas.height = drawCanvas.height);
}
if (postProcess.includeOverlay)
{
// copy overlay canvas so it will be included in post processing
mainContext.drawImage(overlayCanvas, 0, 0);
overlayCanvas.width |= 0;
}
// setup shader program to draw one triangle
glContext.useProgram(postProcess.shader);
glContext.bindBuffer(glContext.ARRAY_BUFFER, glGeometryBuffer);
glContext.pixelStorei(glContext.UNPACK_FLIP_Y_WEBGL, 1);
glContext.disable(glContext.BLEND);
// set textures, pass in the 2d canvas and gl canvas in separate texture channels
glContext.activeTexture(glContext.TEXTURE0);
glContext.bindTexture(glContext.TEXTURE_2D, postProcess.texture);
glContext.texImage2D(glContext.TEXTURE_2D, 0, glContext.RGBA, glContext.RGBA, glContext.UNSIGNED_BYTE, mainCanvas);
// set vertex position attribute
const vertexByteStride = 8;
const pLocation = glContext.getAttribLocation(postProcess.shader, 'p');
glContext.enableVertexAttribArray(pLocation);
glContext.vertexAttribPointer(pLocation, 2, glContext.FLOAT, false, vertexByteStride, 0);
// set uniforms and draw
const uniformLocation = (name)=>glContext.getUniformLocation(postProcess.shader, name);
glContext.uniform1i(uniformLocation('iChannel0'), 0);
glContext.uniform1f(uniformLocation('iTime'), time);
glContext.uniform3f(uniformLocation('iResolution'), mainCanvas.width, mainCanvas.height, 1);
glContext.drawArrays(glContext.TRIANGLE_STRIP, 0, 4);
}
}
}