2018-02-12 16:01:20 +00:00
|
|
|
/**
|
|
|
|
* @author Richard Davey <rich@photonstorm.com>
|
2018-04-05 08:02:36 +00:00
|
|
|
* @author Felipe Alfonso <@bitnenfer>
|
2020-01-15 12:07:09 +00:00
|
|
|
* @copyright 2020 Photon Storm Ltd.
|
2019-05-10 15:15:04 +00:00
|
|
|
* @license {@link https://opensource.org/licenses/MIT|MIT License}
|
2018-02-12 16:01:20 +00:00
|
|
|
*/
|
|
|
|
|
2018-01-26 23:17:11 +00:00
|
|
|
var Class = require('../../../utils/Class');
|
2018-05-08 22:04:57 +00:00
|
|
|
var ShaderSourceFS = require('../shaders/ForwardDiffuse-frag.js');
|
2018-02-09 19:19:21 +00:00
|
|
|
var TextureTintPipeline = require('./TextureTintPipeline');
|
2020-07-16 01:15:53 +00:00
|
|
|
var WebGLPipeline = require('../WebGLPipeline');
|
2018-02-09 19:19:21 +00:00
|
|
|
|
2018-01-30 22:46:43 +00:00
|
|
|
var LIGHT_COUNT = 10;
|
2018-01-26 23:17:11 +00:00
|
|
|
|
2018-02-09 19:19:21 +00:00
|
|
|
/**
|
|
|
|
* @classdesc
|
2018-04-24 19:52:57 +00:00
|
|
|
* ForwardDiffuseLightPipeline implements a forward rendering approach for 2D lights.
|
|
|
|
* This pipeline extends TextureTintPipeline so it implements all it's rendering functions
|
|
|
|
* and batching system.
|
2018-02-09 19:19:21 +00:00
|
|
|
*
|
|
|
|
* @class ForwardDiffuseLightPipeline
|
2018-04-18 11:13:49 +00:00
|
|
|
* @extends Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline
|
2018-10-10 09:49:13 +00:00
|
|
|
* @memberof Phaser.Renderer.WebGL.Pipelines
|
2018-02-09 19:19:21 +00:00
|
|
|
* @constructor
|
|
|
|
* @since 3.0.0
|
|
|
|
*
|
2018-10-19 11:32:43 +00:00
|
|
|
* @param {object} config - The configuration of the pipeline, same as the {@link Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline}. The fragment shader will be replaced with the lighting shader.
|
2018-02-09 19:19:21 +00:00
|
|
|
*/
|
2018-01-26 23:17:11 +00:00
|
|
|
var ForwardDiffuseLightPipeline = new Class({
|
|
|
|
|
|
|
|
Extends: TextureTintPipeline,
|
|
|
|
|
|
|
|
initialize:
|
|
|
|
|
2018-03-05 14:29:48 +00:00
|
|
|
function ForwardDiffuseLightPipeline (config)
|
2018-01-26 23:17:11 +00:00
|
|
|
{
|
2018-10-02 10:09:58 +00:00
|
|
|
LIGHT_COUNT = config.maxLights;
|
|
|
|
|
2018-03-05 14:29:48 +00:00
|
|
|
config.fragShader = ShaderSourceFS.replace('%LIGHT_COUNT%', LIGHT_COUNT.toString());
|
|
|
|
|
|
|
|
TextureTintPipeline.call(this, config);
|
2018-07-13 10:14:22 +00:00
|
|
|
|
2018-10-31 00:03:34 +00:00
|
|
|
/**
|
2018-12-12 11:58:58 +00:00
|
|
|
* Inverse rotation matrix for normal map rotations.
|
|
|
|
*
|
|
|
|
* @name Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#inverseRotationMatrix
|
|
|
|
* @type {Float32Array}
|
|
|
|
* @private
|
|
|
|
* @since 3.16.0
|
2018-10-31 00:03:34 +00:00
|
|
|
*/
|
|
|
|
this.inverseRotationMatrix = new Float32Array([
|
|
|
|
1, 0, 0,
|
|
|
|
0, 1, 0,
|
|
|
|
0, 0, 1
|
|
|
|
]);
|
2020-07-16 02:24:19 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores the previous number of lights rendered.
|
|
|
|
*
|
|
|
|
* @name Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#lightCount
|
|
|
|
* @type {number}
|
|
|
|
* @since 3.25.0
|
|
|
|
*/
|
|
|
|
this.lightCount = 0;
|
2018-07-13 10:14:22 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
2020-07-16 01:15:53 +00:00
|
|
|
* Called every time the pipeline is bound by the renderer.
|
|
|
|
* Sets the shader program, vertex buffer and other resources.
|
|
|
|
* Should only be called when changing pipeline.
|
2018-02-09 19:19:21 +00:00
|
|
|
*
|
2020-07-16 01:15:53 +00:00
|
|
|
* @method Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#bind
|
|
|
|
* @since 3.25.0
|
2018-02-09 19:19:21 +00:00
|
|
|
*
|
2018-09-25 10:36:36 +00:00
|
|
|
* @return {this} This WebGLPipeline instance.
|
2018-02-09 19:19:21 +00:00
|
|
|
*/
|
2020-07-16 01:15:53 +00:00
|
|
|
bind: function ()
|
2018-01-26 23:17:11 +00:00
|
|
|
{
|
2020-07-16 01:15:53 +00:00
|
|
|
WebGLPipeline.prototype.bind.call(this);
|
2018-01-26 23:17:11 +00:00
|
|
|
|
|
|
|
var renderer = this.renderer;
|
2018-01-30 03:38:31 +00:00
|
|
|
var program = this.program;
|
2018-01-26 23:17:11 +00:00
|
|
|
|
2020-07-16 01:15:53 +00:00
|
|
|
renderer.setInt1(program, 'uMainSampler', 0);
|
2018-01-30 03:38:31 +00:00
|
|
|
renderer.setInt1(program, 'uNormSampler', 1);
|
2018-01-26 23:17:11 +00:00
|
|
|
renderer.setFloat2(program, 'uResolution', this.width, this.height);
|
|
|
|
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2018-02-09 19:19:21 +00:00
|
|
|
/**
|
2018-04-24 19:52:57 +00:00
|
|
|
* This function sets all the needed resources for each camera pass.
|
2018-02-09 19:19:21 +00:00
|
|
|
*
|
2018-04-18 11:13:49 +00:00
|
|
|
* @method Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#onRender
|
2018-02-09 19:19:21 +00:00
|
|
|
* @since 3.0.0
|
|
|
|
*
|
2018-10-19 11:32:43 +00:00
|
|
|
* @param {Phaser.Scene} scene - The Scene being rendered.
|
|
|
|
* @param {Phaser.Cameras.Scene2D.Camera} camera - The Scene Camera being rendered with.
|
2018-02-09 19:19:21 +00:00
|
|
|
*
|
2018-09-25 10:36:36 +00:00
|
|
|
* @return {this} This WebGLPipeline instance.
|
2018-02-09 19:19:21 +00:00
|
|
|
*/
|
2018-01-29 21:46:48 +00:00
|
|
|
onRender: function (scene, camera)
|
|
|
|
{
|
2018-05-31 15:57:21 +00:00
|
|
|
this.active = false;
|
|
|
|
|
2018-03-12 12:55:09 +00:00
|
|
|
var lightManager = scene.sys.lights;
|
2018-01-31 01:11:51 +00:00
|
|
|
|
2018-05-31 15:57:21 +00:00
|
|
|
if (!lightManager || lightManager.lights.length <= 0 || !lightManager.active)
|
2018-03-12 02:22:42 +00:00
|
|
|
{
|
2018-05-31 15:57:21 +00:00
|
|
|
// Passthru
|
2018-03-12 02:22:42 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2018-05-31 15:57:21 +00:00
|
|
|
var lights = lightManager.cull(camera);
|
|
|
|
var lightCount = Math.min(lights.length, LIGHT_COUNT);
|
2018-01-31 01:11:51 +00:00
|
|
|
|
2018-05-31 15:57:21 +00:00
|
|
|
if (lightCount === 0)
|
2018-01-31 01:11:51 +00:00
|
|
|
{
|
2018-05-31 15:57:21 +00:00
|
|
|
return this;
|
2018-01-31 01:11:51 +00:00
|
|
|
}
|
|
|
|
|
2018-05-31 15:57:21 +00:00
|
|
|
this.active = true;
|
|
|
|
|
2018-01-30 03:38:31 +00:00
|
|
|
var renderer = this.renderer;
|
|
|
|
var program = this.program;
|
2018-01-30 22:46:43 +00:00
|
|
|
var cameraMatrix = camera.matrix;
|
|
|
|
var point = {x: 0, y: 0};
|
|
|
|
var height = renderer.height;
|
2020-07-16 02:24:19 +00:00
|
|
|
var i;
|
|
|
|
|
|
|
|
if (lightCount !== this.lightCount)
|
|
|
|
{
|
|
|
|
for (i = 0; i < LIGHT_COUNT; i++)
|
|
|
|
{
|
|
|
|
// Reset lights
|
|
|
|
renderer.setFloat1(program, 'uLights[' + i + '].radius', 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.lightCount = lightCount;
|
|
|
|
}
|
2018-01-30 22:46:43 +00:00
|
|
|
|
2020-07-16 02:24:19 +00:00
|
|
|
if (camera.dirty)
|
2018-01-31 01:11:51 +00:00
|
|
|
{
|
2020-07-16 02:24:19 +00:00
|
|
|
renderer.setFloat4(program, 'uCamera', camera.x, camera.y, camera.rotation, camera.zoom);
|
2018-01-31 01:11:51 +00:00
|
|
|
}
|
|
|
|
|
2020-07-16 02:24:19 +00:00
|
|
|
// TODO - Only if dirty! and cache the location
|
2018-01-30 22:46:43 +00:00
|
|
|
renderer.setFloat3(program, 'uAmbientLightColor', lightManager.ambientColor.r, lightManager.ambientColor.g, lightManager.ambientColor.b);
|
|
|
|
|
2020-07-16 02:24:19 +00:00
|
|
|
for (i = 0; i < lightCount; i++)
|
2018-01-30 22:46:43 +00:00
|
|
|
{
|
2020-07-16 02:24:19 +00:00
|
|
|
var light = lights[i];
|
|
|
|
|
|
|
|
if (light.dirty)
|
|
|
|
{
|
|
|
|
var lightName = 'uLights[' + i + '].';
|
|
|
|
|
|
|
|
cameraMatrix.transformPoint(light.x, light.y, point);
|
2018-05-31 15:57:21 +00:00
|
|
|
|
2020-07-16 02:24:19 +00:00
|
|
|
// TODO - Cache the uniform locations!!!
|
|
|
|
renderer.setFloat2(program, lightName + 'position', point.x - (camera.scrollX * light.scrollFactorX * camera.zoom), height - (point.y - (camera.scrollY * light.scrollFactorY) * camera.zoom));
|
|
|
|
renderer.setFloat3(program, lightName + 'color', light.r, light.g, light.b);
|
|
|
|
renderer.setFloat1(program, lightName + 'intensity', light.intensity);
|
|
|
|
renderer.setFloat1(program, lightName + 'radius', light.radius);
|
2018-05-31 15:57:21 +00:00
|
|
|
|
2020-07-16 02:24:19 +00:00
|
|
|
light.dirty = false;
|
|
|
|
}
|
2018-01-30 22:46:43 +00:00
|
|
|
}
|
2019-10-06 18:31:29 +00:00
|
|
|
|
|
|
|
this.currentNormalMapRotation = null;
|
|
|
|
|
2018-01-29 21:46:48 +00:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2018-07-13 10:14:22 +00:00
|
|
|
/**
|
2020-07-16 01:15:53 +00:00
|
|
|
* Rotates the normal map vectors inversely by the given angle.
|
|
|
|
* Only works in 2D space.
|
|
|
|
*
|
|
|
|
* @method Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#setNormalMapRotation
|
|
|
|
* @since 3.16.0
|
|
|
|
*
|
|
|
|
* @param {number} rotation - The angle of rotation in radians.
|
|
|
|
*/
|
|
|
|
setNormalMapRotation: function (rotation)
|
|
|
|
{
|
|
|
|
if (rotation !== this.currentNormalMapRotation || this.vertexCount === 0)
|
|
|
|
{
|
|
|
|
if (this.vertexCount > 0)
|
|
|
|
{
|
|
|
|
this.flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
var inverseRotationMatrix = this.inverseRotationMatrix;
|
|
|
|
|
|
|
|
if (rotation)
|
|
|
|
{
|
|
|
|
var rot = -rotation;
|
|
|
|
var c = Math.cos(rot);
|
|
|
|
var s = Math.sin(rot);
|
|
|
|
|
|
|
|
inverseRotationMatrix[1] = s;
|
|
|
|
inverseRotationMatrix[3] = -s;
|
|
|
|
inverseRotationMatrix[0] = inverseRotationMatrix[4] = c;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
inverseRotationMatrix[0] = inverseRotationMatrix[4] = 1;
|
|
|
|
inverseRotationMatrix[1] = inverseRotationMatrix[3] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.renderer.setMatrix3(this.program, 'uInverseRotationMatrix', false, inverseRotationMatrix);
|
|
|
|
|
|
|
|
this.currentNormalMapRotation = rotation;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generic function for batching a textured quad with a normal map.
|
2018-07-13 10:14:22 +00:00
|
|
|
*
|
2019-01-31 12:19:01 +00:00
|
|
|
* @method Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#batchTexture
|
2018-07-13 10:14:22 +00:00
|
|
|
* @since 3.0.0
|
|
|
|
*
|
|
|
|
* @param {Phaser.GameObjects.GameObject} gameObject - Source GameObject
|
|
|
|
* @param {WebGLTexture} texture - Raw WebGLTexture associated with the quad
|
|
|
|
* @param {integer} textureWidth - Real texture width
|
|
|
|
* @param {integer} textureHeight - Real texture height
|
|
|
|
* @param {number} srcX - X coordinate of the quad
|
|
|
|
* @param {number} srcY - Y coordinate of the quad
|
|
|
|
* @param {number} srcWidth - Width of the quad
|
|
|
|
* @param {number} srcHeight - Height of the quad
|
|
|
|
* @param {number} scaleX - X component of scale
|
|
|
|
* @param {number} scaleY - Y component of scale
|
|
|
|
* @param {number} rotation - Rotation of the quad
|
|
|
|
* @param {boolean} flipX - Indicates if the quad is horizontally flipped
|
|
|
|
* @param {boolean} flipY - Indicates if the quad is vertically flipped
|
|
|
|
* @param {number} scrollFactorX - By which factor is the quad affected by the camera horizontal scroll
|
|
|
|
* @param {number} scrollFactorY - By which factor is the quad effected by the camera vertical scroll
|
|
|
|
* @param {number} displayOriginX - Horizontal origin in pixels
|
|
|
|
* @param {number} displayOriginY - Vertical origin in pixels
|
|
|
|
* @param {number} frameX - X coordinate of the texture frame
|
|
|
|
* @param {number} frameY - Y coordinate of the texture frame
|
|
|
|
* @param {number} frameWidth - Width of the texture frame
|
|
|
|
* @param {number} frameHeight - Height of the texture frame
|
|
|
|
* @param {integer} tintTL - Tint for top left
|
|
|
|
* @param {integer} tintTR - Tint for top right
|
|
|
|
* @param {integer} tintBL - Tint for bottom left
|
|
|
|
* @param {integer} tintBR - Tint for bottom right
|
|
|
|
* @param {number} tintEffect - The tint effect (0 for additive, 1 for replacement)
|
|
|
|
* @param {number} uOffset - Horizontal offset on texture coordinate
|
|
|
|
* @param {number} vOffset - Vertical offset on texture coordinate
|
|
|
|
* @param {Phaser.Cameras.Scene2D.Camera} camera - Current used camera
|
|
|
|
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - Parent container
|
2020-07-16 01:15:53 +00:00
|
|
|
* @param {boolean} [skipFlip=false] - Skip the renderTexture check.
|
2018-07-13 10:14:22 +00:00
|
|
|
*/
|
|
|
|
batchTexture: function (
|
|
|
|
gameObject,
|
|
|
|
texture,
|
|
|
|
textureWidth, textureHeight,
|
|
|
|
srcX, srcY,
|
|
|
|
srcWidth, srcHeight,
|
|
|
|
scaleX, scaleY,
|
|
|
|
rotation,
|
|
|
|
flipX, flipY,
|
|
|
|
scrollFactorX, scrollFactorY,
|
|
|
|
displayOriginX, displayOriginY,
|
|
|
|
frameX, frameY, frameWidth, frameHeight,
|
|
|
|
tintTL, tintTR, tintBL, tintBR, tintEffect,
|
|
|
|
uOffset, vOffset,
|
|
|
|
camera,
|
2020-07-16 01:15:53 +00:00
|
|
|
parentTransformMatrix,
|
|
|
|
skipFlip)
|
2018-07-13 10:14:22 +00:00
|
|
|
{
|
|
|
|
var normalTexture;
|
|
|
|
|
2018-08-23 16:30:27 +00:00
|
|
|
if (gameObject.displayTexture)
|
|
|
|
{
|
|
|
|
normalTexture = gameObject.displayTexture.dataSource[gameObject.displayFrame.sourceIndex];
|
|
|
|
}
|
|
|
|
else if (gameObject.texture)
|
2018-07-13 10:14:22 +00:00
|
|
|
{
|
|
|
|
normalTexture = gameObject.texture.dataSource[gameObject.frame.sourceIndex];
|
|
|
|
}
|
|
|
|
else if (gameObject.tileset)
|
|
|
|
{
|
2019-10-25 22:56:43 +00:00
|
|
|
if (Array.isArray(gameObject.tileset))
|
|
|
|
{
|
|
|
|
normalTexture = gameObject.tileset[0].image.dataSource[0];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
normalTexture = gameObject.tileset.image.dataSource[0];
|
|
|
|
}
|
2018-07-13 10:14:22 +00:00
|
|
|
}
|
|
|
|
|
2020-07-16 01:15:53 +00:00
|
|
|
if (normalTexture)
|
2018-10-31 00:03:34 +00:00
|
|
|
{
|
2020-07-16 01:15:53 +00:00
|
|
|
TextureTintPipeline.prototype.batchTexture.call(this, gameObject, texture, textureWidth, textureHeight, srcX, srcY, srcWidth, srcHeight, scaleX, scaleY, rotation, flipX, flipY, scrollFactorX, scrollFactorY, displayOriginX, displayOriginY, frameX, frameY, frameWidth, frameHeight, tintTL, tintTR, tintBL, tintBR, tintEffect, uOffset, vOffset, camera, parentTransformMatrix, skipFlip, true);
|
2018-10-31 00:03:34 +00:00
|
|
|
|
2020-07-16 01:15:53 +00:00
|
|
|
this.renderer.setNormalMap(normalTexture.glTexture);
|
2018-10-31 00:03:34 +00:00
|
|
|
|
2020-07-16 01:15:53 +00:00
|
|
|
this.setNormalMapRotation(rotation);
|
2019-10-06 18:31:29 +00:00
|
|
|
}
|
2018-10-31 00:03:34 +00:00
|
|
|
},
|
|
|
|
|
2018-02-09 19:19:21 +00:00
|
|
|
/**
|
2018-10-19 11:32:43 +00:00
|
|
|
* Takes a Sprite Game Object, or any object that extends it, which has a normal texture and adds it to the batch.
|
2018-02-09 19:19:21 +00:00
|
|
|
*
|
2018-04-18 11:13:49 +00:00
|
|
|
* @method Phaser.Renderer.WebGL.Pipelines.ForwardDiffuseLightPipeline#batchSprite
|
2018-02-09 19:19:21 +00:00
|
|
|
* @since 3.0.0
|
|
|
|
*
|
2018-10-19 11:32:43 +00:00
|
|
|
* @param {Phaser.GameObjects.Sprite} sprite - The texture-based Game Object to add to the batch.
|
|
|
|
* @param {Phaser.Cameras.Scene2D.Camera} camera - The Camera to use for the rendering transform.
|
|
|
|
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - The transform matrix of the parent container, if set.
|
2018-07-11 15:55:18 +00:00
|
|
|
*/
|
2018-04-18 21:40:27 +00:00
|
|
|
batchSprite: function (sprite, camera, parentTransformMatrix)
|
2018-01-30 03:38:31 +00:00
|
|
|
{
|
2018-05-31 15:57:21 +00:00
|
|
|
if (!this.active)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-04 13:31:19 +00:00
|
|
|
var normalTexture = sprite.texture.dataSource[sprite.frame.sourceIndex];
|
2018-01-30 03:38:31 +00:00
|
|
|
|
|
|
|
if (normalTexture)
|
|
|
|
{
|
2020-07-16 01:15:53 +00:00
|
|
|
TextureTintPipeline.prototype.batchSprite.call(this, sprite, camera, parentTransformMatrix, true);
|
2018-05-31 15:57:21 +00:00
|
|
|
|
2020-07-16 01:15:53 +00:00
|
|
|
this.renderer.setNormalMap(normalTexture.glTexture);
|
2018-01-30 03:38:31 +00:00
|
|
|
|
2020-07-16 01:15:53 +00:00
|
|
|
this.setNormalMapRotation(sprite.rotation);
|
2018-01-30 03:38:31 +00:00
|
|
|
}
|
2018-01-26 23:17:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2018-01-31 01:11:51 +00:00
|
|
|
ForwardDiffuseLightPipeline.LIGHT_COUNT = LIGHT_COUNT;
|
|
|
|
|
2018-01-26 23:17:11 +00:00
|
|
|
module.exports = ForwardDiffuseLightPipeline;
|