phaser/src/renderer/webgl/WebGLRenderer.js

397 lines
12 KiB
JavaScript
Raw Normal View History

var Class = require('../../utils/Class');
var CONST = require('../../const');
2018-01-17 21:25:43 +00:00
var WebGLRenderer = new Class({
2016-12-07 02:28:22 +00:00
initialize:
2018-01-17 21:25:43 +00:00
function WebGLRenderer (game)
2016-12-07 02:28:22 +00:00
{
2018-01-17 21:25:43 +00:00
var renderer = this;
var config = {
backgroundColor: game.config.backgroundColor,
contextCreation: {
alpha: false,
depth: false, // enable when 3D is added in the future
antialias: true,
premultipliedAlpha: true,
stencil: true,
preserveDrawingBuffer: false,
failIfMajorPerformanceCaveat: false
}
};
this.game = game;
this.type = CONST.WEBGL;
this.width = game.config.width * game.config.resolution;
this.height = game.config.height * game.config.resolution;
2018-01-17 21:25:43 +00:00
this.canvas = game.canvas;
this.blendModes = [];
this.contextLost = false;
this.autoResize = false;
this.pipelines = null;
2018-01-17 21:25:43 +00:00
for (var i = 0; i <= 16; i++)
{
this.blendModes.push({ func: [ WebGLRenderingContext.ONE, WebGLRenderingContext.ONE_MINUS_SRC_ALPHA ], equation: WebGLRenderingContext.FUNC_ADD });
}
2018-01-17 21:25:43 +00:00
this.blendModes[1].func = [ WebGLRenderingContext.ONE, WebGLRenderingContext.DST_ALPHA ];
this.blendModes[2].func = [ WebGLRenderingContext.DST_COLOR, WebGLRenderingContext.ONE_MINUS_SRC_ALPHA ];
this.blendModes[3].func = [ WebGLRenderingContext.ONE, WebGLRenderingContext.ONE_MINUS_SRC_COLOR ];
2018-01-17 21:25:43 +00:00
// Intenal Renderer State (Textures, Framebuffers, Pipelines, Buffers, etc)
this.currentTextures = new Array(16);
this.currentFramebuffer = null;
this.currentPipeline = null;
this.currentVertexBuffer = null;
this.currentIndexBuffer = null;
this.currentBlendMode = CONST.BlendModes.NORMAL;
this.currentScissorState = { enabled: false, x: 0, y: 0, w: 0, h: 0 };
// Setup context lost and restore event listeners
this.canvas.addEventListener('webglcontextlost', function (event) {
renderer.contextLost = true;
event.preventDefault();
}, false);
2018-01-17 21:25:43 +00:00
this.canvas.addEventListener('webglcontextrestored', function (event) {
renderer.contextLost = false;
renderer.init(config);
}, false);
// This are initialized post context creation
this.gl = null;
2018-01-17 21:25:43 +00:00
this.supportedExtensions = null;
this.extensions = {};
2018-01-17 21:25:43 +00:00
this.init(config);
},
2018-01-17 21:25:43 +00:00
init: function (config)
{
2018-01-17 21:25:43 +00:00
var canvas = this.canvas;
var clearColor = config.backgroundColor;
var gl = canvas.getContext('webgl', config.contextCreation) || canvas.getContext('experimental-webgl', config.contextCreation);
if (!gl)
2016-12-07 02:28:22 +00:00
{
this.contextLost = true;
2018-01-10 20:03:01 +00:00
throw new Error('This browser does not support WebGL. Try using the Canvas pipeline.');
2016-12-07 02:28:22 +00:00
}
2018-01-17 21:25:43 +00:00
// Load supported extensions
this.supportedExtensions = gl.getSupportedExtensions();
2018-01-17 21:25:43 +00:00
// Setup initial WebGL state
2016-12-07 02:28:22 +00:00
gl.disable(gl.DEPTH_TEST);
gl.disable(gl.CULL_FACE);
2018-01-17 21:25:43 +00:00
gl.disable(gl.SCISSOR_TEST);
2016-12-07 02:28:22 +00:00
gl.enable(gl.BLEND);
2018-01-17 21:25:43 +00:00
gl.clearColor(clearColor.redGL, clearColor.greenGL, clearColor.blueGL, 1.0);
2018-01-17 21:25:43 +00:00
// Initialize all textures to null
for (var index = 0; index < this.currentTextures.length; ++index)
{
2018-01-17 21:25:43 +00:00
this.currentTextures[index] = null;
}
2018-01-17 21:25:43 +00:00
// Clear previous pipelines and reload default ones
this.pipelines = {};
2018-01-17 21:25:43 +00:00
this.setBlendMode(CONST.BlendModes.NORMAL);
this.resize(this.width, this.height, this.game.config.resolution);
return this;
},
2018-01-17 21:25:43 +00:00
resize: function (width, height, resolution)
{
2018-01-17 21:25:43 +00:00
var gl = this.gl;
2018-01-17 21:25:43 +00:00
this.width = width * resolution;
this.height = height * resolution;
this.canvas.width = this.width;
this.canvas.height = this.height;
2018-01-17 21:25:43 +00:00
if (this.autoResize)
{
2018-01-17 21:25:43 +00:00
this.canvas.style.width = (this.width / resolution) + 'px';
this.canvas.style.height = (this.height / resolution) + 'px';
}
2018-01-17 21:25:43 +00:00
gl.viewport(0, 0, this.width, this.height);
2018-01-17 21:25:43 +00:00
// Update all registered pipelines
2018-01-17 21:25:43 +00:00
return this;
},
2018-01-17 21:25:43 +00:00
hasExtension: function (extensionName)
{
2018-01-17 21:25:43 +00:00
return this.supportedExtensions ? this.supportedExtensions.indexOf(extensionName) : false;
},
2018-01-17 21:25:43 +00:00
getExtension: function (extensionName)
2017-01-19 23:20:36 +00:00
{
2018-01-17 21:25:43 +00:00
if (!this.hasExtension(extensionName)) return null;
2018-01-17 21:25:43 +00:00
if (!(extensionName in this.extensions))
2017-01-19 23:20:36 +00:00
{
2018-01-17 21:25:43 +00:00
this.extensions[extensionName] = this.gl.getExtension(extensionName);
2017-01-19 23:20:36 +00:00
}
2018-01-17 21:25:43 +00:00
return this.extensions[extensionName];
2017-01-19 23:20:36 +00:00
},
2018-01-17 21:25:43 +00:00
/* Renderer State Manipulation Functions */
2017-01-19 17:53:20 +00:00
2018-01-17 21:25:43 +00:00
hasPipeline: function (pipelineName)
2017-01-19 22:43:41 +00:00
{
2018-01-17 21:25:43 +00:00
return (pipelineName in this.pipelines);
2017-04-07 01:49:15 +00:00
},
2018-01-17 21:25:43 +00:00
getPipeline: function (pipelineName)
2017-04-07 01:49:15 +00:00
{
2018-01-17 21:25:43 +00:00
if (this.hasPipeline(pipelineName)) return this.pipelines[pipelineName];
return null;
2017-01-19 22:43:41 +00:00
},
2018-01-17 21:25:43 +00:00
removePipeline: function (pipelineName)
2016-12-07 02:28:22 +00:00
{
2018-01-17 21:25:43 +00:00
delete this.pipelines[pipelineName];
return this;
2016-12-07 02:28:22 +00:00
},
2018-01-17 21:25:43 +00:00
addPipeline: function (pipelineName, pipelineInstance)
2016-12-07 02:28:22 +00:00
{
2018-01-17 21:25:43 +00:00
if (!this.hasPipeline(pipelineName)) this.pipelines[pipelineName] = pipelineInstance;
else console.warn('Pipeline', pipelineName, ' already exists.');
2018-01-17 21:25:43 +00:00
this.pipelines[pipelineName].resize(this.width, this.height, this.game.config.resolution);
2016-12-07 02:28:22 +00:00
2018-01-17 21:25:43 +00:00
return this;
},
2018-01-17 21:25:43 +00:00
setBlendMode: function (blendModeId)
{
var gl = this.gl;
2018-01-17 21:25:43 +00:00
var pipeline = this.currentPipeline;
var blendMode = this.blendModes[blendModeId];
2017-02-07 16:12:20 +00:00
2018-01-17 21:25:43 +00:00
if (blendModeId === CONST.BlendModes.SKIP_CHECK || !pipeline)
return;
2018-01-17 21:25:43 +00:00
if (this.currentBlendMode !== blendModeId)
2017-01-19 17:53:20 +00:00
{
2018-01-17 21:25:43 +00:00
pipeline.flush();
2017-12-15 04:07:16 +00:00
2018-01-17 21:25:43 +00:00
gl.enable(gl.BLEND);
gl.blendEquation(blendMode.equation);
2018-01-17 21:25:43 +00:00
if (blendMode.func.length > 2)
2017-11-06 22:12:19 +00:00
{
2018-01-17 21:25:43 +00:00
gl.blendFuncSeparate(blendMode.func[0], blendMode.func[1], blendMode.func[2], blendMode.func[3]);
2017-11-06 22:12:19 +00:00
}
2018-01-17 21:25:43 +00:00
else
2017-11-06 22:12:19 +00:00
{
2018-01-17 21:25:43 +00:00
gl.blendFunc(blendMode.func[0], blendMode.func[1]);
2017-11-06 22:12:19 +00:00
}
2018-01-17 21:25:43 +00:00
this.currentBlendMode = blendModeId;
2017-02-10 00:48:32 +00:00
}
2018-01-17 21:25:43 +00:00
return this;
2017-02-07 19:30:50 +00:00
},
2016-12-07 02:28:22 +00:00
2018-01-17 21:25:43 +00:00
setTexture2D: function (texture, textureUnit)
2017-02-07 19:30:50 +00:00
{
2018-01-17 21:25:43 +00:00
var gl = this.gl;
2017-05-16 19:15:01 +00:00
2018-01-17 21:25:43 +00:00
if (texture !== this.currentTextures[textureUnit])
2017-05-16 19:15:01 +00:00
{
2018-01-17 21:25:43 +00:00
gl.activeTexture(gl.TEXTURE0 + textureUnit);
gl.bindTexture(gl.TEXTURE_2D, texture);
2017-05-16 19:15:01 +00:00
2018-01-17 21:25:43 +00:00
this.currentTextures[textureUnit] = texture;
}
2016-12-07 02:28:22 +00:00
2018-01-17 21:25:43 +00:00
return this;
2017-05-16 19:15:01 +00:00
},
2018-01-17 21:25:43 +00:00
setFramebuffer: function (framebuffer)
2017-01-23 21:42:47 +00:00
{
var gl = this.gl;
2018-01-17 21:25:43 +00:00
if (framebuffer !== this.currentFramebuffer)
{
2018-01-17 21:25:43 +00:00
gl.bindFramebuffer(gl.FRAMEBUFFER, framebuffer);
}
2018-01-17 21:25:43 +00:00
return this;
},
2018-01-17 21:25:43 +00:00
setVertexBuffer: function (vertexBuffer)
{
2018-01-17 21:25:43 +00:00
var gl = this.gl;
2018-01-17 21:25:43 +00:00
if (vertexBuffer !== this.currentVertexBuffer)
{
2018-01-17 21:25:43 +00:00
gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
}
2018-01-17 21:25:43 +00:00
return this;
},
2018-01-17 21:25:43 +00:00
setIndexBuffer: function (indexBuffer)
2017-05-20 01:16:45 +00:00
{
var gl = this.gl;
2018-01-17 21:25:43 +00:00
if (indexBuffer !== this.currentIndexBuffer)
{
2018-01-17 21:25:43 +00:00
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, indexBuffer);
}
2017-05-20 01:16:45 +00:00
2018-01-17 21:25:43 +00:00
return this;
2017-05-20 01:16:45 +00:00
},
2018-01-17 21:25:43 +00:00
/* Renderer Resource Creation Functions */
2018-01-20 00:37:52 +00:00
createTexture2D: function (mipLevel, minFilter, magFilter, wrapT, wrapS, format, pixels, width, height, pma)
2018-01-17 21:25:43 +00:00
{
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var texture = gl.createTexture();
pma = (pma === undefined || pma === null) ? true : pma;
this.setTexture2D(texture, 0);
2018-01-17 21:25:43 +00:00
2018-01-20 00:37:52 +00:00
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, magFilter);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrapS);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrapT);
gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, pma);
if (pixels === null || pixels === undefined)
{
gl.texImage2D(gl.TEXTURE_2D, mipLevel, format, width, height, 0, format, gl.UNSIGNED_BYTE, null);
}
else
{
gl.texImage2D(gl.TEXTURE_2D, mipLevel, format, format, gl.UNSIGNED_BYTE, pixels);
width = pixels.width;
height = pixels.height;
}
this.setTexture2D(null, 0);
texture.isAlphaPremultiplied = pma;
texture.isRenderTexture = false;
texture.width = width;
texture.height = height;
return texture;
2018-01-17 21:25:43 +00:00
},
2018-01-20 00:37:52 +00:00
createFramebuffer: function (width, height, renderTexture, addDepthStencilBuffer)
2018-01-17 21:25:43 +00:00
{
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var framebuffer = gl.createFramebuffer();
var complete = 0;
this.setFramebuffer(framebuffer);
if (addDepthStencilBuffer)
{
var depthStencilBuffer = gl.createRenderbuffer();
gl.bindRenderbuffer(gl.RENDERBUFFER, depthStencilBuffer);
gl.renderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_STENCIL, width, height);
gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, depthStencilBuffer);
}
renderTexture.isRenderTexture = true;
renderTexture.isAlphaPremultiplied = false;
gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, renderTexture, 0);
complete = gl.checkFramebufferStatus(gl.FRAMEBUFFER);
if (complete !== gl.FRAMEBUFFER_COMPLETE)
{
var errors = {
36054: 'Incomplete Attachment',
36055: 'Missing Attachment',
36057: 'Incomplete Dimensions',
36061: 'Framebuffer Unsupported'
};
throw new Error('Framebuffer incomplete. Framebuffer status: ' + errors[complete]);
}
framebuffer.renderTexture = renderTexture;
2018-01-20 00:37:52 +00:00
this.setFramebuffer(null);
return framebuffer;
2018-01-17 21:25:43 +00:00
},
2018-01-20 00:37:52 +00:00
createProgram: function (vertexShader, fragmentShader)
2018-01-17 21:25:43 +00:00
{
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var program = gl.createProgram();
var vs = gl.createShader(gl.VERTEX_SHADER);
var fs = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(vs, vertexShader);
gl.shaderSource(fs, fragmentShader);
gl.compileShader(vs);
gl.compileShader(fs);
if (!gl.getShaderParameter(vs, gl.COMPILE_STATUS))
{
return new Error('Failed to compile Vertex Shader:\n' + gl.getShaderInfoLog(vs));
}
if (!gl.getShaderParameter(fs, gl.COMPILE_STATUS))
{
return new Error('Failed to compile Fragment Shader:\n' + gl.getShaderInfoLog(fs));
}
2017-08-03 20:02:57 +00:00
2018-01-20 00:37:52 +00:00
gl.attachShader(program, vs);
gl.attachShader(program, fs);
gl.linkProgram(program);
if (!gl.getProgramParameter(program, gl.LINK_STATUS))
{
return new Error('Failed to link program:\n' + gl.getProgramInfoLog(program));
}
return program;
2018-01-17 21:25:43 +00:00
},
2018-01-20 00:37:52 +00:00
createVertexBuffer: function (initialDataOrSize, bufferUsage)
2018-01-17 21:25:43 +00:00
{
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var vertexBuffer = gl.createBuffer();
2018-01-20 00:37:52 +00:00
this.setVertexBuffer(vertexBuffer);
gl.bufferData(gl.ARRAY_BUFFER, initialDataOrSize, bufferUsage);
this.setVertexBuffer(null);
return vertexBuffer;
},
2018-01-17 21:25:43 +00:00
createIndexBuffer: function ()
{
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var indexBuffer = gl.createBuffer();
this.setIndexBuffer(indexBuffer);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, initialDataOrSize, bufferUsage);
this.setIndexBuffer(null);
return indexBuffer;
2017-01-23 21:42:47 +00:00
}
});
2016-12-07 02:28:22 +00:00
2018-01-17 21:25:43 +00:00
module.exports = WebGLRenderer;