phaser/v3/src/renderer/webgl/WebGLRenderer.js

346 lines
9.8 KiB
JavaScript
Raw Normal View History

2016-12-07 02:28:22 +00:00
/**
* @author Richard Davey (@photonstorm)
* @author Felipe Alfonso (@bitnenfer)
* @copyright 2017 Photon Storm Ltd.
2016-12-07 02:28:22 +00:00
* @license {@link https://github.com/photonstorm/phaser/blob/master/license.txt|MIT License}
*/
2016-12-07 03:42:41 +00:00
var CONST = require('../../const');
2016-12-07 02:28:22 +00:00
var CreateEmptyTexture = require('./utils/CreateEmptyTexture');
2017-01-19 23:20:36 +00:00
var CreateTexture2DImage = require('./utils/texture/CreateTexture2DImage');
2017-01-19 22:43:41 +00:00
var BlitterBatch = require('./batches/blitter/BlitterBatch');
2017-02-10 00:48:32 +00:00
var AAQuadBatch = require('./batches/aaquad/AAQuadBatch');
var SpriteBatch = require('./batches/sprite/SpriteBatch');
2017-01-23 21:42:47 +00:00
var BlendModes = require('../BlendModes');
2016-12-07 02:28:22 +00:00
var WebGLRenderer = function (game)
{
this.game = game;
2016-12-07 03:42:41 +00:00
this.type = CONST.WEBGL;
this.width = game.config.width * game.config.resolution;
this.height = game.config.height * game.config.resolution;
this.resolution = game.config.resolution;
2016-12-07 02:28:22 +00:00
this.view = game.canvas;
// All of these settings will be able to be controlled via the Game Config
this.config = {
clearBeforeRender: true,
transparent: false,
autoResize: false,
preserveDrawingBuffer: false,
2016-12-07 02:28:22 +00:00
WebGLContextOptions: {
alpha: true,
antialias: true,
premultipliedAlpha: true,
stencil: true,
preserveDrawingBuffer: false
}
2016-12-07 02:28:22 +00:00
};
this.contextLost = false;
this.maxTextures = 1;
this.multiTexture = false;
this.blendModes = [];
this.gl = null;
this.extensions = null;
this.batches = [];
this.blitterBatch = null;
this.aaQuadBatch = null;
this.spriteBatch = null;
2017-01-19 17:53:20 +00:00
this.batch = null;
this.currentTexture2D = null;
this.init();
2016-12-07 02:28:22 +00:00
};
WebGLRenderer.prototype.constructor = WebGLRenderer;
WebGLRenderer.prototype = {
init: function ()
{
console.log('WebGLRenderer.init');
this.gl = this.view.getContext('webgl', this.config.WebGLContextOptions) || this.view.getContext('experimental-webgl', this.config.WebGLContextOptions);
2016-12-07 02:28:22 +00:00
if (!this.gl)
{
this.contextLost = true;
throw new Error('This browser does not support WebGL. Try using the Canvas renderer.');
}
var gl = this.gl;
var color = this.game.config.backgroundColor;
2016-12-07 02:28:22 +00:00
gl.disable(gl.DEPTH_TEST);
gl.disable(gl.CULL_FACE);
gl.enable(gl.BLEND);
2017-02-11 20:25:12 +00:00
gl.clearColor(color.redGL, color.greenGL, color.blueGL, color.alphaGL);
2016-12-07 02:28:22 +00:00
this.resize(this.width, this.height);
// Map Blend Modes
var add = [ gl.SRC_ALPHA, gl.DST_ALPHA ];
2017-01-24 15:21:49 +00:00
var normal = [ gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA ];
2016-12-07 02:28:22 +00:00
var multiply = [ gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA ];
var screen = [ gl.SRC_ALPHA, gl.ONE ];
this.blendModes = [
normal, add, multiply, screen, normal,
normal, normal, normal, normal,
normal, normal, normal, normal,
normal, normal, normal, normal
];
2017-01-23 21:42:47 +00:00
this.blendMode = -1;
this.extensions = gl.getSupportedExtensions();
this.blitterBatch = this.addBatch(new BlitterBatch(this.game, gl, this));
this.aaQuadBatch = this.addBatch(new AAQuadBatch(this.game, gl, this));
this.spriteBatch = this.addBatch(new SpriteBatch(this.game, gl, this));
2016-12-07 02:28:22 +00:00
},
2017-01-19 23:20:36 +00:00
createTexture2D: function (source)
{
var gl = this.gl;
if (!source.glTexture)
{
source.glTexture = CreateTexture2DImage(gl, source.image, gl.NEAREST, 0);
}
this.currentTexture2D = source.glTexture;
},
2017-01-19 22:43:41 +00:00
setTexture2D: function (texture2D)
2017-01-19 17:53:20 +00:00
{
2017-01-20 18:53:53 +00:00
if (this.currentTexture2D !== texture2D)
{
2017-01-20 18:53:53 +00:00
if (this.batch)
2017-01-20 18:51:00 +00:00
{
2017-01-20 18:53:53 +00:00
this.batch.flush();
2017-01-20 18:51:00 +00:00
}
2017-01-20 18:53:53 +00:00
var gl = this.gl;
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, texture2D);
2017-01-20 18:53:53 +00:00
this.currentTexture2D = texture2D;
}
2017-01-19 17:53:20 +00:00
},
setBatch: function (batch, texture2D)
2017-01-19 22:43:41 +00:00
{
this.setTexture2D(texture2D);
2017-01-20 18:53:53 +00:00
if (this.batch !== batch)
2017-01-19 22:43:41 +00:00
{
if (this.batch)
{
this.batch.flush();
}
2017-01-20 18:53:53 +00:00
2017-01-19 22:43:41 +00:00
batch.bind();
2017-01-20 18:53:53 +00:00
2017-01-19 22:43:41 +00:00
this.batch = batch;
}
},
2016-12-07 02:28:22 +00:00
resize: function (width, height)
{
2016-12-07 03:42:41 +00:00
var res = this.game.config.resolution;
this.width = width * res;
this.height = height * res;
2016-12-07 02:28:22 +00:00
this.view.width = this.width;
this.view.height = this.height;
if (this.autoResize)
{
2016-12-07 03:42:41 +00:00
this.view.style.width = (this.width / res) + 'px';
this.view.style.height = (this.height / res) + 'px';
2016-12-07 02:28:22 +00:00
}
this.gl.viewport(0, 0, this.width, this.height);
for (var i = 0, l = this.batches.length; i < l; ++i)
{
this.batches[i].bind();
this.batches[i].resize(width, height, resolution);
}
if (this.batch)
{
this.batch.bind();
}
2016-12-07 02:28:22 +00:00
},
// Call at the start of the render loop
preRender: function ()
2016-12-07 02:28:22 +00:00
{
// No point rendering if our context has been blown up!
if (this.contextLost)
{
return;
}
// Add Pre-render hook
var gl = this.gl;
2017-02-11 20:25:12 +00:00
var color = this.game.config.backgroundColor;
gl.clearColor(color.redGL, color.greenGL, color.blueGL, color.alphaGL);
// Some drivers require to call glClear
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT | gl.STENCIL_BUFFER_BIT);
2016-12-07 02:28:22 +00:00
2017-01-23 21:42:47 +00:00
this.setBlendMode(BlendModes.NORMAL);
},
/**
* Renders a single State.
*
* @method render
* @param {Phaser.State} state - The State to be rendered.
* @param {number} interpolationPercentage - The cumulative amount of time that hasn't been simulated yet, divided
* by the amount of time that will be simulated the next time update()
* runs. Useful for interpolating frames.
*/
render: function (state, children, interpolationPercentage, camera)
{
2016-12-07 02:28:22 +00:00
// Could move to the State Systems or MainLoop
var gl = this.gl;
2017-02-07 19:30:50 +00:00
var scissor = (camera.x !== 0 || camera.y !== 0 || camera.width !== gl.canvas.width || camera.height !== gl.canvas.height);
2017-02-07 16:12:20 +00:00
2017-02-07 19:30:50 +00:00
if (scissor)
{
gl.enable(gl.SCISSOR_TEST);
gl.scissor(camera.x, (gl.drawingBufferHeight - camera.y - camera.height), camera.width, camera.height);
}
2017-02-07 16:12:20 +00:00
// We could either clear color or render a quad
gl.clear(gl.COLOR_BUFFER_BIT);
2017-01-31 21:40:29 +00:00
var list = state.sys.children.list;
var length = list.length;
for (var index = 0; index < length; ++index)
2017-01-19 17:53:20 +00:00
{
2017-01-31 21:40:29 +00:00
var child = list[index];
// Setting blend mode if needed
var batch = this.batch;
var newBlendMode = child.color._blendMode;
if (this.blendMode !== newBlendMode)
{
if (batch)
{
batch.flush();
}
var blend = this.blendModes[newBlendMode];
gl.enable(gl.BLEND);
if (blend.length > 2)
{
gl.blendFuncSeparate(blend[0], blend[1], blend[2], blend[3]);
}
else
{
gl.blendFunc(blend[0], blend[1]);
}
this.blendMode = newBlendMode;
}
// drawing child
child.renderWebGL(this, child, interpolationPercentage, camera);
batch = this.batch;
2017-01-24 13:15:25 +00:00
if (batch && batch.isFull())
{
batch.flush();
}
2017-01-19 17:53:20 +00:00
}
if (this.batch)
{
this.batch.flush();
}
2017-02-10 00:48:32 +00:00
if (camera._fadeAlpha > 0 || camera._flashAlpha > 0)
{
var aaQuadBatch = this.aaQuadBatch;
aaQuadBatch.bind();
// fade rendering
aaQuadBatch.add(
camera.x, camera.y, camera.width, camera.height,
camera._fadeRed,
camera._fadeGreen,
camera._fadeBlue,
camera._fadeAlpha
);
// flash rendering
aaQuadBatch.add(
camera.x, camera.y, camera.width, camera.height,
camera._flashRed,
camera._flashGreen,
camera._flashBlue,
camera._flashAlpha
);
aaQuadBatch.flush();
this.batch.bind();
}
2017-02-07 19:30:50 +00:00
if (scissor)
{
gl.disable(gl.SCISSOR_TEST);
}
},
2016-12-07 02:28:22 +00:00
2017-02-07 19:30:50 +00:00
// Called at the end of the render loop (tidy things up, etc)
postRender: function ()
{
2017-02-08 00:08:09 +00:00
if (this.batch)
{
this.batch.flush();
}
// Add Post-render hook
2016-12-07 02:28:22 +00:00
// console.log('%c render end ', 'color: #ffffff; background: #ff0000;');
},
destroy: function ()
{
this.gl = null;
2017-01-19 17:53:20 +00:00
},
2017-01-23 21:42:47 +00:00
createFBO: function () {},
setBlendMode: function (newBlendMode)
{
var gl = this.gl;
var batch = this.batch;
var blend = null;
if (this.blendMode !== newBlendMode)
{
if (batch)
batch.flush();
blend = this.blendModes[newBlendMode];
gl.enable(gl.BLEND);
2017-01-24 15:21:49 +00:00
if (blend.length > 2)
{
gl.blendFuncSeparate(blend[0], blend[1], blend[2], blend[3]);
}
else
{
gl.blendFunc(blend[0], blend[1]);
}
2017-01-23 21:42:47 +00:00
this.blendMode = newBlendMode;
}
},
addBatch: function (batchInstance)
{
var index = this.batches.indexOf(batchInstance);
if (index < 0)
{
this.batches.push(batchInstance);
return batchInstance;
}
return null;
2017-01-23 21:42:47 +00:00
}
2016-12-07 02:28:22 +00:00
};
module.exports = WebGLRenderer;