phaser/src/renderer/webgl/WebGLRenderer.js

3271 lines
108 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@phaser.io>
2018-04-05 08:02:36 +00:00
* @author Felipe Alfonso <@bitnenfer>
* @copyright 2013-2024 Phaser Studio Inc.
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 ArrayEach = require('../../utils/array/Each');
var ArrayRemove = require('../../utils/array/Remove');
var CameraEvents = require('../../cameras/2d/events');
var Class = require('../../utils/Class');
var CONST = require('../../const');
var EventEmitter = require('eventemitter3');
var Events = require('../events');
2018-01-20 04:05:56 +00:00
var IsSizePowerOfTwo = require('../../math/pow2/IsSizePowerOfTwo');
var Matrix4 = require('../../math/Matrix4');
var NOOP = require('../../utils/NOOP');
var PipelineManager = require('./PipelineManager');
var RenderTarget = require('./RenderTarget');
var ScaleEvents = require('../../scale/events');
2019-01-18 13:41:43 +00:00
var TextureEvents = require('../../textures/events');
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');
var WebGLBufferWrapper = require('./wrappers/WebGLBufferWrapper');
var WebGLProgramWrapper = require('./wrappers/WebGLProgramWrapper');
var WebGLTextureWrapper = require('./wrappers/WebGLTextureWrapper');
var WebGLFramebufferWrapper = require('./wrappers/WebGLFramebufferWrapper');
var WebGLAttribLocationWrapper = require('./wrappers/WebGLAttribLocationWrapper');
var WebGLUniformLocationWrapper = require('./wrappers/WebGLUniformLocationWrapper');
2018-01-17 21:25:43 +00:00
2022-10-31 16:17:16 +00:00
var DEBUG = false;
if (typeof WEBGL_DEBUG)
{
var SPECTOR = require('phaser3spectorjs');
2022-10-31 16:17:16 +00:00
DEBUG = true;
}
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
* @extends Phaser.Events.EventEmitter
* @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
Extends: EventEmitter,
initialize:
2018-01-17 21:25:43 +00:00
function WebGLRenderer (game)
2016-12-07 02:28:22 +00:00
{
EventEmitter.call(this);
var gameConfig = game.config;
var contextCreationConfig = {
2018-11-20 11:02:19 +00:00
alpha: gameConfig.transparent,
desynchronized: gameConfig.desynchronized,
depth: true,
2019-08-29 10:46:58 +00:00
antialias: gameConfig.antialiasGL,
premultipliedAlpha: gameConfig.premultipliedAlpha,
stencil: true,
failIfMajorPerformanceCaveat: gameConfig.failIfMajorPerformanceCaveat,
powerPreference: gameConfig.powerPreference,
preserveDrawingBuffer: gameConfig.preserveDrawingBuffer,
willReadFrequently: false
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,
roundPixels: gameConfig.roundPixels,
maxTextures: gameConfig.maxTextures,
maxTextureSize: gameConfig.maxTextureSize,
batchSize: gameConfig.batchSize,
maxLights: gameConfig.maxLights,
mipmapFilter: gameConfig.mipmapFilter
};
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
2020-11-23 10:22:13 +00:00
* @type {number}
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*/
this.type = CONST.WEBGL;
2018-02-09 19:19:21 +00:00
/**
* An instance of the Pipeline Manager class, that handles all WebGL Pipelines.
*
* Use this to manage all of your interactions with pipelines, such as adding, getting,
* setting and rendering them.
*
* The Pipeline Manager class is created in the `init` method and then populated
* with pipelines during the `boot` method.
*
* Prior to Phaser v3.50.0 this was just a plain JavaScript object, not a class.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#pipelines
* @type {Phaser.Renderer.WebGL.PipelineManager}
* @since 3.50.0
*/
this.pipelines = null;
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
2020-11-23 10:22:13 +00:00
* @type {number}
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
2020-11-23 10:22:13 +00:00
* @type {number}
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 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
/**
* This property is 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
/**
* 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
};
/**
* The maximum number of textures the GPU can handle. The minimum under the WebGL1 spec is 8.
* This is set via the Game Config `maxTextures` property and should never be changed after boot.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#maxTextures
2020-11-23 10:22:13 +00:00
* @type {number}
2020-07-31 12:41:29 +00:00
* @since 3.50.0
*/
this.maxTextures = 0;
/**
* An array of the available WebGL texture units, used to populate the uSampler uniforms.
*
* This array is populated during the init phase and should never be changed after boot.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#textureIndexes
2018-02-13 00:12:17 +00:00
* @type {array}
2020-07-31 12:41:29 +00:00
* @since 3.50.0
2018-02-09 19:19:21 +00:00
*/
this.textureIndexes;
/**
* A list of all WebGLBufferWrappers that have been created by this renderer.
2024-02-19 15:28:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glBufferWrappers
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLBufferWrapper[]}
* @since 3.80.0
*/
this.glBufferWrappers = [];
/**
* A list of all WebGLProgramWrappers that have been created by this renderer.
2024-02-19 15:28:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glProgramWrappers
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper[]}
* @since 3.80.0
*/
this.glProgramWrappers = [];
/**
* A list of all WebGLTextureWrappers that have been created by this renderer.
2024-02-19 15:28:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glTextureWrappers
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper[]}
* @since 3.80.0
*/
this.glTextureWrappers = [];
/**
* A list of all WebGLFramebufferWrappers that have been created by this renderer.
2024-02-19 15:28:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glFramebufferWrappers
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper[]}
* @since 3.80.0
*/
this.glFramebufferWrappers = [];
/**
* A list of all WebGLAttribLocationWrappers that have been created by this renderer.
2024-02-19 15:28:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glAttribLocationWrappers
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLAttribLocationWrapper[]}
* @since 3.80.0
*/
this.glAttribLocationWrappers = [];
/**
* A list of all WebGLUniformLocationWrappers that have been created by this renderer.
2024-02-19 15:28:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glUniformLocationWrappers
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLUniformLocationWrapper[]}
* @since 3.80.0
*/
this.glUniformLocationWrappers = [];
2018-02-09 19:19:21 +00:00
/**
* The currently bound framebuffer in use.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentFramebuffer
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper}
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
/**
* A stack into which the frame buffer objects are pushed and popped.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#fboStack
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper[]}
* @since 3.50.0
*/
this.fboStack = [];
/**
* Current WebGLProgram in use.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentProgram
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper}
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 blend mode in use
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentBlendMode
2020-11-23 10:22:13 +00:00
* @type {number}
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 = 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
/**
* The handler to invoke when the context is lost.
* This should not be changed and is set in the boot method.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#contextLostHandler
* @type {function}
* @since 3.19.0
*/
this.contextLostHandler = NOOP;
2018-01-17 21:25:43 +00:00
/**
* The handler to invoke when the context is restored.
* This should not be changed and is set in the boot method.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#contextRestoredHandler
* @type {function}
* @since 3.19.0
*/
this.contextRestoredHandler = NOOP;
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
/**
* Array of strings that indicate which WebGL extensions are supported by the browser.
* This is populated in the `boot` method.
2018-02-09 19:19:21 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#supportedExtensions
* @type {string[]}
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
/**
* If the browser supports the `ANGLE_instanced_arrays` extension, this property will hold
* a reference to the glExtension for it.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#instancedArraysExtension
* @type {ANGLE_instanced_arrays}
* @default null
* @since 3.50.0
*/
this.instancedArraysExtension = null;
/**
* If the browser supports the `OES_vertex_array_object` extension, this property will hold
* a reference to the glExtension for it.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#vaoExtension
* @type {OES_vertex_array_object}
* @default null
* @since 3.50.0
*/
this.vaoExtension = null;
/**
* The WebGL 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 = {};
/**
* Stores the current WebGL component formats for further use.
*
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
* This array is populated in the `init` method.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#glFormats
* @type {array}
* @since 3.2.0
*/
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
this.glFormats;
/**
* Stores the WebGL texture compression formats that this device and browser supports.
*
* Support for using compressed texture formats was added in Phaser version 3.60.
2021-10-05 15:48:42 +00:00
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#compression
* @type {Phaser.Types.Renderer.WebGL.WebGLTextureCompression}
* @since 3.8.0
*/
2021-10-05 15:48:42 +00:00
this.compression;
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 {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper}
2018-10-09 12:40:00 +00:00
* @readonly
* @since 3.12.0
*/
this.blankTexture = null;
/**
* A blank 1x1 #7f7fff texture, a flat normal map,
* as used by the Graphics system where needed.
* This is set in the `boot` method.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#normalTexture
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper}
* @readonly
* @since 3.80.0
*/
this.normalTexture = null;
/**
* A pure white 4x4 texture, as used by the Graphics system where needed.
* This is set in the `boot` method.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#whiteTexture
* @type {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper}
* @readonly
* @since 3.50.0
*/
this.whiteTexture = null;
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
2020-11-23 10:22:13 +00:00
* @type {number}
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
*
2019-05-01 11:29:45 +00:00
* @name Phaser.Renderer.WebGL.WebGLRenderer#glFuncMap
* @type {any}
* @since 3.17.0
*/
this.glFuncMap = null;
/**
* The `type` of the Game Object being currently rendered.
* This can be used by advanced render functions for batching look-ahead.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#currentType
* @type {string}
* @since 3.19.0
*/
this.currentType = '';
/**
* Is the `type` of the Game Object being currently rendered different than the
* type of the object before it in the display list? I.e. it's a 'new' type.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#newType
* @type {boolean}
* @since 3.19.0
*/
this.newType = false;
/**
* Does the `type` of the next Game Object in the display list match that
* of the object being currently rendered?
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#nextTypeMatch
* @type {boolean}
* @since 3.19.0
*/
this.nextTypeMatch = false;
/**
* Is the Game Object being currently rendered the final one in the list?
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#finalType
* @type {boolean}
* @since 3.50.0
*/
this.finalType = false;
/**
* The mipmap magFilter to be used when creating textures.
*
* You can specify this as a string in the game config, i.e.:
*
* `render: { mipmapFilter: 'NEAREST_MIPMAP_LINEAR' }`
*
* The 6 options for WebGL1 are, in order from least to most computationally expensive:
*
* NEAREST (for pixel art)
* LINEAR (the default)
* NEAREST_MIPMAP_NEAREST
* LINEAR_MIPMAP_NEAREST
* NEAREST_MIPMAP_LINEAR
* LINEAR_MIPMAP_LINEAR
*
* Mipmaps only work with textures that are fully power-of-two in size.
*
* For more details see https://webglfundamentals.org/webgl/lessons/webgl-3d-textures.html
*
* As of v3.60 no mipmaps will be generated unless a string is given in
* the game config. This saves on VRAM use when it may not be required.
* To obtain the previous result set the property to `LINEAR` in the config.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#mipmapFilter
* @type {GLenum}
* @since 3.21.0
*/
this.mipmapFilter = null;
/**
* The default scissor, set during `preRender` and modified during `resize`.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#defaultScissor
* @type {number[]}
* @private
* @since 3.50.0
*/
this.defaultScissor = [ 0, 0, 0, 0 ];
/**
* Has this renderer fully booted yet?
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#isBooted
* @type {boolean}
* @since 3.50.0
*/
this.isBooted = false;
/**
* A Render Target you can use to capture the current state of the Renderer.
*
* A Render Target encapsulates a framebuffer and texture for the WebGL Renderer.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#renderTarget
* @type {Phaser.Renderer.WebGL.RenderTarget}
* @since 3.50.0
*/
this.renderTarget = null;
/**
* The global game Projection matrix, used by shaders as 'uProjectionMatrix' uniform.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#projectionMatrix
* @type {Phaser.Math.Matrix4}
* @since 3.50.0
*/
this.projectionMatrix;
/**
* The cached width of the Projection matrix.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#projectionWidth
* @type {number}
* @since 3.50.0
*/
this.projectionWidth = 0;
/**
* The cached height of the Projection matrix.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#projectionHeight
* @type {number}
* @since 3.50.0
*/
this.projectionHeight = 0;
/**
* A RenderTarget used by the BitmapMask Pipeline.
*
* This is the source, i.e. the masked Game Object itself.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#maskSource
* @type {Phaser.Renderer.WebGL.RenderTarget}
* @since 3.60.0
*/
this.maskSource = null;
/**
* A RenderTarget used by the BitmapMask Pipeline.
*
* This is the target, i.e. the framebuffer the masked objects are drawn to.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#maskTarget
* @type {Phaser.Renderer.WebGL.RenderTarget}
* @since 3.60.0
*/
this.maskTarget = null;
/**
* An instance of SpectorJS used for WebGL Debugging.
*
* Only available in the Phaser Debug build.
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#spector
* @type {function}
* @since 3.60.0
*/
this.spector = null;
2023-04-12 02:45:19 +00:00
/**
* Is Spector currently capturing a WebGL frame?
*
* @name Phaser.Renderer.WebGL.WebGLRenderer#_debugCapture
* @type {boolean}
* @private
* @since 3.60.0
*/
this._debugCapture = false;
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;
if (DEBUG)
{
this.spector = new SPECTOR.Spector();
this.spector.onCapture.add(this.onCapture.bind(this));
}
// 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;
var _this = this;
2024-02-02 03:29:24 +00:00
// Load supported extensions
var setupExtensions = function ()
{
var exts = gl.getSupportedExtensions();
_this.supportedExtensions = exts;
var angleString = 'ANGLE_instanced_arrays';
_this.instancedArraysExtension = (exts.indexOf(angleString) > -1) ? gl.getExtension(angleString) : null;
var vaoString = 'OES_vertex_array_object';
_this.vaoExtension = (exts.indexOf(vaoString) > -1) ? gl.getExtension(vaoString) : null;
};
setupExtensions();
this.contextLostHandler = function (event)
{
_this.contextLost = true;
if (console)
{
console.warn('WebGL Context lost. Renderer disabled');
}
_this.emit(Events.LOSE_WEBGL, _this);
event.preventDefault();
};
canvas.addEventListener('webglcontextlost', this.contextLostHandler, false);
this.contextRestoredHandler = function (event)
{
if (gl.isContextLost())
{
if (console)
{
console.log('WebGL Context restored, but context is still lost');
}
return;
}
// Clear "current" settings so they can be set again.
_this.currentProgram = null;
_this.currentFramebuffer = null;
_this.setBlendMode(CONST.BlendModes.NORMAL);
// Settings we DON'T need to reset:
// Scissor is set during preRender.
// Mask is set during preRender.
// Camera mask is set during preRenderCamera.
// Restore GL flags.
gl.disable(gl.BLEND);
gl.disable(gl.DEPTH_TEST);
gl.enable(gl.CULL_FACE);
// Restore wrapped GL objects.
// Order matters, as some wrappers depend on others.
var wrapperCreateResource = function (wrapper)
{
wrapper.createResource();
};
ArrayEach(_this.glTextureWrappers, wrapperCreateResource);
ArrayEach(_this.glBufferWrappers, wrapperCreateResource);
ArrayEach(_this.glFramebufferWrappers, wrapperCreateResource);
ArrayEach(_this.glProgramWrappers, wrapperCreateResource);
ArrayEach(_this.glAttribLocationWrappers, wrapperCreateResource);
ArrayEach(_this.glUniformLocationWrappers, wrapperCreateResource);
// Create temporary textures.
_this.createTemporaryTextures();
// Restore pipelines.
_this.pipelines.restoreContext();
// Apply resize.
_this.resize(_this.game.scale.baseSize.width, _this.game.scale.baseSize.height);
2024-02-02 03:29:24 +00:00
// Restore GL extensions.
setupExtensions();
// Context has been restored.
_this.contextLost = false;
if (console)
{
console.warn('WebGL Context restored. Renderer running again.');
}
2024-02-01 22:33:27 +00:00
_this.emit(Events.RESTORE_WEBGL, _this);
event.preventDefault();
};
canvas.addEventListener('webglcontextrestored', this.contextRestoredHandler, false);
// 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
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
this.glFormats = [ gl.BYTE, gl.SHORT, gl.UNSIGNED_BYTE, gl.UNSIGNED_SHORT, 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 }
};
if (!config.maxTextures || config.maxTextures === -1)
{
config.maxTextures = gl.getParameter(gl.MAX_TEXTURE_IMAGE_UNITS);
}
2020-10-27 14:41:43 +00:00
if (!config.maxTextureSize)
{
config.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);
}
2021-10-05 15:48:42 +00:00
this.compression = this.getCompressedTextures();
// 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);
// Mipmaps
var validMipMaps = [ 'NEAREST', 'LINEAR', 'NEAREST_MIPMAP_NEAREST', 'LINEAR_MIPMAP_NEAREST', 'NEAREST_MIPMAP_LINEAR', 'LINEAR_MIPMAP_LINEAR' ];
if (validMipMaps.indexOf(config.mipmapFilter) !== -1)
{
this.mipmapFilter = gl[config.mipmapFilter];
}
// Check maximum supported textures
this.maxTextures = Utils.checkShaderMax(gl, config.maxTextures);
this.textureIndexes = [];
this.createTemporaryTextures();
this.pipelines = new PipelineManager(this);
2018-01-17 21:25:43 +00:00
this.setBlendMode(CONST.BlendModes.NORMAL);
2018-07-26 16:50:07 +00:00
this.projectionMatrix = new Matrix4().identity();
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 ()
{
var game = this.game;
var pipelineManager = this.pipelines;
2018-07-26 14:04:46 +00:00
var baseSize = game.scale.baseSize;
var width = baseSize.width;
var height = baseSize.height;
this.width = width;
this.height = height;
this.isBooted = true;
this.renderTarget = new RenderTarget(this, width, height, 1, 0, true, true);
this.maskTarget = new RenderTarget(this, width, height, 1, 0, true, true);
this.maskSource = new RenderTarget(this, width, height, 1, 0, true, true);
// Set-up pipelines
2022-10-04 17:04:26 +00:00
var config = game.config;
2022-10-04 17:04:26 +00:00
pipelineManager.boot(config.pipeline, config.defaultPipeline, config.autoMobilePipeline);
// Set-up default textures, fbo and scissor
2022-10-09 16:29:29 +00:00
this.blankTexture = game.textures.getFrame('__DEFAULT').glTexture;
this.normalTexture = game.textures.getFrame('__NORMAL').glTexture;
2022-10-09 16:29:29 +00:00
this.whiteTexture = game.textures.getFrame('__WHITE').glTexture;
2018-07-26 14:04:46 +00:00
var gl = this.gl;
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
gl.enable(gl.SCISSOR_TEST);
game.scale.on(ScaleEvents.RESIZE, this.onResize, this);
this.resize(width, height);
},
/**
* Create temporary WebGL textures to stop WebGL errors on mac os
*/
createTemporaryTextures: function ()
{
var gl = this.gl;
for (var index = 0; index < this.maxTextures; index++)
{
var tempTexture = gl.createTexture();
gl.activeTexture(gl.TEXTURE0 + index);
gl.bindTexture(gl.TEXTURE_2D, tempTexture);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, 1, 1, 0, gl.RGBA, gl.UNSIGNED_BYTE, new Uint8Array([ 0, 0, 255, 255 ]));
this.textureIndexes.push(index);
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Phaser v3.60 Debug has a build of Spector.js embedded in it, which is a WebGL inspector
* that allows for live inspection of your WebGL calls. Although it's easy to add the Spector
* extension to a desktop browsr, by embedding it in Phaser we can make it available in mobile
* browsers too, making it a powerful tool for debugging WebGL games on mobile devices where
* extensions are not permitted.
*
* See https://github.com/BabylonJS/Spector.js for more details.
*
* This method will capture the current WebGL frame and send it to the Spector.js tool for inspection.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#captureFrame
* @since 3.60.0
*
* @param {boolean} [quickCapture=false] - If `true` thumbnails are not captured in order to speed up the capture.
* @param {boolean} [fullCapture=false] - If `true` all details are captured.
*/
captureFrame: function (quickCapture, fullCapture)
{
if (quickCapture === undefined) { quickCapture = false; }
if (fullCapture === undefined) { fullCapture = false; }
if (DEBUG && this.spector && !this._debugCapture)
{
this.spector.captureCanvas(this.canvas, 0, quickCapture, fullCapture);
this._debugCapture = true;
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Phaser v3.60 Debug has a build of Spector.js embedded in it, which is a WebGL inspector
* that allows for live inspection of your WebGL calls. Although it's easy to add the Spector
* extension to a desktop browsr, by embedding it in Phaser we can make it available in mobile
* browsers too, making it a powerful tool for debugging WebGL games on mobile devices where
* extensions are not permitted.
*
* See https://github.com/BabylonJS/Spector.js for more details.
*
* This method will capture the next WebGL frame and send it to the Spector.js tool for inspection.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#captureNextFrame
* @since 3.60.0
*/
captureNextFrame: function ()
{
if (DEBUG && this.spector && !this._debugCapture)
{
this._debugCapture = true;
this.spector.captureNextFrame(this.canvas);
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Phaser v3.60 Debug has a build of Spector.js embedded in it, which is a WebGL inspector
* that allows for live inspection of your WebGL calls. Although it's easy to add the Spector
* extension to a desktop browsr, by embedding it in Phaser we can make it available in mobile
* browsers too, making it a powerful tool for debugging WebGL games on mobile devices where
* extensions are not permitted.
*
* See https://github.com/BabylonJS/Spector.js for more details.
*
* This method will return the current FPS of the WebGL canvas.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getFps
* @since 3.60.0
*
* @return {number} The current FPS of the WebGL canvas.
*/
getFps: function ()
{
if (DEBUG && this.spector)
{
return this.spector.getFps();
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Phaser v3.60 Debug has a build of Spector.js embedded in it, which is a WebGL inspector
* that allows for live inspection of your WebGL calls. Although it's easy to add the Spector
* extension to a desktop browsr, by embedding it in Phaser we can make it available in mobile
* browsers too, making it a powerful tool for debugging WebGL games on mobile devices where
* extensions are not permitted.
*
* See https://github.com/BabylonJS/Spector.js for more details.
*
* This method adds a command with the name value in the list. This can be filtered in the search.
* All logs can be filtered searching for "LOG".
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#log
* @since 3.60.0
*
* @param {...*} arguments - The arguments to log to Spector.
*
* @return {string} The current log.
*/
2023-03-02 18:45:42 +00:00
log: function ()
{
if (DEBUG && this.spector)
{
2023-03-02 18:45:42 +00:00
var t = Array.prototype.slice.call(arguments).join(' ');
return this.spector.log(t);
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Phaser v3.60 Debug has a build of Spector.js embedded in it, which is a WebGL inspector
* that allows for live inspection of your WebGL calls. Although it's easy to add the Spector
* extension to a desktop browsr, by embedding it in Phaser we can make it available in mobile
* browsers too, making it a powerful tool for debugging WebGL games on mobile devices where
* extensions are not permitted.
*
* See https://github.com/BabylonJS/Spector.js for more details.
*
* This method will start a capture on the Phaser canvas. The capture will stop once it reaches
* the number of commands specified as a parameter, or after 10 seconds. If quick capture is true,
* the thumbnails are not captured in order to speed up the capture.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#startCapture
* @since 3.60.0
*
* @param {number} [commandCount=0] - The number of commands to capture. If zero it will capture for 10 seconds.
* @param {boolean} [quickCapture=false] - If `true` thumbnails are not captured in order to speed up the capture.
* @param {boolean} [fullCapture=false] - If `true` all details are captured.
*/
startCapture: function (commandCount, quickCapture, fullCapture)
{
if (commandCount === undefined) { commandCount = 0; }
if (quickCapture === undefined) { quickCapture = false; }
if (fullCapture === undefined) { fullCapture = false; }
if (DEBUG && this.spector && !this._debugCapture)
{
this.spector.startCapture(this.canvas, commandCount, quickCapture, fullCapture);
this._debugCapture = true;
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Phaser v3.60 Debug has a build of Spector.js embedded in it, which is a WebGL inspector
* that allows for live inspection of your WebGL calls. Although it's easy to add the Spector
* extension to a desktop browsr, by embedding it in Phaser we can make it available in mobile
* browsers too, making it a powerful tool for debugging WebGL games on mobile devices where
* extensions are not permitted.
*
* See https://github.com/BabylonJS/Spector.js for more details.
*
* This method will stop the current capture and returns the result in JSON. It displays the
* result if the UI has been displayed. This returns undefined if the capture has not been completed
* or did not find any commands.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#stopCapture
* @since 3.60.0
*
* @return {object} The current capture.
*/
stopCapture: function ()
{
if (DEBUG && this.spector && this._debugCapture)
{
2023-04-12 02:45:19 +00:00
return this.spector.stopCapture();
}
},
2023-04-12 02:45:19 +00:00
/**
* This method is only available in the Debug Build of Phaser, or a build with the
* `WEBGL_DEBUG` flag set in the Webpack Config.
*
* Internal onCapture handler.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#onCapture
* @private
* @since 3.60.0
*
* @param {object} capture - The capture data.
*/
onCapture: function (capture)
{
if (DEBUG)
{
var view = this.spector.getResultUI();
view.display(capture);
this._debugCapture = false;
}
},
/**
* 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. The canvas width / height values match this.
*/
onResize: function (gameSize, baseSize)
{
// Has the underlying canvas size changed?
if (baseSize.width !== this.width || baseSize.height !== this.height)
{
this.resize(baseSize.width, baseSize.height);
}
2018-07-13 10:13:46 +00:00
},
/**
* Binds the WebGL Renderers Render Target, so all drawn content is now redirected to it.
*
* Make sure to call `endCapture` when you are finished.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#beginCapture
* @since 3.50.0
*
* @param {number} [width] - Optional new width of the Render Target.
* @param {number} [height] - Optional new height of the Render Target.
*/
beginCapture: function (width, height)
{
if (width === undefined) { width = this.width; }
if (height === undefined) { height = this.height; }
this.renderTarget.bind(true, width, height);
this.setProjectionMatrix(width, height);
},
/**
* Unbinds the WebGL Renderers Render Target and returns it, stopping any further content being drawn to it.
*
* If the viewport or scissors were modified during the capture, you should reset them by calling
* `resetViewport` and `resetScissor` accordingly.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#endCapture
* @since 3.50.0
*
* @return {Phaser.Renderer.WebGL.RenderTarget} A reference to the WebGL Renderer Render Target.
*/
endCapture: function ()
{
this.renderTarget.unbind(true);
this.resetProjectionMatrix();
return this.renderTarget;
},
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
* @fires Phaser.Renderer.Events#RESIZE
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
* @param {number} [width] - The new width of the renderer.
* @param {number} [height] - The new height 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)
{
2018-01-17 21:25:43 +00:00
var gl = this.gl;
this.width = width;
this.height = height;
2018-03-19 20:41:24 +00:00
this.setProjectionMatrix(width, height);
gl.viewport(0, 0, width, height);
2018-07-10 15:32:28 +00:00
this.drawingBufferHeight = gl.drawingBufferHeight;
gl.scissor(0, (gl.drawingBufferHeight - height), width, height);
this.defaultScissor[2] = width;
this.defaultScissor[3] = height;
this.emit(Events.RESIZE, width, height);
2018-01-17 21:25:43 +00:00
return this;
},
2021-10-05 15:48:42 +00:00
/**
* Determines which compressed texture formats this browser and device supports.
*
* Called automatically as part of the WebGL Renderer init process. If you need to investigate
* which formats it supports, see the `Phaser.Renderer.WebGL.WebGLRenderer#compression` property instead.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getCompressedTextures
* @since 3.60.0
*
* @return {Phaser.Types.Renderer.WebGL.WebGLTextureCompression} The compression object.
*/
getCompressedTextures: function ()
{
var extString = 'WEBGL_compressed_texture_';
var wkExtString = 'WEBKIT_' + extString;
var extEXTString = 'EXT_texture_compression_';
2021-10-05 15:48:42 +00:00
var hasExt = function (gl, format)
{
var results = gl.getExtension(extString + format) || gl.getExtension(wkExtString + format) || gl.getExtension(extEXTString + format);
2021-10-05 15:48:42 +00:00
if (results)
{
var glEnums = {};
for (var key in results)
{
glEnums[results[key]] = key;
}
return glEnums;
}
};
var gl = this.gl;
return {
ETC: hasExt(gl, 'etc'),
ETC1: hasExt(gl, 'etc1'),
ATC: hasExt(gl, 'atc'),
ASTC: hasExt(gl, 'astc'),
BPTC: hasExt(gl, 'bptc'),
RGTC: hasExt(gl, 'rgtc'),
PVRTC: hasExt(gl, 'pvrtc'),
S3TC: hasExt(gl, 's3tc'),
S3TCSRGB: hasExt(gl, 's3tc_srgb'),
IMG: true
};
},
2021-10-05 17:30:17 +00:00
/**
2021-10-05 17:36:50 +00:00
* Returns a compressed texture format GLenum name based on the given format.
2021-10-05 17:30:17 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getCompressedTextureName
* @since 3.60.0
*
* @param {string} baseFormat - The Base Format to check.
* @param {GLenum} [format] - An optional GLenum format to check within the base format.
*
2021-10-05 17:36:50 +00:00
* @return {string} The compressed texture format name, as a string.
2021-10-05 17:30:17 +00:00
*/
getCompressedTextureName: function (baseFormat, format)
{
var supportedFormats = this.compression[baseFormat.toUpperCase()];
if (format in supportedFormats)
{
return supportedFormats[format];
}
},
2021-10-05 16:04:44 +00:00
/**
* Checks if the given compressed texture format is supported, or not.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#supportsCompressedTexture
* @since 3.60.0
*
* @param {string} baseFormat - The Base Format to check.
* @param {GLenum} [format] - An optional GLenum format to check within the base format.
*
* @return {boolean} True if the format is supported, otherwise false.
*/
supportsCompressedTexture: function (baseFormat, format)
{
var supportedFormats = this.compression[baseFormat.toUpperCase()];
if (supportedFormats)
{
if (format)
{
return format in supportedFormats;
}
else
{
return true;
}
}
return false;
},
2020-12-11 11:05:20 +00:00
/**
* Gets the aspect ratio of the WebGLRenderer dimensions.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getAspectRatio
* @since 3.50.0
*
* @return {number} The aspect ratio of the WebGLRenderer dimensions.
*/
getAspectRatio: function ()
{
return this.width / this.height;
},
/**
* Sets the Projection Matrix of this renderer to the given dimensions.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setProjectionMatrix
* @since 3.50.0
*
* @param {number} width - The new width of the Projection Matrix.
* @param {number} height - The new height of the Projection Matrix.
*
* @return {this} This WebGLRenderer instance.
*/
setProjectionMatrix: function (width, height)
{
if (width !== this.projectionWidth || height !== this.projectionHeight)
{
this.projectionWidth = width;
this.projectionHeight = height;
this.projectionMatrix.ortho(0, width, height, 0, -1000, 1000);
}
return this;
},
/**
* Resets the Projection Matrix back to this renderers width and height.
*
* This is called during `endCapture`, should the matrix have been changed
* as a result of the capture process.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#resetProjectionMatrix
* @since 3.50.0
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
*
* @return {this} This WebGLRenderer instance.
*/
resetProjectionMatrix: function ()
{
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
return this.setProjectionMatrix(this.width, this.height);
},
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 ()
{
this.pipelines.flush();
},
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
*
2020-11-23 10:22:13 +00:00
* @param {number} x - The x position of the scissor.
* @param {number} y - The y position of the scissor.
* @param {number} width - The width of the scissor.
* @param {number} height - The height of the scissor.
* @param {number} [drawingBufferHeight] - Optional drawingBufferHeight override value.
2018-02-09 19:19:21 +00:00
*
2020-11-23 10:23:10 +00:00
* @return {number[]} 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
*
2020-11-23 10:22:13 +00:00
* @param {number} x - The x position of the scissor.
* @param {number} y - The y position of the scissor.
* @param {number} width - The width of the scissor.
* @param {number} height - The height of the scissor.
* @param {number} [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);
}
},
/**
* Resets the gl scissor state to be whatever the current scissor is, if there is one, without
* modifying the scissor stack.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#resetScissor
* @since 3.50.0
*/
resetScissor: function ()
{
var gl = this.gl;
gl.enable(gl.SCISSOR_TEST);
var current = this.currentScissor;
if (current)
{
var x = current[0];
var y = current[1];
var width = current[2];
var height = current[3];
if (width > 0 && height > 0)
{
gl.scissor(x, (this.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;
},
2019-04-26 18:13:24 +00:00
/**
* Is there an active stencil mask?
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#hasActiveStencilMask
* @since 3.17.0
*
2019-04-26 18:13:24 +00:00
* @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));
},
/**
* Resets the gl viewport to the current renderer dimensions.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#resetViewport
* @since 3.50.0
*/
resetViewport: function ()
{
var gl = this.gl;
gl.viewport(0, 0, this.width, this.height);
this.drawingBufferHeight = gl.drawingBufferHeight;
},
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
*
2020-11-23 10:22:13 +00:00
* @param {number} 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()}.
*
2020-11-23 10:22:13 +00:00
* @return {number} 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
*
2020-11-23 10:22:13 +00:00
* @param {number} index - The index of the custom blend mode.
2018-09-28 11:19:21 +00:00
* @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
*
2020-11-23 10:22:13 +00:00
* @param {number} 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-02-09 19:19:21 +00:00
/**
* Pushes a new framebuffer onto the FBO stack and makes it the currently bound framebuffer.
*
* If there was another framebuffer already bound it will force a pipeline flush.
*
* Call `popFramebuffer` to remove it again.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#pushFramebuffer
* @since 3.50.0
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper} framebuffer - The framebuffer that needs to be bound.
* @param {boolean} [updateScissor=false] - Set the gl scissor to match the frame buffer size? Or, if `null` given, pop the scissor from the stack.
2020-11-19 16:40:07 +00:00
* @param {boolean} [setViewport=true] - Should the WebGL viewport be set?
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} [texture=null] - Bind the given frame buffer texture?
* @param {boolean} [clear=false] - Clear the frame buffer after binding?
*
* @return {this} This WebGLRenderer instance.
*/
pushFramebuffer: function (framebuffer, updateScissor, setViewport, texture, clear)
{
if (framebuffer === this.currentFramebuffer)
{
return this;
}
this.fboStack.push(framebuffer);
return this.setFramebuffer(framebuffer, updateScissor, setViewport, texture, clear);
},
/**
* Sets the given framebuffer as the active and currently bound framebuffer.
*
* If there was another framebuffer already bound it will force a pipeline flush.
*
* Typically, you should call `pushFramebuffer` instead of this method.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setFramebuffer
* @since 3.0.0
*
* @param {(Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper|null)} framebuffer - The framebuffer that needs to be bound, or `null` to bind back to the default framebuffer.
* @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.
* @param {boolean} [setViewport=true] - Should the WebGL viewport be set?
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} [texture=null] - Bind the given frame buffer texture?
* @param {boolean} [clear=false] - Clear the frame buffer after binding?
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, setViewport, texture, clear)
2017-01-23 21:42:47 +00:00
{
if (updateScissor === undefined) { updateScissor = false; }
if (setViewport === undefined) { setViewport = true; }
if (texture === undefined) { texture = null; }
if (clear === undefined) { clear = false; }
2020-10-27 12:25:53 +00:00
if (framebuffer === this.currentFramebuffer)
{
return this;
}
var gl = this.gl;
var width = this.width;
var height = this.height;
2020-11-19 16:40:07 +00:00
if (framebuffer && framebuffer.renderTexture && setViewport)
{
2020-10-27 12:25:53 +00:00
width = framebuffer.renderTexture.width;
height = framebuffer.renderTexture.height;
}
else
{
this.flush();
}
2018-01-23 19:29:47 +00:00
if (framebuffer)
{
gl.bindFramebuffer(gl.FRAMEBUFFER, framebuffer.webGLFramebuffer);
}
else
{
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
}
if (setViewport)
{
gl.viewport(0, 0, width, height);
}
if (texture)
{
gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture.webGLTexture, 0);
}
if (clear)
{
gl.clearColor(0, 0, 0, 0);
gl.clear(gl.COLOR_BUFFER_BIT);
}
2020-10-27 12:25:53 +00:00
if (updateScissor)
{
if (framebuffer)
{
2020-10-27 12:25:53 +00:00
this.drawingBufferHeight = height;
this.pushScissor(0, 0, width, height);
}
2020-10-27 12:25:53 +00:00
else
{
this.drawingBufferHeight = this.height;
2020-10-27 12:25:53 +00:00
this.popScissor();
}
}
2020-10-27 12:25:53 +00:00
this.currentFramebuffer = framebuffer;
2018-01-17 21:25:43 +00:00
return this;
},
/**
* Pops the previous framebuffer from the fbo stack and sets it.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#popFramebuffer
* @since 3.50.0
2020-11-13 14:19:55 +00:00
*
* @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.
* @param {boolean} [setViewport=true] - Should the WebGL viewport be set?
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper} The Framebuffer that was set, or `null` if there aren't any more in the stack.
*/
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
popFramebuffer: function (updateScissor, setViewport)
{
if (updateScissor === undefined) { updateScissor = false; }
if (setViewport === undefined) { setViewport = true; }
var fboStack = this.fboStack;
// Remove the current fbo
fboStack.pop();
// Reset the previous framebuffer
var framebuffer = fboStack[fboStack.length - 1];
if (!framebuffer)
{
framebuffer = null;
}
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
this.setFramebuffer(framebuffer, updateScissor, setViewport);
2020-11-13 14:19:55 +00:00
return framebuffer;
},
/**
* Restores the previous framebuffer from the fbo stack and sets it.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#restoreFramebuffer
* @since 3.60.0
*
* @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.
* @param {boolean} [setViewport=true] - Should the WebGL viewport be set?
*/
restoreFramebuffer: function (updateScissor, setViewport)
{
if (updateScissor === undefined) { updateScissor = false; }
if (setViewport === undefined) { setViewport = true; }
var fboStack = this.fboStack;
var framebuffer = fboStack[fboStack.length - 1];
if (!framebuffer)
{
framebuffer = null;
}
2023-03-16 18:23:30 +00:00
this.currentFramebuffer = null;
this.setFramebuffer(framebuffer, updateScissor, setViewport);
},
2018-02-09 19:19:21 +00:00
/**
* Binds a shader program.
*
* If there was a different program already bound it will force a pipeline flush first.
*
* If the same program given to this method is already set as the current program, no change
* will take place and this method will return `false`.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#setProgram
* @since 3.0.0
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper} program - The program that needs to be bound.
2018-02-09 19:19:21 +00:00
*
* @return {boolean} `true` if the given program was bound, otherwise `false`.
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
setProgram: function (program)
{
if (program !== this.currentProgram)
{
2018-01-23 19:29:47 +00:00
this.flush();
this.gl.useProgram(program.webGLProgram);
2018-01-25 05:26:14 +00:00
this.currentProgram = program;
return true;
2018-01-22 21:21:47 +00:00
}
return false;
},
/**
* Rebinds whatever program `WebGLRenderer.currentProgram` is set as, without
* changing anything, or flushing.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#resetProgram
* @since 3.50.0
*
* @return {this} This WebGLRenderer instance.
*/
resetProgram: function ()
{
2024-01-29 22:23:27 +00:00
this.gl.useProgram(this.currentProgram.webGLProgramWrapper);
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
* 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.
2020-11-23 10:22:13 +00:00
* @param {number} width - The width of the texture.
* @param {number} height - The height of the texture.
* @param {number} scaleMode - The scale mode to be used by the texture.
* @param {boolean} [forceClamp=false] - Force the texture to use the CLAMP_TO_EDGE wrap mode, even if a power of two?
2018-02-09 19:19:21 +00:00
*
* @return {?Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The WebGLTextureWrapper that was created, or `null` if it couldn't be created.
2018-02-09 19:19:21 +00:00
*/
createTextureFromSource: function (source, width, height, scaleMode, forceClamp)
2018-02-16 18:17:51 +00:00
{
if (forceClamp === undefined) { forceClamp = false; }
2018-01-20 04:05:56 +00:00
var gl = this.gl;
var minFilter = gl.NEAREST;
var magFilter = gl.NEAREST;
2018-01-20 04:05:56 +00:00
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;
var pow = IsSizePowerOfTwo(width, height);
if (pow && !forceClamp)
2018-01-20 04:05:56 +00:00
{
wrap = gl.REPEAT;
}
if (scaleMode === CONST.ScaleModes.LINEAR && this.config.antialias)
2018-01-20 04:05:56 +00:00
{
var isCompressed = source && source.compressed;
var isMip = (!isCompressed && pow) || (isCompressed && source.mipmaps.length > 1);
// Filters above LINEAR only work with MIPmaps.
// These are only generated for power of two (POT) textures.
// Compressed textures with mipmaps are always POT,
// but POT compressed textures might not have mipmaps.
minFilter = (this.mipmapFilter && isMip) ? this.mipmapFilter : gl.LINEAR;
magFilter = gl.LINEAR;
}
2018-01-24 00:40:20 +00:00
if (!source && typeof width === 'number' && typeof height === 'number')
{
texture = this.createTexture2D(0, minFilter, magFilter, wrap, wrap, gl.RGBA, null, width, height);
2018-01-24 00:40:20 +00:00
}
else
{
texture = this.createTexture2D(0, minFilter, magFilter, 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
/**
* A wrapper for creating a WebGLTextureWrapper. 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
*
2020-11-23 10:22:13 +00:00
* @param {number} mipLevel - Mip level of the texture.
* @param {number} minFilter - Filtering of the texture.
* @param {number} magFilter - Filtering of the texture.
* @param {number} wrapT - Wrapping mode of the texture.
* @param {number} wrapS - Wrapping mode of the texture.
* @param {number} format - Which format does the texture use.
2019-10-04 10:13:05 +00:00
* @param {?object} pixels - pixel data.
* @param {?number} width - Width of the texture in pixels. If not supplied, it must be derived from `pixels`.
* @param {?number} height - Height of the texture in pixels. If not supplied, it must be derived from `pixels`.
2019-10-04 10:13:05 +00:00
* @param {boolean} [pma=true] - Does the texture have premultiplied alpha?
* @param {boolean} [forceSize=false] - If `true` it will use the width and height passed to this method, regardless of the pixels dimension.
* @param {boolean} [flipY=false] - Sets the `UNPACK_FLIP_Y_WEBGL` flag the WebGL Texture uses during upload.
2018-04-25 23:30:41 +00:00
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The WebGLTextureWrapper that was created.
2018-02-09 19:19:21 +00:00
*/
createTexture2D: function (mipLevel, minFilter, magFilter, wrapT, wrapS, format, pixels, width, height, pma, forceSize, flipY)
2018-01-17 21:25:43 +00:00
{
if (typeof width !== 'number') { width = pixels ? pixels.width : 1; }
if (typeof height !== 'number') { height = pixels ? pixels.height : 1; }
var texture = new WebGLTextureWrapper(this.gl, mipLevel, minFilter, magFilter, wrapT, wrapS, format, pixels, width, height, pma, forceSize, flipY);
2021-11-08 18:49:48 +00:00
this.glTextureWrappers.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
/**
2020-11-06 11:42:25 +00:00
* Creates a WebGL Framebuffer object and optionally binds a depth stencil render buffer.
2018-02-09 19:19:21 +00:00
*
* This will unbind any currently bound framebuffer.
*
2018-02-09 19:19:21 +00:00
* @method Phaser.Renderer.WebGL.WebGLRenderer#createFramebuffer
* @since 3.0.0
*
2020-11-23 10:22:13 +00:00
* @param {number} width - If `addDepthStencilBuffer` is true, this controls the width of the depth stencil.
* @param {number} height - If `addDepthStencilBuffer` is true, this controls the height of the depth stencil.
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} renderTexture - The color texture where the color pixels are written.
2020-11-06 11:42:25 +00:00
* @param {boolean} [addDepthStencilBuffer=false] - Create a Renderbuffer for the depth stencil?
2018-02-09 19:19:21 +00:00
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper} Wrapped framebuffer which is safe to use with the renderer.
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
{
this.currentFramebuffer = null;
var framebuffer = new WebGLFramebufferWrapper(this.gl, width, height, renderTexture, addDepthStencilBuffer);
this.glFramebufferWrappers.push(framebuffer);
2018-01-20 00:37:52 +00:00
return framebuffer;
2018-01-17 21:25:43 +00:00
},
/**
* Binds necessary resources and renders the mask to a separated framebuffer.
* The framebuffer for the masked object is also bound for further use.
*
2022-10-03 21:07:07 +00:00
* @method Phaser.Renderer.WebGL.WebGLRenderer#beginBitmapMask
* @since 3.60.0
*
* @param {Phaser.Display.Masks.BitmapMask} mask - The BitmapMask instance that called beginMask.
* @param {Phaser.Cameras.Scene2D.Camera} camera - The camera rendering the current mask.
*/
2022-10-03 21:07:07 +00:00
beginBitmapMask: function (bitmapMask, camera)
{
var gl = this.gl;
if (gl)
{
this.flush();
this.maskTarget.bind(true);
if (this.currentCameraMask.mask !== bitmapMask)
{
this.currentMask.mask = bitmapMask;
this.currentMask.camera = camera;
}
}
},
/**
* Binds necessary resources and renders the mask to a separated framebuffer.
* The framebuffer for the masked object is also bound for further use.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#drawBitmapMask
* @since 3.60.0
*
* @param {Phaser.Display.Masks.BitmapMask} mask - The BitmapMask instance that called beginMask.
* @param {Phaser.Cameras.Scene2D.Camera} camera - The camera rendering the current mask.
* @param {Phaser.Renderer.WebGL.Pipelines.BitmapMaskPipeline} bitmapMaskPipeline - The BitmapMask Pipeline instance that is requesting the draw.
*/
drawBitmapMask: function (bitmapMask, camera, bitmapMaskPipeline)
{
// mask.mainFramebuffer should now contain all the Game Objects we want masked
this.flush();
this.maskSource.bind();
this.setBlendMode(0, true);
bitmapMask.renderWebGL(this, bitmapMask, camera);
this.maskSource.unbind(true);
this.maskTarget.unbind();
// Is there a stencil further up the stack?
var gl = this.gl;
var prev = this.getCurrentStencilMask();
if (prev)
{
gl.enable(gl.STENCIL_TEST);
prev.mask.applyStencil(this, prev.camera, true);
}
else
{
this.currentMask.mask = null;
}
// Bind this pipeline and draw
this.pipelines.set(bitmapMaskPipeline);
gl.activeTexture(gl.TEXTURE0);
gl.bindTexture(gl.TEXTURE_2D, this.maskTarget.texture.webGLTexture);
gl.activeTexture(gl.TEXTURE1);
gl.bindTexture(gl.TEXTURE_2D, this.maskSource.texture.webGLTexture);
},
2018-02-09 19:19:21 +00:00
/**
2020-10-26 15:00:25 +00:00
* Creates a WebGLProgram instance based on the given vertex and fragment shader source.
*
* Then compiles, attaches and links the program before wrapping and returning it.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createProgram
* @since 3.0.0
*
2020-10-26 15:00:25 +00:00
* @param {string} vertexShader - The vertex shader source code as a single string.
* @param {string} fragmentShader - The fragment shader source code as a single string.
2018-02-09 19:19:21 +00:00
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper} The wrapped, linked WebGLProgram created from the given shader source.
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
{
var wrapper = new WebGLProgramWrapper(this.gl, vertexShader, fragmentShader);
this.glProgramWrappers.push(wrapper);
return wrapper;
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
2020-11-23 10:22:13 +00:00
* @param {number} bufferUsage - How the buffer is used. gl.DYNAMIC_DRAW, gl.STATIC_DRAW or gl.STREAM_DRAW
2018-02-09 19:19:21 +00:00
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLBufferWrapper} Wrapped 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 = new WebGLBufferWrapper(gl, initialDataOrSize, gl.ARRAY_BUFFER, bufferUsage);
this.glBufferWrappers.push(vertexBuffer);
2018-01-20 00:37:52 +00:00
return vertexBuffer;
},
/**
* Creates a WebGLAttribLocationWrapper instance based on the given WebGLProgramWrapper and attribute name.
2024-02-19 15:28:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createAttribLocation
* @since 3.80.0
2024-02-19 15:28:21 +00:00
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper} program - The WebGLProgramWrapper instance.
* @param {string} name - The name of the attribute.
*/
createAttribLocation: function (program, name)
{
var attrib = new WebGLAttribLocationWrapper(this.gl, program, name);
this.glAttribLocationWrappers.push(attrib);
return attrib;
},
/**
* Creates a WebGLUniformLocationWrapper instance based on the given WebGLProgramWrapper and uniform name.
2024-02-19 15:28:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createUniformLocation
* @since 3.80.0
2024-02-19 15:28:21 +00:00
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper} program - The WebGLProgramWrapper instance.
* @param {string} name - The name of the uniform.
*/
createUniformLocation: function (program, name)
{
var uniform = new WebGLUniformLocationWrapper(this.gl, program, name);
this.glUniformLocationWrappers.push(uniform);
return uniform;
},
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.
2020-11-23 10:22:13 +00:00
* @param {number} bufferUsage - How the buffer is used. gl.DYNAMIC_DRAW, gl.STATIC_DRAW or gl.STREAM_DRAW.
2018-02-09 19:19:21 +00:00
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLBufferWrapper} Wrapped 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 = new WebGLBufferWrapper(gl, initialDataOrSize, gl.ELEMENT_ARRAY_BUFFER, bufferUsage);
this.glBufferWrappers.push(indexBuffer);
2018-01-20 00:37:52 +00:00
return indexBuffer;
2018-01-20 04:05:56 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Removes a texture from the GPU.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteTexture
* @since 3.0.0
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} 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
*/
Due to all of the changes with how WebGL texture batching works a lot of mostly internal methods and properties have been removed. This is the complete list * The `WebGLRenderer.currentActiveTexture` property has been removed. * The `WebGLRenderer.startActiveTexture` property has been removed. * The `WebGLRenderer.tempTextures` property has been removed. * The `WebGLRenderer.textureZero` property has been removed. * The `WebGLRenderer.normalTexture` property has been removed. * The `WebGLRenderer.textueFlush` property has been removed. * The `WebGLRenderer.isTextureClean` property has been removed. * The `WebGLRenderer.setBlankTexture` method has been removed. * The `WebGLRenderer.setTextureSource` method has been removed. * The `WebGLRenderer.isNewNormalMap` method has been removed. * The `WebGLRenderer.setTextureZero` method has been removed. * The `WebGLRenderer.clearTextureZero` method has been removed. * The `WebGLRenderer.setNormalMap` method has been removed. * The `WebGLRenderer.clearNormalMap` method has been removed. * The `WebGLRenderer.unbindTextures` method has been removed. * The `WebGLRenderer.resetTextures` method has been removed. * The `WebGLRenderer.setTexture2D` method has been removed. * The `WebGLRenderer.pushFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.setFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.popFramebuffer` method has had the `resetTextures` argument removed. * The `WebGLRenderer.deleteTexture` method has had the `reset` argument removed.
2022-10-07 15:08:39 +00:00
deleteTexture: function (texture)
2018-01-25 00:15:51 +00:00
{
if (!texture)
{
return;
}
ArrayRemove(this.glTextureWrappers, texture);
texture.destroy();
2018-01-25 00:15:51 +00:00
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Deletes a Framebuffer from the GL instance.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteFramebuffer
* @since 3.0.0
*
* @param {(Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper|null)} 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
{
if (!framebuffer)
{
return this;
}
ArrayRemove(this.fboStack, framebuffer);
ArrayRemove(this.glFramebufferWrappers, framebuffer);
framebuffer.destroy();
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
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLProgramWrapper} 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
{
if (program)
{
ArrayRemove(this.glProgramWrappers, program);
program.destroy();
}
2018-01-25 00:15:51 +00:00
return this;
},
/**
* Deletes a WebGLAttribLocation from the GL instance.
2024-02-19 15:28:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteAttribLocation
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLAttribLocationWrapper} attrib - The attrib location to be deleted.
* @since 3.80.0
*/
deleteAttribLocation: function (attrib)
{
if (attrib)
{
ArrayRemove(this.glAttribLocationWrappers, attrib);
attrib.destroy();
}
return this;
},
/**
* Deletes a WebGLUniformLocation from the GL instance.
2024-02-19 15:28:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#deleteUniformLocation
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLUniformLocationWrapper} uniform - The uniform location to be deleted.
* @since 3.80.0
*/
deleteUniformLocation: function (uniform)
{
if (uniform)
{
ArrayRemove(this.glUniformLocationWrappers, uniform);
uniform.destroy();
}
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
*
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLBufferWrapper} 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
{
if (!buffer) { return this; }
ArrayRemove(this.glBufferWrappers, buffer);
buffer.destroy();
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)
{
var cx = camera.x;
var cy = camera.y;
var cw = camera.width;
var ch = camera.height;
var color = camera.backgroundColor;
camera.emit(CameraEvents.PRE_RENDER, camera);
2020-11-13 14:19:55 +00:00
this.pipelines.preBatchCamera(camera);
this.pushScissor(cx, cy, cw, ch);
if (camera.mask)
{
this.currentCameraMask.mask = camera.mask;
this.currentCameraMask.camera = camera._maskCamera;
camera.mask.preRenderWebGL(this, camera, camera._maskCamera);
}
if (color.alphaGL > 0)
{
2020-11-16 15:45:45 +00:00
var pipeline = this.pipelines.setMulti();
pipeline.drawFillRect(
cx, cy, cw, ch,
2020-11-16 15:45:45 +00:00
Utils.getTintFromFloats(color.blueGL, color.greenGL, color.redGL, 1),
color.alphaGL
);
}
2018-01-24 03:57:33 +00:00
},
2020-07-16 10:03:49 +00:00
/**
* Return the current stencil mask.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#getCurrentStencilMask
* @private
2020-07-31 12:41:29 +00:00
* @since 3.50.0
2020-07-16 10:03:49 +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, then 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 flashEffect = camera.flashEffect;
var fadeEffect = camera.fadeEffect;
if (flashEffect.isRunning || (fadeEffect.isRunning || fadeEffect.isComplete))
{
2020-11-16 15:45:45 +00:00
var pipeline = this.pipelines.setMulti();
2018-01-24 03:57:33 +00:00
flashEffect.postRenderWebGL(pipeline, Utils.getTintFromFloats);
fadeEffect.postRenderWebGL(pipeline, 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.mask)
{
this.currentCameraMask.mask = null;
camera.mask.postRenderWebGL(this, camera._maskCamera);
}
2019-04-12 16:47:28 +00:00
this.pipelines.postBatchCamera(camera);
2019-04-26 18:13:24 +00:00
camera.emit(CameraEvents.POST_RENDER, camera);
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
* @fires Phaser.Renderer.Events#PRE_RENDER
2018-02-09 19:19:21 +00:00
* @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;
// 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);
this.currentScissor = this.defaultScissor;
this.scissorStack.length = 0;
this.scissorStack.push(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.emit(Events.PRE_RENDER);
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.
*
2020-10-16 13:11:21 +00:00
* Iterates through the given array of Game Objects 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.
*
2018-11-16 10:56:43 +00:00
* 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
* @fires Phaser.Renderer.Events#RENDER
2018-02-09 19:19:21 +00:00
* @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 - An array of filtered Game Objects that can be rendered by the given Camera.
2018-09-28 11:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - The Scene Camera to render with.
2018-02-09 19:19:21 +00:00
*/
render: function (scene, children, camera)
2018-01-20 04:05:56 +00:00
{
2018-02-16 18:17:51 +00:00
if (this.contextLost) { return; }
2018-01-20 04:05:56 +00:00
var childCount = children.length;
2018-01-29 21:46:48 +00:00
this.emit(Events.RENDER, 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
// Nothing to render, so bail out
if (childCount === 0)
{
this.setBlendMode(CONST.BlendModes.NORMAL);
// Applies camera effects and pops the scissor, if set
this.postRenderCamera(camera);
return;
}
// Reset the current type
this.currentType = '';
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
{
this.finalType = (i === childCount - 1);
var child = children[i];
2018-01-22 21:21:47 +00:00
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);
}
if (child.blendMode !== this.currentBlendMode)
{
this.setBlendMode(child.blendMode);
}
var type = child.type;
if (type !== this.currentType)
{
this.newType = true;
this.currentType = type;
}
if (!this.finalType)
{
this.nextTypeMatch = (children[i + 1].type === this.currentType);
}
else
{
this.nextTypeMatch = false;
}
child.renderWebGL(this, child, camera);
this.newType = false;
}
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
* @fires Phaser.Renderer.Events#POST_RENDER
2018-02-09 19:19:21 +00:00
* @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
2020-11-18 10:26:44 +00:00
this.flush();
this.emit(Events.POST_RENDER);
2018-01-22 22:51:15 +00:00
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.gl, state);
2019-01-25 10:01:53 +00:00
state.callback = null;
2018-01-20 04:05:56 +00:00
}
},
/**
2023-03-16 21:56:55 +00:00
* Disables the STENCIL_TEST but does not change the status
* of the current stencil mask.
*
2023-03-16 21:56:55 +00:00
* @method Phaser.Renderer.WebGL.WebGLRenderer#clearStencilMask
* @since 3.60.0
*/
2023-03-16 21:56:55 +00:00
clearStencilMask: function ()
{
2023-03-16 21:56:55 +00:00
this.gl.disable(this.gl.STENCIL_TEST);
},
/**
2023-03-16 21:56:55 +00:00
* Restores the current stencil function to the one that was in place
* before `clearStencilMask` was called.
*
2023-03-16 21:56:55 +00:00
* @method Phaser.Renderer.WebGL.WebGLRenderer#restoreStencilMask
* @since 3.60.0
*/
restoreStencilMask: function ()
{
var gl = this.gl;
var current = this.getCurrentStencilMask();
if (current)
{
var mask = current.mask;
2023-03-16 18:23:30 +00:00
gl.enable(gl.STENCIL_TEST);
2023-03-16 21:56:55 +00:00
// colorMask + stencilOp(KEEP)
2023-03-16 18:23:30 +00:00
if (mask.invertAlpha)
{
gl.stencilFunc(gl.NOTEQUAL, mask.level, 0xff);
}
else
{
gl.stencilFunc(gl.EQUAL, mask.level, 0xff);
}
}
},
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
*
2022-10-31 22:46:44 +00:00
* @param {number} x - The x coordinate to grab from. This is based on the game viewport, not the world.
* @param {number} y - The y coordinate to grab from. This is based on the game viewport, not the world.
2020-11-23 10:22:13 +00:00
* @param {number} width - The width of the area to grab.
* @param {number} 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 = width;
state.height = height;
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
*
2022-10-31 22:46:44 +00:00
* @param {number} x - The x coordinate of the pixel to get. This is based on the game viewport, not the world.
* @param {number} y - The y coordinate of the pixel to get. This is based on the game viewport, not the world.
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 {Phaser.Renderer.WebGL.Wrappers.WebGLFramebufferWrapper} framebuffer - The framebuffer to grab from.
2020-11-23 10:22:13 +00:00
* @param {number} bufferWidth - The width of the framebuffer.
* @param {number} 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?
2022-10-31 22:46:44 +00:00
* @param {number} [x=0] - The x coordinate to grab from. This is based on the framebuffer, not the world.
* @param {number} [y=0] - The y coordinate to grab from. This is based on the framebuffer, not the world.
2020-11-23 10:22:13 +00:00
* @param {number} [width=bufferWidth] - The width of the area to grab.
* @param {number} [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; }
if (type === 'pixel')
{
getPixel = true;
type = 'image/png';
}
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;
// Ensure they're not trying to grab an area larger than the framebuffer
state.width = Math.min(state.width, bufferWidth);
state.height = Math.min(state.height, bufferHeight);
this.setFramebuffer(framebuffer);
WebGLSnapshot(this.gl, state);
this.setFramebuffer(currentFramebuffer);
state.callback = null;
state.isFramebuffer = false;
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Creates a new WebGL Texture based on the given Canvas Element.
*
* If the `dstTexture` parameter is given, the WebGL Texture is updated, rather than created fresh.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#canvasToTexture
* @since 3.0.0
*
* @param {HTMLCanvasElement} srcCanvas - The Canvas to create the WebGL Texture from
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} [dstTexture] - The destination WebGLTextureWrapper to set.
* @param {boolean} [noRepeat=false] - Should this canvas be allowed to set `REPEAT` (such as for Text objects?)
* @param {boolean} [flipY=false] - Should the WebGL Texture set `UNPACK_MULTIPLY_FLIP_Y`?
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The newly created, or updated, WebGLTextureWrapper.
*/
canvasToTexture: function (srcCanvas, dstTexture, noRepeat, flipY)
{
if (noRepeat === undefined) { noRepeat = false; }
if (flipY === undefined) { flipY = false; }
if (!dstTexture)
{
return this.createCanvasTexture(srcCanvas, noRepeat, flipY);
}
else
{
return this.updateCanvasTexture(srcCanvas, dstTexture, flipY, noRepeat);
}
},
/**
* Creates a new WebGL Texture based on the given Canvas Element.
2018-02-09 19:19:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createCanvasTexture
* @since 3.20.0
*
2022-10-31 22:46:44 +00:00
* @param {HTMLCanvasElement} srcCanvas - The Canvas to create the WebGL Texture from.
* @param {boolean} [noRepeat=false] - Should this canvas be allowed to set `REPEAT` (such as for Text objects?)
* @param {boolean} [flipY=false] - Should the WebGL Texture set `UNPACK_MULTIPLY_FLIP_Y`?
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The newly created WebGLTextureWrapper.
2018-02-09 19:19:21 +00:00
*/
createCanvasTexture: function (srcCanvas, noRepeat, flipY)
2018-01-20 04:05:56 +00:00
{
if (noRepeat === undefined) { noRepeat = false; }
if (flipY === undefined) { flipY = false; }
2018-01-20 04:05:56 +00:00
var gl = this.gl;
var minFilter = gl.NEAREST;
var magFilter = gl.NEAREST;
2018-01-20 04:05:56 +00:00
var width = srcCanvas.width;
var height = srcCanvas.height;
var wrapping = gl.CLAMP_TO_EDGE;
var pow = IsSizePowerOfTwo(width, height);
if (!noRepeat && pow)
{
wrapping = gl.REPEAT;
}
2018-01-20 04:05:56 +00:00
if (this.config.antialias)
{
minFilter = (pow && this.mipmapFilter) ? this.mipmapFilter : gl.LINEAR;
magFilter = gl.LINEAR;
}
return this.createTexture2D(0, minFilter, magFilter, wrapping, wrapping, gl.RGBA, srcCanvas, width, height, true, false, flipY);
},
/**
* Updates a WebGL Texture based on the given Canvas Element.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#updateCanvasTexture
* @since 3.20.0
*
* @param {HTMLCanvasElement} srcCanvas - The Canvas to update the WebGL Texture from.
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} dstTexture - The destination WebGLTextureWrapper to update.
* @param {boolean} [flipY=false] - Should the WebGL Texture set `UNPACK_MULTIPLY_FLIP_Y`?
* @param {boolean} [noRepeat=false] - Should this canvas be allowed to set `REPEAT` (such as for Text objects?)
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The updated WebGLTextureWrapper. This is the same wrapper object as `dstTexture`.
*/
updateCanvasTexture: function (srcCanvas, dstTexture, flipY, noRepeat)
{
if (flipY === undefined) { flipY = false; }
if (noRepeat === undefined) { noRepeat = false; }
var gl = this.gl;
var minFilter = gl.NEAREST;
var magFilter = gl.NEAREST;
var width = srcCanvas.width;
var height = srcCanvas.height;
var wrapping = gl.CLAMP_TO_EDGE;
var pow = IsSizePowerOfTwo(width, height);
if (!noRepeat && pow)
{
wrapping = gl.REPEAT;
}
if (this.config.antialias)
{
minFilter = (pow && this.mipmapFilter) ? this.mipmapFilter : gl.LINEAR;
magFilter = gl.LINEAR;
}
dstTexture.update(srcCanvas, width, height, flipY, wrapping, wrapping, minFilter, magFilter);
return dstTexture;
2018-01-20 04:05:56 +00:00
},
/**
* Creates a new WebGL Texture based on the given HTML Video Element.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createVideoTexture
2019-10-04 10:13:05 +00:00
* @since 3.20.0
*
* @param {HTMLVideoElement} srcVideo - The Video to create the WebGL Texture from
* @param {boolean} [noRepeat=false] - Should this canvas be allowed to set `REPEAT`?
* @param {boolean} [flipY=false] - Should the WebGL Texture set `UNPACK_MULTIPLY_FLIP_Y`?
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The newly created WebGLTextureWrapper.
*/
createVideoTexture: function (srcVideo, noRepeat, flipY)
{
if (noRepeat === undefined) { noRepeat = false; }
if (flipY === undefined) { flipY = false; }
var gl = this.gl;
var minFilter = gl.NEAREST;
var magFilter = gl.NEAREST;
var width = srcVideo.videoWidth;
var height = srcVideo.videoHeight;
var wrapping = gl.CLAMP_TO_EDGE;
var pow = IsSizePowerOfTwo(width, height);
if (!noRepeat && pow)
{
wrapping = gl.REPEAT;
}
if (this.config.antialias)
{
minFilter = (pow && this.mipmapFilter) ? this.mipmapFilter : gl.LINEAR;
magFilter = gl.LINEAR;
}
return this.createTexture2D(0, minFilter, magFilter, wrapping, wrapping, gl.RGBA, srcVideo, width, height, true, true, flipY);
},
/**
* Updates a WebGL Texture based on the given HTML Video Element.
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#updateVideoTexture
* @since 3.20.0
*
* @param {HTMLVideoElement} srcVideo - The Video to update the WebGL Texture with.
* @param {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} dstTexture - The destination WebGLTextureWrapper to update.
* @param {boolean} [flipY=false] - Should the WebGL Texture set `UNPACK_MULTIPLY_FLIP_Y`?
* @param {boolean} [noRepeat=false] - Should this canvas be allowed to set `REPEAT`?
*
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The updated WebGLTextureWrapper. This is the same wrapper object as `dstTexture`.
*/
updateVideoTexture: function (srcVideo, dstTexture, flipY, noRepeat)
{
if (flipY === undefined) { flipY = false; }
if (noRepeat === undefined) { noRepeat = false; }
var gl = this.gl;
var minFilter = gl.NEAREST;
var magFilter = gl.NEAREST;
var width = srcVideo.videoWidth;
var height = srcVideo.videoHeight;
var wrapping = gl.CLAMP_TO_EDGE;
var pow = IsSizePowerOfTwo(width, height);
if (!noRepeat && pow)
{
wrapping = gl.REPEAT;
}
if (this.config.antialias)
{
minFilter = (pow && this.mipmapFilter) ? this.mipmapFilter : gl.LINEAR;
magFilter = gl.LINEAR;
}
dstTexture.update(srcVideo, width, height, flipY, wrapping, wrapping, minFilter, magFilter);
return dstTexture;
},
/**
* Create a WebGLTexture from a Uint8Array.
2024-02-19 15:28:21 +00:00
*
* The Uint8Array is assumed to be RGBA values, one byte per color component.
2024-02-19 15:28:21 +00:00
*
* The texture will be filtered with `gl.NEAREST` and will not be mipped.
2024-02-19 15:28:21 +00:00
*
* @method Phaser.Renderer.WebGL.WebGLRenderer#createUint8ArrayTexture
* @since 3.80.0
* @param {Uint8Array} data - The Uint8Array to create the texture from.
* @param {number} width - The width of the texture.
* @param {number} height - The height of the texture.
* @return {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} The newly created WebGLTextureWrapper.
*/
createUint8ArrayTexture: function (data, width, height)
{
var gl = this.gl;
var minFilter = gl.NEAREST;
var magFilter = gl.NEAREST;
var wrap = gl.CLAMP_TO_EDGE;
var pow = IsSizePowerOfTwo(width, height);
if (pow)
{
wrap = gl.REPEAT;
}
return this.createTexture2D(0, minFilter, magFilter, wrap, wrap, gl.RGBA, data, width, height);
},
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 {Phaser.Renderer.WebGL.Wrappers.WebGLTextureWrapper} texture - The texture to set the filter for.
2020-11-23 10:22:13 +00:00
* @param {number} 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;
2023-11-07 18:28:27 +00:00
var glFilter = (filter === 0) ? gl.LINEAR : gl.NEAREST;
2018-01-20 04:05:56 +00:00
gl.activeTexture(gl.TEXTURE0);
var currentTexture = gl.getParameter(gl.TEXTURE_BINDING_2D);
gl.bindTexture(gl.TEXTURE_2D, texture.webGLTexture);
2018-01-20 04:05:56 +00:00
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, glFilter);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, glFilter);
// Update wrapper.
texture.minFilter = glFilter;
texture.magFilter = glFilter;
if (currentTexture)
{
gl.bindTexture(gl.TEXTURE_2D, currentTexture);
}
2018-01-20 04:05:56 +00:00
2018-01-22 21:21:47 +00:00
return this;
},
/**
* 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
*
2020-11-23 10:22:13 +00:00
* @return {number} 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 ()
{
this.canvas.removeEventListener('webglcontextlost', this.contextLostHandler, false);
this.canvas.removeEventListener('webglcontextrestored', this.contextRestoredHandler, false);
var wrapperDestroy = function (wrapper)
{
wrapper.destroy();
};
ArrayEach(this.glAttribLocationWrappers, wrapperDestroy);
ArrayEach(this.glBufferWrappers, wrapperDestroy);
ArrayEach(this.glFramebufferWrappers, wrapperDestroy);
ArrayEach(this.glProgramWrappers, wrapperDestroy);
ArrayEach(this.glTextureWrappers, wrapperDestroy);
ArrayEach(this.glUniformLocationWrappers, wrapperDestroy);
this.maskTarget.destroy();
this.maskSource.destroy();
this.pipelines.destroy();
2018-04-23 18:11:16 +00:00
this.removeAllListeners();
this.fboStack = [];
this.maskStack = [];
this.extensions = {};
this.textureIndexes = [];
this.gl = null;
this.game = null;
this.canvas = null;
this.contextLost = true;
this.currentMask = null;
this.currentCameraMask = null;
2023-04-12 02:45:19 +00:00
if (DEBUG)
{
this.spector = null;
}
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;