phaser/src/renderer/webgl/WebGLRenderer.js

2731 lines
87 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
2018-04-05 08:02:36 +00:00
* @author Felipe Alfonso <@bitnenfer>
2019-01-15 16:20:22 +00:00
* @copyright 2019 Photon Storm Ltd.
2019-05-10 15:15:04 +00:00
* @license {@link https://opensource.org/licenses/MIT|MIT License}
2018-02-12 16:01:20 +00:00
*/
var BaseCamera = require('../../cameras/2d/BaseCamera');
var CameraEvents = require('../../cameras/2d/events');
var Class = require('../../utils/Class');
var CONST = require('../../const');
2018-01-20 04:05:56 +00:00
var IsSizePowerOfTwo = require('../../math/pow2/IsSizePowerOfTwo');
var ScaleEvents = require('../../scale/events');
var SpliceOne = require('../../utils/array/SpliceOne');
2019-01-18 13:41:43 +00:00
var TextureEvents = require('../../textures/events');
var TransformMatrix = require('../../gameobjects/components/TransformMatrix');
2018-01-24 03:57:33 +00:00
var Utils = require('./Utils');
2018-02-09 19:19:21 +00:00
var WebGLSnapshot = require('../snapshot/WebGLSnapshot');
2018-01-17 21:25:43 +00:00
// Default Pipelines
var BitmapMaskPipeline = require('./pipelines/BitmapMaskPipeline');
2018-01-29 21:46:48 +00:00
var ForwardDiffuseLightPipeline = require('./pipelines/ForwardDiffuseLightPipeline');
2018-02-09 19:19:21 +00:00
var TextureTintPipeline = require('./pipelines/TextureTintPipeline');
2018-01-22 21:21:47 +00:00
2018-03-19 20:41:24 +00:00
/**
* @callback WebGLContextCallback
*
* @param {Phaser.Renderer.WebGL.WebGLRenderer} renderer - The WebGL Renderer which owns the context.
2018-03-19 20:41:24 +00:00
*/
2018-01-31 13:54:44 +00:00
/**
2018-02-09 19:19:21 +00:00
* @classdesc
2018-04-25 23:30:41 +00:00
* WebGLRenderer is a class that contains the needed functionality to keep the
* WebGLRenderingContext state clean. The main idea of the WebGLRenderer is to keep track of
* any context change that happens for WebGL rendering inside of Phaser. This means
* if raw webgl functions are called outside the WebGLRenderer of the Phaser WebGL
* rendering ecosystem they might pollute the current WebGLRenderingContext state producing
* unexpected behavior. It's recommended that WebGL interaction is done through
2018-04-25 23:30:41 +00:00
* WebGLRenderer and/or WebGLPipeline.
2018-02-09 19:19:21 +00:00
*
* @class WebGLRenderer
2018-10-10 09:49:13 +00:00
* @memberof Phaser.Renderer.WebGL
2018-02-09 19:19:21 +00:00
* @constructor
* @since 3.0.0
*
* @param {Phaser.Game} game - The Game instance which owns this WebGL Renderer.
2018-01-31 13:54:44 +00:00
*/
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-02-16 18:44:07 +00:00
// eslint-disable-next-line consistent-this
2018-01-17 21:25:43 +00:00
var renderer = this;
2018-02-09 19:19:21 +00:00
var gameConfig = game.config;
var contextCreationConfig = {
2018-11-20 11:02:19 +00:00
alpha: gameConfig.transparent,
desynchronized: gameConfig.desynchronized,
depth: false,
antialias: gameConfig.antialias,
premultipliedAlpha: gameConfig.premultipliedAlpha,
stencil: true,
failIfMajorPerformanceCaveat: gameConfig.failIfMajorPerformanceCaveat,
powerPreference: gameConfig.powerPreference
2018-01-17 21:25:43 +00:00
};
2018-02-09 19:19:21 +00:00
/**
* The local configuration settings of this WebGL Renderer.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#config
2019-02-04 17:16:08 +00:00
* @type {object}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
this.config = {
clearBeforeRender: gameConfig.clearBeforeRender,
antialias: gameConfig.antialias,
backgroundColor: gameConfig.backgroundColor,
contextCreation: contextCreationConfig,
resolution: gameConfig.resolution,
roundPixels: gameConfig.roundPixels,
maxTextures: gameConfig.maxTextures,
maxTextureSize: gameConfig.maxTextureSize,
batchSize: gameConfig.batchSize,
maxLights: gameConfig.maxLights
};
2018-02-09 19:19:21 +00:00
/**
* The Game instance which owns this WebGL Renderer.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#game
* @type {Phaser.Game}
* @since 3.0.0
*/
this.game = game;
2018-02-09 19:19:21 +00:00
/**
* A constant which allows the renderer to be easily identified as a WebGL Renderer.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#type
* @type {integer}
* @since 3.0.0
*/
this.type = CONST.WEBGL;
2018-02-09 19:19:21 +00:00
/**
* The width of the canvas being rendered to.
* This is populated in the onResize event handler.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#width
* @type {integer}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
this.width = 0;
2018-02-09 19:19:21 +00:00
/**
* The height of the canvas being rendered to.
* This is populated in the onResize event handler.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#height
* @type {integer}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
this.height = 0;
2018-02-09 19:19:21 +00:00
/**
* The canvas which this WebGL Renderer draws to.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#canvas
* @type {HTMLCanvasElement}
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.canvas = game.canvas;
2018-02-09 19:19:21 +00:00
/**
* An array of functions to invoke if the WebGL context is lost.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#lostContextCallbacks
2018-03-19 20:41:24 +00:00
* @type {WebGLContextCallback[]}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
2018-01-24 00:40:20 +00:00
this.lostContextCallbacks = [];
2018-02-09 19:19:21 +00:00
/**
* An array of functions to invoke if the WebGL context is restored.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#restoredContextCallbacks
2018-03-19 20:41:24 +00:00
* @type {WebGLContextCallback[]}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
2018-01-24 00:40:20 +00:00
this.restoredContextCallbacks = [];
2018-02-09 19:19:21 +00:00
/**
* An array of blend modes supported by the WebGL Renderer.
*
* This array includes the default blend modes as well as any custom blend modes added through {@link #addBlendMode}.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#blendModes
* @type {array}
* @default []
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.blendModes = [];
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Keeps track of any WebGLTexture created with the current WebGLRenderingContext
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#nativeTextures
* @type {array}
* @default []
* @since 3.0.0
*/
this.nativeTextures = [];
2018-02-09 19:19:21 +00:00
/**
* Set to `true` if the WebGL context of the renderer is lost.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#contextLost
* @type {boolean}
* @default false
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.contextLost = false;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* This object will store all pipelines created through addPipeline
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#pipelines
2018-02-13 00:12:17 +00:00
* @type {object}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.pipelines = null;
2018-02-09 19:19:21 +00:00
/**
* Details about the currently scheduled snapshot.
*
* If a non-null `callback` is set in this object, a snapshot of the canvas will be taken after the current frame is fully rendered.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#snapshotState
2019-05-09 11:36:36 +00:00
* @type {Phaser.Types.Renderer.Snapshot.SnapshotState}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
2018-01-20 04:05:56 +00:00
this.snapshotState = {
2019-01-25 10:01:53 +00:00
x: 0,
y: 0,
width: 1,
height: 1,
2019-01-25 10:01:53 +00:00
getPixel: false,
2018-01-20 04:05:56 +00:00
callback: null,
type: 'image/png',
encoder: 0.92,
isFramebuffer: false,
bufferWidth: 0,
bufferHeight: 0
2018-01-20 04:05:56 +00:00
};
2018-02-16 18:07:49 +00:00
// Internal Renderer State (Textures, Framebuffers, Pipelines, Buffers, etc)
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Cached value for the last texture unit that was used
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentActiveTextureUnit
2018-02-16 18:07:49 +00:00
* @type {integer}
2018-02-15 14:31:15 +00:00
* @since 3.1.0
*/
this.currentActiveTextureUnit = 0;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* An array of the last texture handles that were bound to the WebGLRenderingContext
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentTextures
2018-02-13 00:12:17 +00:00
* @type {array}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.currentTextures = new Array(16);
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Current framebuffer in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentFramebuffer
2018-02-13 00:12:17 +00:00
* @type {WebGLFramebuffer}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.currentFramebuffer = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Current WebGLPipeline in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentPipeline
2018-02-13 00:12:17 +00:00
* @type {Phaser.Renderer.WebGL.WebGLPipeline}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.currentPipeline = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Current WebGLProgram in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentProgram
2018-02-13 00:12:17 +00:00
* @type {WebGLProgram}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-22 21:21:47 +00:00
this.currentProgram = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Current WebGLBuffer (Vertex buffer) in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentVertexBuffer
2018-02-13 00:12:17 +00:00
* @type {WebGLBuffer}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.currentVertexBuffer = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Current WebGLBuffer (Index buffer) in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentIndexBuffer
2018-02-13 00:12:17 +00:00
* @type {WebGLBuffer}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.currentIndexBuffer = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Current blend mode in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentBlendMode
2018-02-16 18:07:49 +00:00
* @type {integer}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
2018-01-22 21:21:47 +00:00
this.currentBlendMode = Infinity;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Indicates if the the scissor state is enabled in WebGLRenderingContext
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentScissorEnabled
* @type {boolean}
* @default false
* @since 3.0.0
*/
2018-01-24 18:55:23 +00:00
this.currentScissorEnabled = false;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Stores the current scissor data
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentScissor
* @type {Uint32Array}
* @since 3.0.0
*/
2018-07-26 16:50:07 +00:00
// this.currentScissor = new Uint32Array([ 0, 0, this.width, this.height ]);
this.currentScissor = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Stack of scissor data
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#scissorStack
* @type {Uint32Array}
* @since 3.0.0
*/
2018-07-26 16:50:07 +00:00
this.scissorStack = [];
2018-01-17 21:25:43 +00:00
// Setup context lost and restore event listeners
2018-02-09 19:19:21 +00:00
2018-02-16 18:17:51 +00:00
this.canvas.addEventListener('webglcontextlost', function (event)
{
2018-01-17 21:25:43 +00:00
renderer.contextLost = true;
event.preventDefault();
2018-01-24 00:40:20 +00:00
for (var index = 0; index < renderer.lostContextCallbacks.length; ++index)
{
var callback = renderer.lostContextCallbacks[index];
callback[0].call(callback[1], renderer);
}
2018-01-17 21:25:43 +00:00
}, false);
2018-02-16 18:44:07 +00:00
this.canvas.addEventListener('webglcontextrestored', function ()
2018-02-16 18:17:51 +00:00
{
2018-01-17 21:25:43 +00:00
renderer.contextLost = false;
renderer.init(renderer.config);
2018-01-24 00:40:20 +00:00
for (var index = 0; index < renderer.restoredContextCallbacks.length; ++index)
{
var callback = renderer.restoredContextCallbacks[index];
callback[0].call(callback[1], renderer);
}
2018-01-17 21:25:43 +00:00
}, false);
2018-04-24 14:32:08 +00:00
// These are initialized post context creation
2018-02-09 19:19:21 +00:00
/**
* The underlying WebGL context of the renderer.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#gl
2018-02-13 00:12:17 +00:00
* @type {WebGLRenderingContext}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
this.gl = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Array of strings that indicate which WebGL extensions are supported by the browser
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#supportedExtensions
2018-02-13 00:12:17 +00:00
* @type {object}
2018-02-09 19:19:21 +00:00
* @default null
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.supportedExtensions = null;
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Extensions loaded into the current context
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#extensions
* @type {object}
* @default {}
* @since 3.0.0
*/
2018-01-17 21:25:43 +00:00
this.extensions = {};
/**
2018-04-25 23:30:41 +00:00
* Stores the current WebGL component formats for further use
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glFormats
* @type {array}
* @default []
* @since 3.2.0
*/
this.glFormats = [];
/**
* Stores the supported WebGL texture compression formats.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#compression
* @type {array}
* @since 3.8.0
*/
this.compression = {
ETC1: false,
PVRTC: false,
S3TC: false
};
2018-07-10 15:32:28 +00:00
/**
* Cached drawing buffer height to reduce gl calls.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#drawingBufferHeight
* @type {number}
2018-10-09 12:40:00 +00:00
* @readonly
2018-07-10 15:32:28 +00:00
* @since 3.11.0
*/
this.drawingBufferHeight = 0;
/**
* A blank 32x32 transparent texture, as used by the Graphics system where needed.
* This is set in the `boot` method.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#blankTexture
* @type {WebGLTexture}
2018-10-09 12:40:00 +00:00
* @readonly
* @since 3.12.0
*/
this.blankTexture = null;
/**
* A default Camera used in calls when no other camera has been provided.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#defaultCamera
* @type {Phaser.Cameras.Scene2D.BaseCamera}
* @since 3.12.0
*/
this.defaultCamera = new BaseCamera(0, 0, 0, 0);
/**
* A temporary Transform Matrix, re-used internally during batching.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#_tempMatrix1
* @private
* @type {Phaser.GameObjects.Components.TransformMatrix}
* @since 3.12.0
*/
this._tempMatrix1 = new TransformMatrix();
/**
* A temporary Transform Matrix, re-used internally during batching.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#_tempMatrix2
* @private
* @type {Phaser.GameObjects.Components.TransformMatrix}
* @since 3.12.0
*/
this._tempMatrix2 = new TransformMatrix();
/**
* A temporary Transform Matrix, re-used internally during batching.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#_tempMatrix3
* @private
* @type {Phaser.GameObjects.Components.TransformMatrix}
* @since 3.12.0
*/
this._tempMatrix3 = new TransformMatrix();
/**
* A temporary Transform Matrix, re-used internally during batching.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#_tempMatrix4
* @private
* @type {Phaser.GameObjects.Components.TransformMatrix}
* @since 3.12.0
*/
this._tempMatrix4 = new TransformMatrix();
2019-04-12 16:47:28 +00:00
/**
* The total number of masks currently stacked.
2019-04-12 16:47:28 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#maskCount
* @type {integer}
2019-04-12 16:47:28 +00:00
* @since 3.17.0
*/
2019-04-15 15:46:19 +00:00
this.maskCount = 0;
/**
* The mask stack.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#maskStack
* @type {Phaser.Display.Masks.GeometryMask[]}
* @since 3.17.0
*/
2019-04-15 15:46:19 +00:00
this.maskStack = [];
/**
* Internal property that tracks the currently set mask.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentMask
2019-04-29 13:45:54 +00:00
* @type {any}
* @since 3.17.0
*/
2019-04-29 13:45:54 +00:00
this.currentMask = { mask: null, camera: null };
/**
* Internal property that tracks the currently set camera mask.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentCameraMask
2019-04-29 13:45:54 +00:00
* @type {any}
* @since 3.17.0
*/
2019-04-29 13:28:58 +00:00
this.currentCameraMask = { mask: null, camera: null };
2019-05-01 11:29:45 +00:00
/**
* Internal gl function mapping for uniform look-up.
* https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/uniform
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glFuncMap
* @type {any}
* @since 3.17.0
*/
this.glFuncMap = null;
this.init(this.config);
},
2018-02-09 19:19:21 +00:00
/**
* Creates a new WebGLRenderingContext and initializes all internal state.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#init
* @since 3.0.0
*
* @param {object} config - The configuration object for the renderer.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
init: function (config)
{
var gl;
2019-01-18 13:41:43 +00:00
var game = this.game;
2018-01-17 21:25:43 +00:00
var canvas = this.canvas;
var clearColor = config.backgroundColor;
// Did they provide their own context?
2019-01-18 13:41:43 +00:00
if (game.config.context)
{
2019-01-18 13:41:43 +00:00
gl = game.config.context;
}
else
{
gl = canvas.getContext('webgl', config.contextCreation) || canvas.getContext('experimental-webgl', config.contextCreation);
}
2018-03-19 20:41:24 +00:00
if (!gl || gl.isContextLost())
2016-12-07 02:28:22 +00:00
{
this.contextLost = true;
2018-07-26 16:50:07 +00:00
throw new Error('WebGL unsupported');
2016-12-07 02:28:22 +00:00
}
2018-01-22 21:21:47 +00:00
this.gl = gl;
// Set it back into the Game, so developers can access it from there too
2019-01-18 13:41:43 +00:00
game.context = gl;
for (var i = 0; i <= 27; i++)
{
this.blendModes.push({ func: [ gl.ONE, gl.ONE_MINUS_SRC_ALPHA ], equation: gl.FUNC_ADD });
}
2018-11-13 10:32:24 +00:00
// ADD
this.blendModes[1].func = [ gl.ONE, gl.DST_ALPHA ];
2018-11-13 10:32:24 +00:00
// MULTIPLY
this.blendModes[2].func = [ gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA ];
2018-11-13 10:32:24 +00:00
// SCREEN
this.blendModes[3].func = [ gl.ONE, gl.ONE_MINUS_SRC_COLOR ];
2018-11-13 10:32:24 +00:00
// ERASE
this.blendModes[17] = { func: [ gl.ZERO, gl.ONE_MINUS_SRC_ALPHA ], equation: gl.FUNC_REVERSE_SUBTRACT };
2018-11-13 10:32:24 +00:00
this.glFormats[0] = gl.BYTE;
this.glFormats[1] = gl.SHORT;
this.glFormats[2] = gl.UNSIGNED_BYTE;
this.glFormats[3] = gl.UNSIGNED_SHORT;
this.glFormats[4] = gl.FLOAT;
2019-05-01 11:29:45 +00:00
// Set the gl function map
this.glFuncMap = {
mat2: { func: gl.uniformMatrix2fv, length: 1, matrix: true },
mat3: { func: gl.uniformMatrix3fv, length: 1, matrix: true },
mat4: { func: gl.uniformMatrix4fv, length: 1, matrix: true },
'1f': { func: gl.uniform1f, length: 1 },
'1fv': { func: gl.uniform1fv, length: 1 },
'1i': { func: gl.uniform1i, length: 1 },
'1iv': { func: gl.uniform1iv, length: 1 },
'2f': { func: gl.uniform2f, length: 2 },
'2fv': { func: gl.uniform2fv, length: 1 },
'2i': { func: gl.uniform2i, length: 2 },
'2iv': { func: gl.uniform2iv, length: 1 },
'3f': { func: gl.uniform3f, length: 3 },
'3fv': { func: gl.uniform3fv, length: 1 },
'3i': { func: gl.uniform3i, length: 3 },
'3iv': { func: gl.uniform3iv, length: 1 },
'4f': { func: gl.uniform4f, length: 4 },
'4fv': { func: gl.uniform4fv, length: 1 },
'4i': { func: gl.uniform4i, length: 4 },
'4iv': { func: gl.uniform4iv, length: 1 }
};
2018-01-17 21:25:43 +00:00
// Load supported extensions
var exts = gl.getSupportedExtensions();
if (!config.maxTextures)
{
config.maxTextures = gl.getParameter(gl.MAX_TEXTURE_IMAGE_UNITS);
}
if (!config.maxTextureSize)
{
config.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);
}
2018-05-09 13:32:32 +00:00
var extString = 'WEBGL_compressed_texture_';
var wkExtString = 'WEBKIT_' + extString;
this.compression.ETC1 = gl.getExtension(extString + 'etc1') || gl.getExtension(wkExtString + 'etc1');
this.compression.PVRTC = gl.getExtension(extString + 'pvrtc') || gl.getExtension(wkExtString + 'pvrtc');
this.compression.S3TC = gl.getExtension(extString + 's3tc') || gl.getExtension(wkExtString + 's3tc');
this.supportedExtensions = exts;
// Setup initial WebGL state
2016-12-07 02:28:22 +00:00
gl.disable(gl.DEPTH_TEST);
gl.disable(gl.CULL_FACE);
2016-12-07 02:28:22 +00:00
gl.enable(gl.BLEND);
gl.clearColor(clearColor.redGL, clearColor.greenGL, clearColor.blueGL, clearColor.alphaGL);
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 = {};
2019-01-18 13:41:43 +00:00
this.addPipeline('TextureTintPipeline', new TextureTintPipeline({ game: game, renderer: this }));
this.addPipeline('BitmapMaskPipeline', new BitmapMaskPipeline({ game: game, renderer: this }));
this.addPipeline('Light2D', new ForwardDiffuseLightPipeline({ game: game, renderer: this, maxLights: config.maxLights }));
2018-03-19 20:41:24 +00:00
2018-01-17 21:25:43 +00:00
this.setBlendMode(CONST.BlendModes.NORMAL);
2018-07-26 16:50:07 +00:00
2019-01-18 13:41:43 +00:00
game.textures.once(TextureEvents.READY, this.boot, this);
2018-07-13 10:13:46 +00:00
2018-01-17 21:25:43 +00:00
return this;
},
2018-07-13 10:13:46 +00:00
/**
* Internal boot handler. Calls 'boot' on each pipeline.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#boot
* @private
* @since 3.11.0
*/
boot: function ()
{
for (var pipelineName in this.pipelines)
{
this.pipelines[pipelineName].boot();
}
2018-07-26 14:04:46 +00:00
var blank = this.game.textures.getFrame('__DEFAULT');
this.pipelines.TextureTintPipeline.currentFrame = blank;
2018-07-26 14:04:46 +00:00
this.blankTexture = blank;
var gl = this.gl;
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
gl.enable(gl.SCISSOR_TEST);
this.setPipeline(this.pipelines.TextureTintPipeline);
this.game.scale.on(ScaleEvents.RESIZE, this.onResize, this);
var baseSize = this.game.scale.baseSize;
this.resize(baseSize.width, baseSize.height, this.game.scale.resolution);
},
/**
* The event handler that manages the `resize` event dispatched by the Scale Manager.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#onResize
* @since 3.16.0
*
* @param {Phaser.Structs.Size} gameSize - The default Game Size object. This is the un-modified game dimensions.
* @param {Phaser.Structs.Size} baseSize - The base Size object. The game dimensions multiplied by the resolution. The canvas width / height values match this.
* @param {Phaser.Structs.Size} displaySize - The display Size object. The size of the canvas style width / height attributes.
* @param {number} [resolution] - The Scale Manager resolution setting.
*/
onResize: function (gameSize, baseSize, displaySize, resolution)
{
// Has the underlying canvas size changed?
if (baseSize.width !== this.width || baseSize.height !== this.height || resolution !== this.resolution)
{
this.resize(baseSize.width, baseSize.height, resolution);
}
2018-07-13 10:13:46 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Resizes the drawing buffer to match that required by the Scale Manager.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#resize
* @since 3.0.0
*
* @param {number} [width] - The new width of the renderer.
* @param {number} [height] - The new height of the renderer.
* @param {number} [resolution] - The new resolution of the renderer.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
resize: function (width, height, resolution)
{
2018-01-17 21:25:43 +00:00
var gl = this.gl;
2018-01-22 21:21:47 +00:00
var pipelines = this.pipelines;
this.width = width;
this.height = height;
this.resolution = resolution;
2018-03-19 20:41:24 +00:00
gl.viewport(0, 0, width, height);
2018-07-13 10:13:46 +00:00
// Update all registered pipelines
2018-01-20 04:05:56 +00:00
for (var pipelineName in pipelines)
{
2018-01-22 21:21:47 +00:00
pipelines[pipelineName].resize(width, height, resolution);
2018-01-20 04:05:56 +00:00
}
2018-07-10 15:32:28 +00:00
this.drawingBufferHeight = gl.drawingBufferHeight;
gl.scissor(0, (gl.drawingBufferHeight - height), width, height);
this.defaultCamera.setSize(width, height);
2018-01-17 21:25:43 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Adds a callback to be invoked when the WebGL context has been restored by the browser.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#onContextRestored
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLContextCallback} callback - The callback to be invoked on context restoration.
* @param {object} target - The context of the callback.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-24 00:40:20 +00:00
onContextRestored: function (callback, target)
{
2018-02-16 18:17:51 +00:00
this.restoredContextCallbacks.push([ callback, target ]);
2018-01-24 00:40:20 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Adds a callback to be invoked when the WebGL context has been lost by the browser.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#onContextLost
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLContextCallback} callback - The callback to be invoked on context loss.
* @param {object} target - The context of the callback.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-24 00:40:20 +00:00
onContextLost: function (callback, target)
{
2018-02-16 18:17:51 +00:00
this.lostContextCallbacks.push([ callback, target ]);
2018-01-24 00:40:20 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Checks if a WebGL extension is supported
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#hasExtension
* @since 3.0.0
*
2018-04-25 23:30:41 +00:00
* @param {string} extensionName - Name of the WebGL extension
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {boolean} `true` if the extension is supported, otherwise `false`.
2018-02-09 19:19:21 +00:00
*/
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-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Loads a WebGL extension
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getExtension
* @since 3.0.0
*
* @param {string} extensionName - The name of the extension to load.
2018-02-09 19:19:21 +00:00
*
2018-04-25 23:30:41 +00:00
* @return {object} WebGL extension if the extension is supported
2018-02-09 19:19:21 +00:00
*/
2018-01-17 21:25:43 +00:00
getExtension: function (extensionName)
2017-01-19 23:20:36 +00:00
{
2018-02-16 18:17:51 +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-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Flushes the current pipeline if the pipeline is bound
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#flush
* @since 3.0.0
*/
2018-01-20 04:05:56 +00:00
flush: function ()
{
if (this.currentPipeline)
{
this.currentPipeline.flush();
}
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Checks if a pipeline is present in the current WebGLRenderer
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#hasPipeline
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {string} pipelineName - The name of the pipeline.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {boolean} `true` if the given pipeline is loaded, otherwise `false`.
2018-02-09 19:19:21 +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-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Returns the pipeline by name if the pipeline exists
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getPipeline
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {string} pipelineName - The name of the pipeline.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {Phaser.Renderer.WebGL.WebGLPipeline} The pipeline instance, or `null` if not found.
2018-02-09 19:19:21 +00:00
*/
2018-01-17 21:25:43 +00:00
getPipeline: function (pipelineName)
2017-04-07 01:49:15 +00:00
{
2018-02-09 19:19:21 +00:00
return (this.hasPipeline(pipelineName)) ? this.pipelines[pipelineName] : null;
2017-01-19 22:43:41 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Removes a pipeline by name.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#removePipeline
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {string} pipelineName - The name of the pipeline to be removed.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +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];
2018-01-17 21:25:43 +00:00
return this;
2016-12-07 02:28:22 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Adds a pipeline instance into the collection of pipelines
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#addPipeline
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {string} pipelineName - A unique string-based key for the pipeline.
* @param {Phaser.Renderer.WebGL.WebGLPipeline} pipelineInstance - A pipeline instance which must extend WebGLPipeline.
2018-02-09 19:19:21 +00:00
*
* @return {Phaser.Renderer.WebGL.WebGLPipeline} The pipeline instance that was passed.
2018-02-09 19:19:21 +00:00
*/
2018-01-17 21:25:43 +00:00
addPipeline: function (pipelineName, pipelineInstance)
2016-12-07 02:28:22 +00:00
{
if (!this.hasPipeline(pipelineName))
{
this.pipelines[pipelineName] = pipelineInstance;
}
else
{
2018-09-28 11:19:21 +00:00
console.warn('Pipeline exists: ' + pipelineName);
}
2018-01-30 22:46:43 +00:00
pipelineInstance.name = pipelineName;
this.pipelines[pipelineName].resize(this.width, this.height, this.config.resolution);
2016-12-07 02:28:22 +00:00
return pipelineInstance;
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Pushes a new scissor state. This is used to set nested scissor states.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#pushScissor
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {integer} x - The x position of the scissor.
* @param {integer} y - The y position of the scissor.
* @param {integer} width - The width of the scissor.
* @param {integer} height - The height of the scissor.
* @param {integer} [drawingBufferHeight] - Optional drawingBufferHeight override value.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {integer[]} An array containing the scissor values.
2018-02-09 19:19:21 +00:00
*/
pushScissor: function (x, y, width, height, drawingBufferHeight)
2018-01-24 18:55:23 +00:00
{
if (drawingBufferHeight === undefined) { drawingBufferHeight = this.drawingBufferHeight; }
2018-01-24 18:55:23 +00:00
var scissorStack = this.scissorStack;
2018-09-28 11:19:21 +00:00
var scissor = [ x, y, width, height ];
2018-07-26 16:50:07 +00:00
scissorStack.push(scissor);
2018-03-19 20:41:24 +00:00
this.setScissor(x, y, width, height, drawingBufferHeight);
2018-12-13 14:24:49 +00:00
this.currentScissor = scissor;
2018-07-26 16:50:07 +00:00
return scissor;
2018-01-24 18:55:23 +00:00
},
/**
2018-09-28 11:19:21 +00:00
* Sets the current scissor state.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setScissor
* @since 3.0.0
2018-09-28 11:19:21 +00:00
*
* @param {integer} x - The x position of the scissor.
* @param {integer} y - The y position of the scissor.
* @param {integer} width - The width of the scissor.
* @param {integer} height - The height of the scissor.
* @param {integer} [drawingBufferHeight] - Optional drawingBufferHeight override value.
*/
setScissor: function (x, y, width, height, drawingBufferHeight)
{
if (drawingBufferHeight === undefined) { drawingBufferHeight = this.drawingBufferHeight; }
var gl = this.gl;
var current = this.currentScissor;
var setScissor = (width > 0 && height > 0);
if (current && setScissor)
{
var cx = current[0];
var cy = current[1];
var cw = current[2];
var ch = current[3];
2018-08-31 17:19:31 +00:00
setScissor = (cx !== x || cy !== y || cw !== width || ch !== height);
}
2018-12-13 14:24:49 +00:00
if (setScissor)
{
this.flush();
// https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/scissor
gl.scissor(x, (drawingBufferHeight - y - height), width, height);
}
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Pops the last scissor state and sets it.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#popScissor
* @since 3.0.0
*/
2018-01-24 18:55:23 +00:00
popScissor: function ()
{
var scissorStack = this.scissorStack;
2018-03-19 20:41:24 +00:00
// Remove the current scissor
scissorStack.pop();
// Reset the previous scissor
var scissor = scissorStack[scissorStack.length - 1];
if (scissor)
{
this.setScissor(scissor[0], scissor[1], scissor[2], scissor[3]);
}
2018-01-24 18:55:23 +00:00
this.currentScissor = scissor;
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Binds a WebGLPipeline and sets it as the current pipeline to be used.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setPipeline
* @since 3.0.0
*
2018-07-11 15:24:20 +00:00
* @param {Phaser.Renderer.WebGL.WebGLPipeline} pipelineInstance - The pipeline instance to be activated.
* @param {Phaser.GameObjects.GameObject} [gameObject] - The Game Object that invoked this pipeline, if any.
2018-02-09 19:19:21 +00:00
*
2018-07-11 15:24:20 +00:00
* @return {Phaser.Renderer.WebGL.WebGLPipeline} The pipeline that was activated.
2018-02-09 19:19:21 +00:00
*/
2018-07-11 15:24:20 +00:00
setPipeline: function (pipelineInstance, gameObject)
2018-01-20 04:05:56 +00:00
{
2018-01-25 05:26:14 +00:00
if (this.currentPipeline !== pipelineInstance ||
2018-01-30 03:38:31 +00:00
this.currentPipeline.vertexBuffer !== this.currentVertexBuffer ||
this.currentPipeline.program !== this.currentProgram)
2018-01-20 04:05:56 +00:00
{
2018-01-24 03:08:14 +00:00
this.flush();
2018-01-23 16:38:58 +00:00
this.currentPipeline = pipelineInstance;
2018-01-30 03:38:31 +00:00
this.currentPipeline.bind();
2018-01-20 04:05:56 +00:00
}
2018-07-11 15:24:20 +00:00
this.currentPipeline.onBind(gameObject);
2018-01-25 05:26:14 +00:00
2018-01-23 19:29:47 +00:00
return this.currentPipeline;
2018-01-20 04:05:56 +00:00
},
2019-04-26 18:13:24 +00:00
/**
* Is there an active stencil mask?
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#hasActiveStencilMask
* @since 3.17.0
*
* @return {boolean} `true` if there is an active stencil mask, otherwise `false`.
*/
hasActiveStencilMask: function ()
{
2019-04-29 13:45:54 +00:00
var mask = this.currentMask.mask;
2019-04-29 13:28:58 +00:00
var camMask = this.currentCameraMask.mask;
2019-04-26 18:13:24 +00:00
return ((mask && mask.isStencil) || (camMask && camMask.isStencil));
},
/**
* Use this to reset the gl context to the state that Phaser requires to continue rendering.
* Calling this will:
*
* * Disable `DEPTH_TEST`, `CULL_FACE` and `STENCIL_TEST`.
* * Clear the depth buffer and stencil buffers.
* * Reset the viewport size.
* * Reset the blend mode.
* * Bind a blank texture as the active texture on texture unit zero.
* * Rebinds the given pipeline instance.
*
* You should call this having previously called `clearPipeline` and then wishing to return
* control to Phaser again.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#rebindPipeline
* @since 3.16.0
*
* @param {Phaser.Renderer.WebGL.WebGLPipeline} pipelineInstance - The pipeline instance to be activated.
*/
rebindPipeline: function (pipelineInstance)
{
var gl = this.gl;
gl.disable(gl.DEPTH_TEST);
gl.disable(gl.CULL_FACE);
2019-04-26 18:13:24 +00:00
if (this.hasActiveStencilMask())
{
2019-04-26 18:13:24 +00:00
gl.clear(gl.DEPTH_BUFFER_BIT);
}
else
{
2019-04-26 18:13:24 +00:00
// If there wasn't a stencil mask set before this call, we can disable it safely
gl.disable(gl.STENCIL_TEST);
gl.clear(gl.DEPTH_BUFFER_BIT | gl.STENCIL_BUFFER_BIT);
}
gl.viewport(0, 0, this.width, this.height);
this.setBlendMode(0, true);
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, this.blankTexture.glTexture);
this.currentActiveTextureUnit = 0;
this.currentTextures[0] = this.blankTexture.glTexture;
this.currentPipeline = pipelineInstance;
this.currentPipeline.bind();
this.currentPipeline.onBind();
},
/**
* Flushes the current WebGLPipeline being used and then clears it, along with the
* the current shader program and vertex buffer. Then resets the blend mode to NORMAL.
* Call this before jumping to your own gl context handler, and then call `rebindPipeline` when
* you wish to return control to Phaser again.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#clearPipeline
* @since 3.16.0
*/
clearPipeline: function ()
{
this.flush();
this.currentPipeline = null;
this.currentProgram = null;
this.currentVertexBuffer = null;
this.currentIndexBuffer = null;
this.setBlendMode(0, true);
},
2018-02-09 19:19:21 +00:00
/**
2018-07-23 00:39:32 +00:00
* Sets the blend mode to the value given.
*
* If the current blend mode is different from the one given, the pipeline is flushed and the new
* blend mode is enabled.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setBlendMode
* @since 3.0.0
*
2018-07-23 00:39:32 +00:00
* @param {integer} blendModeId - The blend mode to be set. Can be a `BlendModes` const or an integer value.
* @param {boolean} [force=false] - Force the blend mode to be set, regardless of the currently set blend mode.
2018-02-09 19:19:21 +00:00
*
2018-07-23 00:39:32 +00:00
* @return {boolean} `true` if the blend mode was changed as a result of this call, forcing a flush, otherwise `false`.
2018-02-09 19:19:21 +00:00
*/
setBlendMode: function (blendModeId, force)
{
if (force === undefined) { force = false; }
var gl = this.gl;
2018-01-17 21:25:43 +00:00
var blendMode = this.blendModes[blendModeId];
2017-02-07 16:12:20 +00:00
if (force || (blendModeId !== CONST.BlendModes.SKIP_CHECK && this.currentBlendMode !== blendModeId))
2017-01-19 17:53:20 +00:00
{
2018-01-20 04:05:56 +00:00
this.flush();
2017-12-15 04:07:16 +00:00
2018-01-17 21:25:43 +00:00
gl.enable(gl.BLEND);
2018-11-16 10:46:22 +00:00
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;
2018-07-23 00:39:32 +00:00
return true;
2017-02-10 00:48:32 +00:00
}
2018-07-23 00:39:32 +00:00
return false;
},
/**
* Creates a new custom blend mode for the renderer.
2019-06-28 13:53:12 +00:00
*
* See https://developer.mozilla.org/en-US/docs/Web/API/WebGL_API/Constants#Blending_modes
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#addBlendMode
* @since 3.0.0
*
2019-06-28 13:53:12 +00:00
* @param {GLenum[]} func - An array containing the WebGL functions to use for the source and the destination blending factors, respectively. See the possible constants for {@link WebGLRenderingContext#blendFunc()}.
* @param {GLenum} equation - The equation to use for combining the RGB and alpha components of a new pixel with a rendered one. See the possible constants for {@link WebGLRenderingContext#blendEquation()}.
*
* @return {integer} The index of the new blend mode, used for referencing it in the future.
*/
addBlendMode: function (func, equation)
{
var index = this.blendModes.push({ func: func, equation: equation });
return index - 1;
},
/**
2018-09-28 11:19:21 +00:00
* Updates the function bound to a given custom blend mode.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#updateBlendMode
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {integer} index - The index of the custom blend mode.
* @param {function} func - The function to use for the blend mode.
* @param {function} equation - The equation to use for the blend mode.
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
*/
updateBlendMode: function (index, func, equation)
{
if (this.blendModes[index])
{
this.blendModes[index].func = func;
if (equation)
{
this.blendModes[index].equation = equation;
}
}
return this;
},
/**
2018-09-28 11:19:21 +00:00
* Removes a custom blend mode from the renderer.
* Any Game Objects still using this blend mode will error, so be sure to clear them first.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#removeBlendMode
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {integer} index - The index of the custom blend mode to be removed.
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
*/
removeBlendMode: function (index)
{
2018-11-10 04:22:13 +00:00
if (index > 17 && this.blendModes[index])
{
this.blendModes.splice(index, 1);
}
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-07-26 14:04:46 +00:00
/**
* Sets the current active texture for texture unit zero to be a blank texture.
* This only happens if there isn't a texture already in use by texture unit zero.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setBlankTexture
* @private
* @since 3.12.0
*
* @param {boolean} [force=false] - Force a blank texture set, regardless of what's already bound?
2018-07-26 14:04:46 +00:00
*/
setBlankTexture: function (force)
2018-07-26 14:04:46 +00:00
{
if (force === undefined) { force = false; }
if (force || this.currentActiveTextureUnit !== 0 || !this.currentTextures[0])
2018-07-26 14:04:46 +00:00
{
this.setTexture2D(this.blankTexture.glTexture, 0);
}
},
2018-02-09 19:19:21 +00:00
/**
* Binds a texture at a texture unit. If a texture is already
2018-04-25 23:30:41 +00:00
* bound to that unit it will force a flush on the current pipeline.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setTexture2D
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLTexture} texture - The WebGL texture that needs to be bound.
* @param {integer} textureUnit - The texture unit to which the texture will be bound.
2018-12-12 11:09:00 +00:00
* @param {boolean} [flush=true] - Will the current pipeline be flushed if this is a new texture, or not?
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-12-11 14:59:50 +00:00
setTexture2D: function (texture, textureUnit, flush)
2017-02-07 19:30:50 +00:00
{
2018-12-11 14:59:50 +00:00
if (flush === undefined) { flush = true; }
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-12-11 14:59:50 +00:00
if (flush)
{
this.flush();
}
2018-01-23 19:29:47 +00:00
if (this.currentActiveTextureUnit !== textureUnit)
{
gl.activeTexture(gl.TEXTURE0 + textureUnit);
this.currentActiveTextureUnit = textureUnit;
}
2018-01-17 21:25:43 +00:00
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-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Binds a framebuffer. If there was another framebuffer already bound it will force a pipeline flush.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFramebuffer
* @since 3.0.0
*
* @param {WebGLFramebuffer} framebuffer - The framebuffer that needs to be bound.
* @param {boolean} [updateScissor=false] - If a framebuffer is given, set the gl scissor to match the frame buffer size? Or, if `null` given, pop the scissor from the stack.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
setFramebuffer: function (framebuffer, updateScissor)
2017-01-23 21:42:47 +00:00
{
if (updateScissor === undefined) { updateScissor = false; }
var gl = this.gl;
var width = this.width;
var height = this.height;
2018-01-17 21:25:43 +00:00
if (framebuffer !== this.currentFramebuffer)
{
if (framebuffer && framebuffer.renderTexture)
{
width = framebuffer.renderTexture.width;
height = framebuffer.renderTexture.height;
}
else
{
this.flush();
}
2018-01-23 19:29:47 +00:00
2018-01-17 21:25:43 +00:00
gl.bindFramebuffer(gl.FRAMEBUFFER, framebuffer);
gl.viewport(0, 0, width, height);
if (updateScissor)
{
if (framebuffer)
{
this.drawingBufferHeight = height;
this.pushScissor(0, 0, width, height);
}
else
{
this.drawingBufferHeight = this.height;
this.popScissor();
}
}
2018-01-25 05:26:14 +00:00
this.currentFramebuffer = framebuffer;
}
2018-01-17 21:25:43 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Binds a program. If there was another program already bound it will force a pipeline flush.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setProgram
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLProgram} program - The program that needs to be bound.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setProgram: function (program)
{
var gl = this.gl;
if (program !== this.currentProgram)
{
2018-01-23 19:29:47 +00:00
this.flush();
2018-01-22 21:21:47 +00:00
gl.useProgram(program);
2018-01-25 05:26:14 +00:00
this.currentProgram = program;
2018-01-22 21:21:47 +00:00
}
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Bounds a vertex buffer. If there is a vertex buffer already bound it'll force a pipeline flush.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setVertexBuffer
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLBuffer} vertexBuffer - The buffer that needs to be bound.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
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-23 19:29:47 +00:00
this.flush();
2018-01-17 21:25:43 +00:00
gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
2018-01-25 05:26:14 +00:00
this.currentVertexBuffer = vertexBuffer;
}
2018-01-17 21:25:43 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Bounds a index buffer. If there is a index buffer already bound it'll force a pipeline flush.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setIndexBuffer
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLBuffer} indexBuffer - The buffer the needs to be bound.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
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-23 19:29:47 +00:00
this.flush();
2018-03-19 20:41:24 +00:00
2018-01-17 21:25:43 +00:00
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, indexBuffer);
2018-01-25 05:26:14 +00:00
this.currentIndexBuffer = 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-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Creates a texture from an image source. If the source is not valid it creates an empty texture.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createTextureFromSource
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {object} source - The source of the texture.
* @param {integer} width - The width of the texture.
* @param {integer} height - The height of the texture.
* @param {integer} scaleMode - The scale mode to be used by the texture.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {?WebGLTexture} The WebGL Texture that was created, or `null` if it couldn't be created.
2018-02-09 19:19:21 +00:00
*/
2018-01-24 00:40:20 +00:00
createTextureFromSource: function (source, width, height, scaleMode)
2018-02-16 18:17:51 +00:00
{
2018-01-20 04:05:56 +00:00
var gl = this.gl;
var filter = gl.NEAREST;
var wrap = gl.CLAMP_TO_EDGE;
2018-01-24 00:40:20 +00:00
var texture = null;
2018-01-20 04:05:56 +00:00
width = source ? source.width : width;
height = source ? source.height : height;
if (IsSizePowerOfTwo(width, height))
{
wrap = gl.REPEAT;
}
if (scaleMode === CONST.ScaleModes.LINEAR && this.config.antialias)
2018-01-20 04:05:56 +00:00
{
2018-01-24 00:40:20 +00:00
filter = gl.LINEAR;
}
2018-01-20 04:05:56 +00:00
2018-01-24 00:40:20 +00:00
if (!source && typeof width === 'number' && typeof height === 'number')
{
texture = this.createTexture2D(0, filter, filter, wrap, wrap, gl.RGBA, null, width, height);
}
else
{
texture = this.createTexture2D(0, filter, filter, wrap, wrap, gl.RGBA, source);
2018-01-20 04:05:56 +00:00
}
2018-01-24 00:40:20 +00:00
return texture;
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* A wrapper for creating a WebGLTexture. If no pixel data is passed it will create an empty texture.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createTexture2D
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {integer} mipLevel - Mip level of the texture.
* @param {integer} minFilter - Filtering of the texture.
* @param {integer} magFilter - Filtering of the texture.
* @param {integer} wrapT - Wrapping mode of the texture.
* @param {integer} wrapS - Wrapping mode of the texture.
* @param {integer} format - Which format does the texture use.
* @param {object} pixels - pixel data.
* @param {integer} width - Width of the texture in pixels.
* @param {integer} height - Height of the texture in pixels.
* @param {boolean} pma - Does the texture have premultiplied alpha?
2018-04-25 23:30:41 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {WebGLTexture} The WebGLTexture that was created.
2018-02-09 19:19:21 +00:00
*/
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
{
pma = (pma === undefined || pma === null) ? true : pma;
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var texture = gl.createTexture();
this.setTexture2D(texture, 0);
2018-03-19 20:41:24 +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);
2018-01-20 00:37:52 +00:00
width = pixels.width;
height = pixels.height;
}
this.setTexture2D(null, 0);
texture.isAlphaPremultiplied = pma;
texture.isRenderTexture = false;
texture.width = width;
texture.height = height;
this.nativeTextures.push(texture);
2018-01-20 00:37:52 +00:00
return texture;
2018-01-17 21:25:43 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Wrapper for creating WebGLFramebuffer.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createFramebuffer
* @since 3.0.0
*
2018-04-25 23:30:41 +00:00
* @param {integer} width - Width in pixels of the framebuffer
* @param {integer} height - Height in pixels of the framebuffer
* @param {WebGLTexture} renderTexture - The color texture to where the color pixels are written
2018-04-25 23:30:41 +00:00
* @param {boolean} addDepthStencilBuffer - Indicates if the current framebuffer support depth and stencil buffers
2018-02-09 19:19:21 +00:00
*
2018-04-25 23:30:41 +00:00
* @return {WebGLFramebuffer} Raw WebGLFramebuffer
2018-02-09 19:19:21 +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'
};
2018-01-20 00:37:52 +00:00
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-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Wrapper for creating a WebGLProgram
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createProgram
* @since 3.0.0
*
2018-04-25 23:30:41 +00:00
* @param {string} vertexShader - Source to the vertex shader
* @param {string} fragmentShader - Source to the fragment shader
2018-02-09 19:19:21 +00:00
*
2018-04-25 23:30:41 +00:00
* @return {WebGLProgram} Raw WebGLProgram
2018-02-09 19:19:21 +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))
{
throw new Error('Failed to compile Vertex Shader:\n' + gl.getShaderInfoLog(vs));
2018-01-20 00:37:52 +00:00
}
if (!gl.getShaderParameter(fs, gl.COMPILE_STATUS))
{
throw new Error('Failed to compile Fragment Shader:\n' + gl.getShaderInfoLog(fs));
2018-01-20 00:37:52 +00:00
}
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))
{
throw new Error('Failed to link program:\n' + gl.getProgramInfoLog(program));
2018-01-20 00:37:52 +00:00
}
return program;
2018-01-17 21:25:43 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Wrapper for creating a vertex buffer.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createVertexBuffer
* @since 3.0.0
*
2018-04-25 23:30:41 +00:00
* @param {ArrayBuffer} initialDataOrSize - It's either ArrayBuffer or an integer indicating the size of the vbo
* @param {integer} bufferUsage - How the buffer is used. gl.DYNAMIC_DRAW, gl.STATIC_DRAW or gl.STREAM_DRAW
2018-02-09 19:19:21 +00:00
*
2018-04-25 23:30:41 +00:00
* @return {WebGLBuffer} Raw vertex buffer
2018-02-09 19:19:21 +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);
2018-01-20 00:37:52 +00:00
gl.bufferData(gl.ARRAY_BUFFER, initialDataOrSize, bufferUsage);
2018-01-20 00:37:52 +00:00
this.setVertexBuffer(null);
return vertexBuffer;
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Wrapper for creating a vertex buffer.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createIndexBuffer
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {ArrayBuffer} initialDataOrSize - Either ArrayBuffer or an integer indicating the size of the vbo.
* @param {integer} bufferUsage - How the buffer is used. gl.DYNAMIC_DRAW, gl.STATIC_DRAW or gl.STREAM_DRAW.
2018-02-09 19:19:21 +00:00
*
2018-04-25 23:30:41 +00:00
* @return {WebGLBuffer} Raw index buffer
2018-02-09 19:19:21 +00:00
*/
2018-01-25 00:15:51 +00:00
createIndexBuffer: function (initialDataOrSize, bufferUsage)
{
2018-01-20 00:37:52 +00:00
var gl = this.gl;
var indexBuffer = gl.createBuffer();
this.setIndexBuffer(indexBuffer);
2018-01-20 00:37:52 +00:00
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, initialDataOrSize, bufferUsage);
2018-01-20 00:37:52 +00:00
this.setIndexBuffer(null);
return indexBuffer;
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Removes the given texture from the nativeTextures array and then deletes it from the GPU.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteTexture
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLTexture} texture - The WebGL Texture to be deleted.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
deleteTexture: function (texture)
2018-01-25 00:15:51 +00:00
{
var index = this.nativeTextures.indexOf(texture);
if (index !== -1)
{
SpliceOne(this.nativeTextures, index);
}
this.gl.deleteTexture(texture);
2019-02-04 23:29:22 +00:00
if (this.currentTextures[0] === texture && !this.game.pendingDestroy)
{
// texture we just deleted is in use, so bind a blank texture
this.setBlankTexture(true);
}
2018-01-25 00:15:51 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Deletes a WebGLFramebuffer from the GL instance.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteFramebuffer
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLFramebuffer} framebuffer - The Framebuffer to be deleted.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
deleteFramebuffer: function (framebuffer)
2018-01-25 00:15:51 +00:00
{
this.gl.deleteFramebuffer(framebuffer);
2018-01-25 00:15:51 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Deletes a WebGLProgram from the GL instance.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteProgram
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLProgram} program - The shader program to be deleted.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
deleteProgram: function (program)
2018-01-25 00:15:51 +00:00
{
this.gl.deleteProgram(program);
2018-01-25 00:15:51 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Deletes a WebGLBuffer from the GL instance.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteBuffer
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {WebGLBuffer} vertexBuffer - The WebGLBuffer to be deleted.
2018-02-09 19:19:21 +00:00
*
2018-09-28 11:19:21 +00:00
* @return {this} This WebGLRenderer instance.
2018-02-09 19:19:21 +00:00
*/
deleteBuffer: function (buffer)
2018-01-25 00:15:51 +00:00
{
this.gl.deleteBuffer(buffer);
2018-01-25 00:15:51 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Controls the pre-render operations for the given camera.
* Handles any clipping needed by the camera and renders the background color if a color is visible.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#preRenderCamera
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - The Camera to pre-render.
2018-02-09 19:19:21 +00:00
*/
2018-01-24 03:57:33 +00:00
preRenderCamera: function (camera)
{
2018-07-18 13:32:06 +00:00
var cx = camera._cx;
var cy = camera._cy;
var cw = camera._cw;
var ch = camera._ch;
var TextureTintPipeline = this.pipelines.TextureTintPipeline;
var color = camera.backgroundColor;
if (camera.renderToTexture)
2018-01-24 03:57:33 +00:00
{
2018-08-31 17:19:31 +00:00
this.flush();
this.pushScissor(cx, cy, cw, -ch);
this.setFramebuffer(camera.framebuffer);
2018-01-24 03:57:33 +00:00
var gl = this.gl;
gl.clearColor(0, 0, 0, 0);
gl.clear(gl.COLOR_BUFFER_BIT);
TextureTintPipeline.projOrtho(cx, cw + cx, cy, ch + cy, -1000, 1000);
2019-04-29 13:28:58 +00:00
if (camera.mask)
{
this.currentCameraMask.mask = camera.mask;
this.currentCameraMask.camera = camera._maskCamera;
camera.mask.preRenderWebGL(this, camera, camera._maskCamera);
}
if (color.alphaGL > 0)
{
TextureTintPipeline.drawFillRect(
cx, cy, cw + cx, ch + cy,
Utils.getTintFromFloats(color.redGL, color.greenGL, color.blueGL, 1),
color.alphaGL
);
}
camera.emit(CameraEvents.PRE_RENDER, camera);
}
else
{
this.pushScissor(cx, cy, cw, ch);
2019-04-12 16:47:28 +00:00
if (camera.mask)
2019-04-10 17:10:03 +00:00
{
2019-04-29 13:28:58 +00:00
this.currentCameraMask.mask = camera.mask;
this.currentCameraMask.camera = camera._maskCamera;
camera.mask.preRenderWebGL(this, camera, camera._maskCamera);
2019-04-10 17:10:03 +00:00
}
if (color.alphaGL > 0)
{
TextureTintPipeline.drawFillRect(
cx, cy, cw , ch,
Utils.getTintFromFloats(color.redGL, color.greenGL, color.blueGL, 1),
color.alphaGL
);
}
}
2018-01-24 03:57:33 +00:00
},
2019-04-29 13:28:58 +00:00
getCurrentStencilMask: function ()
{
var prev = null;
var stack = this.maskStack;
var cameraMask = this.currentCameraMask;
if (stack.length > 0)
{
prev = stack[stack.length - 1];
}
else if (cameraMask.mask && cameraMask.mask.isStencil)
{
prev = cameraMask;
}
return prev;
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* Controls the post-render operations for the given camera.
* Renders the foreground camera effects like flash and fading. It resets the current scissor state.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#postRenderCamera
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - The Camera to post-render.
2018-02-09 19:19:21 +00:00
*/
2018-01-24 03:57:33 +00:00
postRenderCamera: function (camera)
{
var TextureTintPipeline = this.pipelines.TextureTintPipeline;
2018-01-24 03:57:33 +00:00
camera.flashEffect.postRenderWebGL(TextureTintPipeline, Utils.getTintFromFloats);
camera.fadeEffect.postRenderWebGL(TextureTintPipeline, Utils.getTintFromFloats);
2018-01-24 03:57:33 +00:00
camera.dirty = false;
2018-01-24 18:55:23 +00:00
this.popScissor();
if (camera.renderToTexture)
{
TextureTintPipeline.flush();
this.setFramebuffer(null);
camera.emit(CameraEvents.POST_RENDER, camera);
TextureTintPipeline.projOrtho(0, TextureTintPipeline.width, TextureTintPipeline.height, 0, -1000.0, 1000.0);
var getTint = Utils.getTintAppendFloatAlpha;
2018-08-08 00:33:55 +00:00
2018-08-31 17:19:31 +00:00
var pipeline = (camera.pipeline) ? camera.pipeline : TextureTintPipeline;
2018-08-31 17:19:31 +00:00
pipeline.batchTexture(
camera,
camera.glTexture,
camera.width, camera.height,
camera.x, camera.y,
camera.width, camera.height,
camera.zoom, camera.zoom,
camera.rotation,
camera.flipX, !camera.flipY,
1, 1,
0, 0,
0, 0, camera.width, camera.height,
2018-08-08 00:33:55 +00:00
getTint(camera._tintTL, camera._alphaTL),
getTint(camera._tintTR, camera._alphaTR),
getTint(camera._tintBL, camera._alphaBL),
getTint(camera._tintBR, camera._alphaBR),
(camera._isTinted && camera.tintFill),
0, 0,
this.defaultCamera,
null
);
// Force clear the current texture so that items next in the batch (like Graphics) don't try and use it
this.setBlankTexture(true);
}
2019-04-12 16:47:28 +00:00
if (camera.mask)
{
2019-04-29 13:28:58 +00:00
this.currentCameraMask.mask = null;
2019-04-26 18:13:24 +00:00
camera.mask.postRenderWebGL(this, camera._maskCamera);
2019-04-12 16:47:28 +00:00
}
2018-01-24 03:57:33 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Clears the current vertex buffer and updates pipelines.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#preRender
* @since 3.0.0
*/
2018-01-20 04:05:56 +00:00
preRender: function ()
{
2018-02-16 18:17:51 +00:00
if (this.contextLost) { return; }
2018-01-20 04:05:56 +00:00
2018-01-22 22:51:15 +00:00
var gl = this.gl;
2018-01-26 23:17:11 +00:00
var pipelines = this.pipelines;
2018-01-22 22:51:15 +00:00
// Make sure we are bound to the main frame buffer
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
if (this.config.clearBeforeRender)
{
var clearColor = this.config.backgroundColor;
gl.clearColor(clearColor.redGL, clearColor.greenGL, clearColor.blueGL, clearColor.alphaGL);
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT | gl.STENCIL_BUFFER_BIT);
}
2018-01-26 23:17:11 +00:00
gl.enable(gl.SCISSOR_TEST);
2018-01-26 23:17:11 +00:00
for (var key in pipelines)
{
pipelines[key].onPreRender();
}
// TODO - Find a way to stop needing to create these arrays every frame
// and equally not need a huge array buffer created to hold them
this.currentScissor = [ 0, 0, this.width, this.height ];
this.scissorStack = [ this.currentScissor ];
if (this.game.scene.customViewports)
{
gl.scissor(0, (this.drawingBufferHeight - this.height), this.width, this.height);
}
2018-07-26 16:50:07 +00:00
2019-04-29 13:45:54 +00:00
this.currentMask.mask = null;
2019-04-29 13:28:58 +00:00
this.currentCameraMask.mask = null;
this.maskStack.length = 0;
this.setPipeline(this.pipelines.TextureTintPipeline);
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-11-16 10:56:43 +00:00
* The core render step for a Scene Camera.
*
2018-09-28 11:19:21 +00:00
* Iterates through the given Game Object's array and renders them with the given Camera.
2018-11-16 10:56:43 +00:00
*
* This is called by the `CameraManager.render` method. The Camera Manager instance belongs to a Scene, and is invoked
* by the Scene Systems.render method.
*
* This method is not called if `Camera.visible` is `false`, or `Camera.alpha` is zero.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#render
* @since 3.0.0
*
2018-09-28 11:19:21 +00:00
* @param {Phaser.Scene} scene - The Scene to render.
* @param {Phaser.GameObjects.GameObject} children - The Game Object's within the Scene to be rendered.
* @param {number} interpolationPercentage - The interpolation percentage to apply. Currently un-used.
* @param {Phaser.Cameras.Scene2D.Camera} camera - The Scene Camera to render with.
2018-02-09 19:19:21 +00:00
*/
2018-01-20 04:05:56 +00:00
render: function (scene, children, interpolationPercentage, camera)
{
2018-02-16 18:17:51 +00:00
if (this.contextLost) { return; }
2018-01-20 04:05:56 +00:00
2018-01-22 21:21:47 +00:00
var list = children.list;
var childCount = list.length;
2018-01-29 21:46:48 +00:00
var pipelines = this.pipelines;
for (var key in pipelines)
{
pipelines[key].onRender(scene, camera);
}
2018-01-22 21:37:47 +00:00
// Apply scissor for cam region + render background color, if not transparent
2018-01-24 03:57:33 +00:00
this.preRenderCamera(camera);
2018-01-22 21:37:47 +00:00
2019-04-29 13:45:54 +00:00
var current = this.currentMask;
for (var i = 0; i < childCount; i++)
2018-01-22 21:21:47 +00:00
{
var child = list[i];
2018-01-22 21:21:47 +00:00
if (!child.willRender(camera))
2018-01-22 21:21:47 +00:00
{
continue;
}
2018-01-22 21:37:47 +00:00
if (child.blendMode !== this.currentBlendMode)
{
this.setBlendMode(child.blendMode);
}
var mask = child.mask;
2019-04-29 13:45:54 +00:00
current = this.currentMask;
if (current.mask && current.mask !== mask)
2018-01-22 21:37:47 +00:00
{
2019-04-29 13:28:58 +00:00
// Render out the previously set mask
2019-04-29 13:45:54 +00:00
current.mask.postRenderWebGL(this, current.camera);
2019-04-29 13:28:58 +00:00
}
2019-04-29 13:45:54 +00:00
if (mask && current.mask !== mask)
2019-04-29 13:28:58 +00:00
{
mask.preRenderWebGL(this, child, camera);
}
child.renderWebGL(this, child, interpolationPercentage, camera);
}
2019-04-29 13:45:54 +00:00
current = this.currentMask;
if (current.mask)
{
2019-04-26 18:13:24 +00:00
// Render out the previously set mask, if it was the last item in the display list
2019-04-29 13:45:54 +00:00
current.mask.postRenderWebGL(this, current.camera);
2018-01-22 21:37:47 +00:00
}
2018-01-24 03:57:33 +00:00
this.setBlendMode(CONST.BlendModes.NORMAL);
// Applies camera effects and pops the scissor, if set
2018-01-24 03:57:33 +00:00
this.postRenderCamera(camera);
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
2018-09-28 11:19:21 +00:00
* The post-render step happens after all Cameras in all Scenes have been rendered.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#postRender
* @since 3.0.0
*/
2018-01-20 04:05:56 +00:00
postRender: function ()
{
2018-02-16 18:17:51 +00:00
if (this.contextLost) { return; }
2018-01-20 04:05:56 +00:00
this.flush();
2018-01-22 22:51:15 +00:00
// Unbind custom framebuffer here
2019-01-25 10:01:53 +00:00
var state = this.snapshotState;
if (state.callback)
2018-01-20 04:05:56 +00:00
{
WebGLSnapshot(this.canvas, state);
2019-01-25 10:01:53 +00:00
state.callback = null;
2018-01-20 04:05:56 +00:00
}
2018-01-26 23:17:11 +00:00
var pipelines = this.pipelines;
for (var key in pipelines)
{
pipelines[key].onPostRender();
}
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Schedules a snapshot of the entire game viewport to be taken after the current frame is rendered.
*
* To capture a specific area see the `snapshotArea` method. To capture a specific pixel, see `snapshotPixel`.
*
* Only one snapshot can be active _per frame_. If you have already called `snapshotPixel`, for example, then
* calling this method will override it.
*
* Snapshots work by using the WebGL `readPixels` feature to grab every pixel from the frame buffer into an ArrayBufferView.
* It then parses this, copying the contents to a temporary Canvas and finally creating an Image object from it,
* which is the image returned to the callback provided. All in all, this is a computationally expensive and blocking process,
* which gets more expensive the larger the canvas size gets, so please be careful how you employ this in your game.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#snapshot
* @since 3.0.0
*
2019-05-09 14:32:53 +00:00
* @param {Phaser.Types.Renderer.Snapshot.SnapshotCallback} callback - The Function to invoke after the snapshot image is created.
* @param {string} [type='image/png'] - The format of the image to create, usually `image/png` or `image/jpeg`.
* @param {number} [encoderOptions=0.92] - The image quality, between 0 and 1. Used for image formats with lossy compression, such as `image/jpeg`.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer.
2018-02-09 19:19:21 +00:00
*/
2018-01-20 04:05:56 +00:00
snapshot: function (callback, type, encoderOptions)
{
return this.snapshotArea(0, 0, this.gl.drawingBufferWidth, this.gl.drawingBufferHeight, callback, type, encoderOptions);
},
/**
* Schedules a snapshot of the given area of the game viewport to be taken after the current frame is rendered.
*
* To capture the whole game viewport see the `snapshot` method. To capture a specific pixel, see `snapshotPixel`.
*
* Only one snapshot can be active _per frame_. If you have already called `snapshotPixel`, for example, then
* calling this method will override it.
*
* Snapshots work by using the WebGL `readPixels` feature to grab every pixel from the frame buffer into an ArrayBufferView.
* It then parses this, copying the contents to a temporary Canvas and finally creating an Image object from it,
* which is the image returned to the callback provided. All in all, this is a computationally expensive and blocking process,
* which gets more expensive the larger the canvas size gets, so please be careful how you employ this in your game.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#snapshotArea
* @since 3.16.0
*
* @param {integer} x - The x coordinate to grab from.
* @param {integer} y - The y coordinate to grab from.
* @param {integer} width - The width of the area to grab.
* @param {integer} height - The height of the area to grab.
2019-05-09 14:32:53 +00:00
* @param {Phaser.Types.Renderer.Snapshot.SnapshotCallback} callback - The Function to invoke after the snapshot image is created.
* @param {string} [type='image/png'] - The format of the image to create, usually `image/png` or `image/jpeg`.
* @param {number} [encoderOptions=0.92] - The image quality, between 0 and 1. Used for image formats with lossy compression, such as `image/jpeg`.
*
* @return {this} This WebGL Renderer.
*/
snapshotArea: function (x, y, width, height, callback, type, encoderOptions)
2018-01-20 04:05:56 +00:00
{
2019-01-25 10:01:53 +00:00
var state = this.snapshotState;
state.callback = callback;
state.type = type;
state.encoder = encoderOptions;
state.getPixel = false;
state.x = x;
state.y = y;
state.width = Math.min(width, this.gl.drawingBufferWidth);
state.height = Math.min(height, this.gl.drawingBufferHeight);
2019-01-25 10:01:53 +00:00
return this;
},
/**
* Schedules a snapshot of the given pixel from the game viewport to be taken after the current frame is rendered.
*
* To capture the whole game viewport see the `snapshot` method. To capture a specific area, see `snapshotArea`.
*
* Only one snapshot can be active _per frame_. If you have already called `snapshotArea`, for example, then
* calling this method will override it.
*
* Unlike the other two snapshot methods, this one will return a `Color` object containing the color data for
* the requested pixel. It doesn't need to create an internal Canvas or Image object, so is a lot faster to execute,
* using less memory.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#snapshotPixel
* @since 3.16.0
*
* @param {integer} x - The x coordinate of the pixel to get.
* @param {integer} y - The y coordinate of the pixel to get.
2019-05-09 14:32:53 +00:00
* @param {Phaser.Types.Renderer.Snapshot.SnapshotCallback} callback - The Function to invoke after the snapshot pixel data is extracted.
*
* @return {this} This WebGL Renderer.
*/
snapshotPixel: function (x, y, callback)
2019-01-25 10:01:53 +00:00
{
this.snapshotArea(x, y, 1, 1, callback);
2019-01-25 10:01:53 +00:00
this.snapshotState.getPixel = true;
2018-01-20 04:05:56 +00:00
return this;
},
/**
* Takes a snapshot of the given area of the given frame buffer.
*
* Unlike the other snapshot methods, this one is processed immediately and doesn't wait for the next render.
*
* Snapshots work by using the WebGL `readPixels` feature to grab every pixel from the frame buffer into an ArrayBufferView.
* It then parses this, copying the contents to a temporary Canvas and finally creating an Image object from it,
* which is the image returned to the callback provided. All in all, this is a computationally expensive and blocking process,
* which gets more expensive the larger the canvas size gets, so please be careful how you employ this in your game.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#snapshotFramebuffer
* @since 3.19.0
*
* @param {WebGLFramebuffer} framebuffer - The framebuffer to grab from.
* @param {integer} bufferWidth - The width of the framebuffer.
* @param {integer} bufferHeight - The height of the framebuffer.
* @param {Phaser.Types.Renderer.Snapshot.SnapshotCallback} callback - The Function to invoke after the snapshot image is created.
* @param {boolean} [getPixel=false] - Grab a single pixel as a Color object, or an area as an Image object?
* @param {integer} [x=0] - The x coordinate to grab from.
* @param {integer} [y=0] - The y coordinate to grab from.
* @param {integer} [width=bufferWidth] - The width of the area to grab.
* @param {integer} [height=bufferHeight] - The height of the area to grab.
* @param {string} [type='image/png'] - The format of the image to create, usually `image/png` or `image/jpeg`.
* @param {number} [encoderOptions=0.92] - The image quality, between 0 and 1. Used for image formats with lossy compression, such as `image/jpeg`.
*
* @return {this} This WebGL Renderer.
*/
snapshotFramebuffer: function (framebuffer, bufferWidth, bufferHeight, callback, getPixel, x, y, width, height, type, encoderOptions)
{
if (getPixel === undefined) { getPixel = false; }
if (x === undefined) { x = 0; }
if (y === undefined) { y = 0; }
if (width === undefined) { width = bufferWidth; }
if (height === undefined) { height = bufferHeight; }
var currentFramebuffer = this.currentFramebuffer;
this.snapshotArea(x, y, width, height, callback, type, encoderOptions);
var state = this.snapshotState;
state.getPixel = getPixel;
state.isFramebuffer = true;
state.bufferWidth = bufferWidth;
state.bufferHeight = bufferHeight;
this.setFramebuffer(framebuffer);
WebGLSnapshot(this.canvas, state);
this.setFramebuffer(currentFramebuffer);
state.callback = null;
state.isFramebuffer = false;
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Creates a WebGL Texture based on the given canvas element.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#canvasToTexture
* @since 3.0.0
*
* @param {HTMLCanvasElement} srcCanvas - The Canvas element that will be used to populate the texture.
* @param {WebGLTexture} [dstTexture] - Is this going to replace an existing texture? If so, pass it here.
* @param {boolean} [noRepeat=false] - Should this canvas never be allowed to set REPEAT? (such as for Text objects)
2018-02-09 19:19:21 +00:00
*
* @return {WebGLTexture} The newly created WebGL Texture.
2018-02-09 19:19:21 +00:00
*/
canvasToTexture: function (srcCanvas, dstTexture, noRepeat)
2018-01-20 04:05:56 +00:00
{
if (noRepeat === undefined) { noRepeat = false; }
2018-01-20 04:05:56 +00:00
var gl = this.gl;
2018-01-20 04:05:56 +00:00
if (!dstTexture)
{
var wrapping = gl.CLAMP_TO_EDGE;
2018-01-20 04:05:56 +00:00
if (!noRepeat && IsSizePowerOfTwo(srcCanvas.width, srcCanvas.height))
{
wrapping = gl.REPEAT;
}
var filter = (this.config.antialias) ? gl.LINEAR : gl.NEAREST;
dstTexture = this.createTexture2D(0, filter, filter, wrapping, wrapping, gl.RGBA, srcCanvas, srcCanvas.width, srcCanvas.height, true);
}
else
{
this.setTexture2D(dstTexture, 0);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, srcCanvas);
dstTexture.width = srcCanvas.width;
dstTexture.height = srcCanvas.height;
this.setTexture2D(null, 0);
}
return dstTexture;
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Sets the minification and magnification filter for a texture.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setTextureFilter
* @since 3.0.0
*
* @param {integer} texture - The texture to set the filter for.
* @param {integer} filter - The filter to set. 0 for linear filtering, 1 for nearest neighbor (blocky) filtering.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-20 04:05:56 +00:00
setTextureFilter: function (texture, filter)
{
var gl = this.gl;
var glFilter = [ gl.LINEAR, gl.NEAREST ][filter];
this.setTexture2D(texture, 0);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, glFilter);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, glFilter);
this.setTexture2D(null, 0);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* [description]
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat1
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {number} x - [description]
*
* @return {this} This WebGL Renderer instance.
*/
setFloat1: function (program, name, x)
{
this.setProgram(program);
this.gl.uniform1f(this.gl.getUniformLocation(program, name), x);
return this;
},
/**
* [description]
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat2
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {number} x - [description]
* @param {number} y - [description]
*
* @return {this} This WebGL Renderer instance.
*/
setFloat2: function (program, name, x, y)
{
this.setProgram(program);
this.gl.uniform2f(this.gl.getUniformLocation(program, name), x, y);
return this;
},
/**
* [description]
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat3
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {number} x - [description]
* @param {number} y - [description]
* @param {number} z - [description]
*
* @return {this} This WebGL Renderer instance.
*/
setFloat3: function (program, name, x, y, z)
{
this.setProgram(program);
this.gl.uniform3f(this.gl.getUniformLocation(program, name), x, y, z);
return this;
},
/**
* Sets uniform of a WebGLProgram
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat4
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {number} x - X component
* @param {number} y - Y component
* @param {number} z - Z component
* @param {number} w - W component
*
* @return {this} This WebGL Renderer instance.
*/
setFloat4: function (program, name, x, y, z, w)
{
this.setProgram(program);
this.gl.uniform4f(this.gl.getUniformLocation(program, name), x, y, z, w);
return this;
},
/**
* Sets the value of a uniform variable in the given WebGLProgram.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat1v
* @since 3.13.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {Float32Array} arr - The new value to be used for the uniform variable.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
setFloat1v: function (program, name, arr)
2018-01-22 21:21:47 +00:00
{
this.setProgram(program);
this.gl.uniform1fv(this.gl.getUniformLocation(program, name), arr);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat2v
* @since 3.13.0
2018-02-09 19:19:21 +00:00
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {Float32Array} arr - The new value to be used for the uniform variable.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
setFloat2v: function (program, name, arr)
2018-01-22 21:21:47 +00:00
{
this.setProgram(program);
this.gl.uniform2fv(this.gl.getUniformLocation(program, name), arr);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat3v
* @since 3.13.0
2018-02-09 19:19:21 +00:00
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {Float32Array} arr - The new value to be used for the uniform variable.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
setFloat3v: function (program, name, arr)
2018-01-22 21:21:47 +00:00
{
this.setProgram(program);
this.gl.uniform3fv(this.gl.getUniformLocation(program, name), arr);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFloat4v
* @since 3.13.0
2018-02-09 19:19:21 +00:00
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
* @param {Float32Array} arr - The new value to be used for the uniform variable.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
setFloat4v: function (program, name, arr)
2018-01-22 21:21:47 +00:00
{
this.setProgram(program);
this.gl.uniform4fv(this.gl.getUniformLocation(program, name), arr);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setInt1
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-02-16 18:07:49 +00:00
* @param {integer} x - [description]
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setInt1: function (program, name, x)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniform1i(this.gl.getUniformLocation(program, name), x);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setInt2
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-12-03 15:16:23 +00:00
* @param {integer} x - The new X component
* @param {integer} y - The new Y component
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setInt2: function (program, name, x, y)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniform2i(this.gl.getUniformLocation(program, name), x, y);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setInt3
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-12-03 15:16:23 +00:00
* @param {integer} x - The new X component
* @param {integer} y - The new Y component
* @param {integer} z - The new Z component
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setInt3: function (program, name, x, y, z)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniform3i(this.gl.getUniformLocation(program, name), x, y, z);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Sets the value of a uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setInt4
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-04-25 23:30:41 +00:00
* @param {integer} x - X component
* @param {integer} y - Y component
* @param {integer} z - Z component
* @param {integer} w - W component
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setInt4: function (program, name, x, y, z, w)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniform4i(this.gl.getUniformLocation(program, name), x, y, z, w);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-12-03 15:16:23 +00:00
* Sets the value of a 2x2 matrix uniform variable in the given WebGLProgram.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setMatrix2
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-12-03 15:16:23 +00:00
* @param {boolean} transpose - The value indicating whether to transpose the matrix. Must be false.
* @param {Float32Array} matrix - The new matrix value.
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setMatrix2: function (program, name, transpose, matrix)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniformMatrix2fv(this.gl.getUniformLocation(program, name), transpose, matrix);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* [description]
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setMatrix3
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-02-13 00:12:17 +00:00
* @param {boolean} transpose - [description]
* @param {Float32Array} matrix - [description]
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setMatrix3: function (program, name, transpose, matrix)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniformMatrix3fv(this.gl.getUniformLocation(program, name), transpose, matrix);
2018-01-22 21:21:47 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
2018-04-25 23:30:41 +00:00
* Sets uniform of a WebGLProgram
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setMatrix4
* @since 3.0.0
*
* @param {WebGLProgram} program - The target WebGLProgram from which the uniform location will be looked-up.
* @param {string} name - The name of the uniform to look-up and modify.
2018-04-25 23:30:41 +00:00
* @param {boolean} transpose - Is the matrix transposed
* @param {Float32Array} matrix - Matrix data
2018-02-09 19:19:21 +00:00
*
* @return {this} This WebGL Renderer instance.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setMatrix4: function (program, name, transpose, matrix)
{
this.setProgram(program);
2018-01-22 21:21:47 +00:00
this.gl.uniformMatrix4fv(this.gl.getUniformLocation(program, name), transpose, matrix);
2018-01-20 04:05:56 +00:00
return this;
},
/**
* Returns the maximum number of texture units that can be used in a fragment shader.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getMaxTextures
* @since 3.8.0
*
* @return {integer} The maximum number of textures WebGL supports.
*/
getMaxTextures: function ()
{
return this.config.maxTextures;
},
/**
* Returns the largest texture size (either width or height) that can be created.
* Note that VRAM may not allow a texture of any given size, it just expresses
* hardware / driver support for a given size.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getMaxTextureSize
* @since 3.8.0
*
* @return {integer} The maximum supported texture size.
*/
getMaxTextureSize: function ()
{
return this.config.maxTextureSize;
},
2018-02-09 19:19:21 +00:00
/**
2018-12-03 15:16:23 +00:00
* Destroy this WebGLRenderer, cleaning up all related resources such as pipelines, native textures, etc.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#destroy
* @since 3.0.0
*/
destroy: function ()
{
// Clear-up anything that should be cleared :)
for (var key in this.pipelines)
{
this.pipelines[key].destroy();
2018-04-23 18:11:16 +00:00
delete this.pipelines[key];
}
2019-02-04 23:29:22 +00:00
for (var index = 0; index < this.nativeTextures.length; index++)
{
this.deleteTexture(this.nativeTextures[index]);
2018-04-23 18:11:16 +00:00
2018-02-16 18:17:51 +00:00
delete this.nativeTextures[index];
}
delete this.gl;
delete this.game;
this.maskStack.length = 0;
this.contextLost = true;
this.extensions = {};
this.nativeTextures.length = 0;
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;