phaser/src/renderer/webgl/pipelines/TextureTintPipeline.js

2251 lines
76 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
2018-04-05 08:02:36 +00:00
* @author Felipe Alfonso <@bitnenfer>
2018-02-12 16:01:20 +00:00
* @copyright 2018 Photon Storm Ltd.
* @license {@link https://github.com/photonstorm/phaser/blob/master/license.txt|MIT License}
*/
2018-01-22 21:21:47 +00:00
var Class = require('../../../utils/Class');
2018-01-25 05:26:14 +00:00
var ModelViewProjection = require('./components/ModelViewProjection');
var ShaderSourceFS = require('../shaders/TextureTint-frag.js');
var ShaderSourceVS = require('../shaders/TextureTint-vert.js');
2018-02-09 19:19:21 +00:00
var Utils = require('../Utils');
var WebGLPipeline = require('../WebGLPipeline');
2018-01-22 21:21:47 +00:00
2018-02-09 19:19:21 +00:00
/**
* @classdesc
* TextureTintPipeline implements the rendering infrastructure
* for displaying textured objects
* The config properties are:
* - game: Current game instance.
* - renderer: Current WebGL renderer.
* - topology: This indicates how the primitives are rendered. The default value is GL_TRIANGLES.
* Here is the full list of rendering primitives (https://developer.mozilla.org/en-US/docs/Web/API/WebGL_API/Constants).
* - vertShader: Source for vertex shader as a string.
* - fragShader: Source for fragment shader as a string.
* - vertexCapacity: The amount of vertices that shall be allocated
* - vertexSize: The size of a single vertex in bytes.
2018-02-09 19:19:21 +00:00
*
* @class TextureTintPipeline
* @extends Phaser.Renderer.WebGL.WebGLPipeline
2018-04-18 11:13:49 +00:00
* @memberOf Phaser.Renderer.WebGL.Pipelines
2018-02-09 19:19:21 +00:00
* @constructor
* @since 3.0.0
*
* @param {object} config - [description]
2018-02-09 19:19:21 +00:00
*/
2018-01-22 22:51:15 +00:00
var TextureTintPipeline = new Class({
2018-01-22 21:21:47 +00:00
Extends: WebGLPipeline,
2018-01-25 05:26:14 +00:00
Mixins: [
ModelViewProjection
],
2018-01-22 21:21:47 +00:00
initialize:
function TextureTintPipeline (config)
2018-01-22 21:21:47 +00:00
{
WebGLPipeline.call(this, {
game: config.game,
renderer: config.renderer,
gl: config.renderer.gl,
topology: (config.topology ? config.topology : config.renderer.gl.TRIANGLES),
vertShader: (config.vertShader ? config.vertShader : ShaderSourceVS),
fragShader: (config.fragShader ? config.fragShader : ShaderSourceFS),
vertexCapacity: (config.vertexCapacity ? config.vertexCapacity : 6 * 2000),
2018-01-22 21:21:47 +00:00
vertexSize: (config.vertexSize ? config.vertexSize :
2018-02-16 18:07:49 +00:00
Float32Array.BYTES_PER_ELEMENT * 2 +
Float32Array.BYTES_PER_ELEMENT * 2 +
Float32Array.BYTES_PER_ELEMENT * 1 +
Uint8Array.BYTES_PER_ELEMENT * 4
),
2018-01-22 21:21:47 +00:00
attributes: [
{
name: 'inPosition',
size: 2,
type: config.renderer.gl.FLOAT,
2018-01-22 21:21:47 +00:00
normalized: false,
offset: 0
},
{
name: 'inTexCoord',
size: 2,
type: config.renderer.gl.FLOAT,
2018-01-22 21:21:47 +00:00
normalized: false,
offset: Float32Array.BYTES_PER_ELEMENT * 2
},
{
name: 'inTintEffect',
size: 1,
type: config.renderer.gl.FLOAT,
normalized: false,
offset: Float32Array.BYTES_PER_ELEMENT * 4
},
2018-01-22 21:21:47 +00:00
{
name: 'inTint',
size: 4,
type: config.renderer.gl.UNSIGNED_BYTE,
2018-01-22 21:21:47 +00:00
normalized: true,
offset: Float32Array.BYTES_PER_ELEMENT * 5
2018-01-22 21:21:47 +00:00
}
]
});
2018-02-09 19:19:21 +00:00
/**
* Float32 view of the array buffer containing the pipeline's vertices.
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @name Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#vertexViewF32
2018-02-09 19:19:21 +00:00
* @type {Float32Array}
* @since 3.0.0
*/
2018-01-22 21:21:47 +00:00
this.vertexViewF32 = new Float32Array(this.vertexData);
2018-02-09 19:19:21 +00:00
/**
* Uint32 view of the array buffer containing the pipeline's vertices.
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @name Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#vertexViewU32
2018-02-09 19:19:21 +00:00
* @type {Uint32Array}
* @since 3.0.0
*/
2018-01-22 21:21:47 +00:00
this.vertexViewU32 = new Uint32Array(this.vertexData);
2018-02-09 19:19:21 +00:00
/**
* Size of the batch.
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @name Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#maxQuads
2018-02-09 19:19:21 +00:00
* @type {integer}
* @default 2000
* @since 3.0.0
*/
this.maxQuads = 2000;
2018-02-09 19:19:21 +00:00
2018-02-14 19:52:37 +00:00
/**
* Collection of batch information
2018-02-14 19:52:37 +00:00
*
2018-04-18 11:13:49 +00:00
* @name Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batches
2018-02-14 19:52:37 +00:00
* @type {array}
2018-02-15 14:31:15 +00:00
* @since 3.1.0
2018-02-14 19:52:37 +00:00
*/
this.batches = [];
2018-01-25 05:26:14 +00:00
this.mvpInit();
},
2018-02-14 19:52:37 +00:00
/**
* Assigns a texture to the current batch. If a texture is already set it creates
* a new batch object.
2018-02-14 19:52:37 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#setTexture2D
2018-02-15 14:31:15 +00:00
* @since 3.1.0
2018-02-14 19:52:37 +00:00
*
* @param {WebGLTexture} texture - WebGLTexture that will be assigned to the current batch.
* @param {integer} textureUnit - Texture unit to which the texture needs to be bound.
2018-02-14 19:52:37 +00:00
*
2018-04-18 11:13:49 +00:00
* @return {Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline} [description]
2018-02-14 19:52:37 +00:00
*/
setTexture2D: function (texture, unit)
{
2018-02-16 18:07:49 +00:00
if (!texture)
{
return this;
}
var batches = this.batches;
if (batches.length === 0)
{
this.pushBatch();
}
var batch = batches[batches.length - 1];
if (unit > 0)
{
if (batch.textures[unit - 1] &&
batch.textures[unit - 1] !== texture)
{
this.pushBatch();
}
batches[batches.length - 1].textures[unit - 1] = texture;
}
else
{
if (batch.texture !== null &&
batch.texture !== texture)
{
this.pushBatch();
}
batches[batches.length - 1].texture = texture;
}
2018-02-14 19:52:37 +00:00
return this;
},
2018-02-14 19:52:37 +00:00
/**
* Creates a new batch object and pushes it to a batch array.
* The batch object contains information relevant to the current
* vertex batch like the offset in the vertex buffer, vertex count and
* the textures used by that batch.
2018-02-14 19:52:37 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#pushBatch
2018-02-15 14:31:15 +00:00
* @since 3.1.0
2018-02-14 19:52:37 +00:00
*/
pushBatch: function ()
{
var batch = {
first: this.vertexCount,
texture: null,
textures: []
};
this.batches.push(batch);
},
2018-02-14 19:52:37 +00:00
/**
* Binds, uploads resources and processes all batches generating draw calls.
2018-02-14 19:52:37 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#flush
2018-02-15 14:31:15 +00:00
* @since 3.1.0
2018-03-05 21:49:08 +00:00
*
2018-04-18 11:13:49 +00:00
* @return {Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline} This Pipeline.
2018-02-14 19:52:37 +00:00
*/
flush: function ()
{
2018-02-16 18:07:49 +00:00
if (this.flushLocked)
{
return this;
}
this.flushLocked = true;
var gl = this.gl;
var renderer = this.renderer;
var vertexCount = this.vertexCount;
var topology = this.topology;
var vertexSize = this.vertexSize;
var batches = this.batches;
var batchCount = batches.length;
var batchVertexCount = 0;
var batch = null;
2018-02-16 18:07:49 +00:00
var batchNext;
var textureIndex;
var nTexture;
2018-02-16 18:07:49 +00:00
if (batchCount === 0 || vertexCount === 0)
{
this.flushLocked = false;
return this;
}
2018-02-16 18:07:49 +00:00
gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.bytes.subarray(0, vertexCount * vertexSize));
for (var index = 0; index < batches.length - 1; ++index)
{
batch = batches[index];
batchNext = batches[index + 1];
if (batch.textures.length > 0)
{
2018-02-16 18:07:49 +00:00
for (textureIndex = 0; textureIndex < batch.textures.length; ++textureIndex)
{
2018-02-16 18:07:49 +00:00
nTexture = batch.textures[textureIndex];
if (nTexture)
{
renderer.setTexture2D(nTexture, 1 + textureIndex);
}
}
2018-02-16 18:07:49 +00:00
gl.activeTexture(gl.TEXTURE0);
}
batchVertexCount = batchNext.first - batch.first;
2018-02-16 18:07:49 +00:00
if (batch.texture === null || batchVertexCount <= 0) { continue; }
renderer.setTexture2D(batch.texture, 0);
gl.drawArrays(topology, batch.first, batchVertexCount);
}
// Left over data
batch = batches[batches.length - 1];
if (batch.textures.length > 0)
{
2018-02-16 18:07:49 +00:00
for (textureIndex = 0; textureIndex < batch.textures.length; ++textureIndex)
{
2018-02-16 18:07:49 +00:00
nTexture = batch.textures[textureIndex];
if (nTexture)
{
renderer.setTexture2D(nTexture, 1 + textureIndex);
}
}
2018-02-16 18:07:49 +00:00
gl.activeTexture(gl.TEXTURE0);
}
batchVertexCount = vertexCount - batch.first;
if (batch.texture && batchVertexCount > 0)
{
renderer.setTexture2D(batch.texture, 0);
gl.drawArrays(topology, batch.first, batchVertexCount);
}
this.vertexCount = 0;
batches.length = 0;
this.pushBatch();
this.flushLocked = false;
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Called every time the pipeline needs to be used.
* It binds all necessary resources.
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#onBind
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-04-18 11:13:49 +00:00
* @return {Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline} [description]
2018-02-09 19:19:21 +00:00
*/
2018-01-25 05:26:14 +00:00
onBind: function ()
{
WebGLPipeline.prototype.onBind.call(this);
2018-04-19 12:28:10 +00:00
2018-01-25 05:26:14 +00:00
this.mvpUpdate();
if (this.batches.length === 0)
{
this.pushBatch();
}
2018-01-25 05:26:14 +00:00
return this;
2018-01-22 21:21:47 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* [description]
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#resize
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
* @param {number} width - [description]
* @param {number} height - [description]
* @param {number} resolution - [description]
*
2018-04-18 11:13:49 +00:00
* @return {Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline} [description]
2018-02-09 19:19:21 +00:00
*/
2018-01-22 21:21:47 +00:00
resize: function (width, height, resolution)
{
WebGLPipeline.prototype.resize.call(this, width, height, resolution);
2018-01-25 05:26:14 +00:00
this.projOrtho(0, this.width, this.height, 0, -1000.0, 1000.0);
return this;
},
2018-02-09 19:19:21 +00:00
/**
* Renders immediately a static tilemap. This function won't use
2018-05-09 13:30:31 +00:00
* the batching functionality of the pipeline.
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#drawStaticTilemapLayer
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-02-13 00:12:17 +00:00
* @param {Phaser.Tilemaps.StaticTilemapLayer} tilemap - [description]
2018-02-09 19:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
2018-04-19 12:28:10 +00:00
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
2018-02-16 18:07:49 +00:00
drawStaticTilemapLayer: function (tilemap)
2018-01-25 05:26:14 +00:00
{
if (tilemap.vertexCount > 0)
{
var pipelineVertexBuffer = this.vertexBuffer;
var gl = this.gl;
var renderer = this.renderer;
var frame = tilemap.tileset.image.get();
if (renderer.currentPipeline &&
renderer.currentPipeline.vertexCount > 0)
{
renderer.flush();
}
2018-01-22 21:21:47 +00:00
2018-01-25 05:26:14 +00:00
this.vertexBuffer = tilemap.vertexBuffer;
renderer.setPipeline(this);
renderer.setTexture2D(frame.source.glTexture, 0);
2018-01-25 05:26:14 +00:00
gl.drawArrays(this.topology, 0, tilemap.vertexCount);
this.vertexBuffer = pipelineVertexBuffer;
}
2018-01-22 21:21:47 +00:00
this.viewIdentity();
this.modelIdentity();
2018-01-22 21:21:47 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Renders contents of a ParticleEmitterManager. It'll batch all particles if possible.
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#drawEmitterManager
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-02-13 00:12:17 +00:00
* @param {Phaser.GameObjects.Particles.ParticleEmitterManager} emitterManager - [description]
2018-02-09 19:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
drawEmitterManager: function (emitterManager, camera, parentTransformMatrix)
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
this.renderer.setPipeline(this);
var roundPixels = camera.roundPixels;
var emitters = emitterManager.emitters.list;
var emitterCount = emitters.length;
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var renderer = this.renderer;
var maxQuads = this.maxQuads;
var cameraScrollX = camera.scrollX;
var cameraScrollY = camera.scrollY;
var cameraMatrix = camera.matrix.matrix;
var cma = cameraMatrix[0];
var cmb = cameraMatrix[1];
var cmc = cameraMatrix[2];
var cmd = cameraMatrix[3];
var cme = cameraMatrix[4];
var cmf = cameraMatrix[5];
var sin = Math.sin;
var cos = Math.cos;
var vertexComponentCount = this.vertexComponentCount;
var vertexCapacity = this.vertexCapacity;
var texture = emitterManager.defaultFrame.source.glTexture;
2018-03-26 22:55:38 +00:00
var pca, pcb, pcc, pcd, pce, pcf;
var pma, pmb, pmc, pmd, pme, pmf;
2018-03-26 22:55:38 +00:00
2018-04-05 10:01:24 +00:00
if (parentMatrix)
2018-03-26 22:55:38 +00:00
{
pma = parentMatrix[0];
pmb = parentMatrix[1];
pmc = parentMatrix[2];
pmd = parentMatrix[3];
pme = parentMatrix[4];
pmf = parentMatrix[5];
2018-03-26 22:55:38 +00:00
}
this.setTexture2D(texture, 0);
for (var emitterIndex = 0; emitterIndex < emitterCount; ++emitterIndex)
{
var emitter = emitters[emitterIndex];
var particles = emitter.alive;
var aliveLength = particles.length;
var batchCount = Math.ceil(aliveLength / maxQuads);
var particleOffset = 0;
var scrollX = cameraScrollX * emitter.scrollFactorX;
var scrollY = cameraScrollY * emitter.scrollFactorY;
if (parentMatrix)
{
var cse = -scrollX;
var csf = -scrollY;
var pse = cse * cma + csf * cmc + cme;
var psf = cse * cmb + csf * cmd + cmf;
pca = pma * cma + pmb * cmc;
pcb = pma * cmb + pmb * cmd;
pcc = pmc * cma + pmd * cmc;
pcd = pmc * cmb + pmd * cmd;
pce = pme * cma + pmf * cmc + pse;
pcf = pme * cmb + pmf * cmd + psf;
cma = pca;
cmb = pcb;
cmc = pcc;
cmd = pcd;
cme = pce;
cmf = pcf;
scrollX = 0.0;
scrollY = 0.0;
}
if (!emitter.visible || aliveLength === 0)
{
continue;
}
renderer.setBlendMode(emitter.blendMode);
if (this.vertexCount >= vertexCapacity)
{
this.flush();
this.setTexture2D(texture, 0);
}
var tintEffect = false;
for (var batchIndex = 0; batchIndex < batchCount; ++batchIndex)
{
var batchSize = Math.min(aliveLength, maxQuads);
for (var index = 0; index < batchSize; ++index)
{
var particle = particles[particleOffset + index];
if (particle.alpha <= 0)
{
continue;
}
var frame = particle.frame;
var uvs = frame.uvs;
var x = -(frame.halfWidth);
var y = -(frame.halfHeight);
var color = particle.color;
var xw = x + frame.width;
var yh = y + frame.height;
var sr = sin(particle.rotation);
var cr = cos(particle.rotation);
2018-06-25 21:53:42 +00:00
var sra = cr * particle.scaleX;
var srb = sr * particle.scaleX;
var src = -sr * particle.scaleY;
var srd = cr * particle.scaleY;
var sre = particle.x - scrollX;
var srf = particle.y - scrollY;
2018-06-25 21:53:42 +00:00
var mva = sra * cma + srb * cmc;
var mvb = sra * cmb + srb * cmd;
var mvc = src * cma + srd * cmc;
var mvd = src * cmb + srd * cmd;
var mve = sre * cma + srf * cmc + cme;
var mvf = sre * cmb + srf * cmd + cmf;
2018-06-25 21:53:42 +00:00
var tx0 = x * mva + y * mvc + mve;
var ty0 = x * mvb + y * mvd + mvf;
var tx1 = x * mva + yh * mvc + mve;
var ty1 = x * mvb + yh * mvd + mvf;
var tx2 = xw * mva + yh * mvc + mve;
var ty2 = xw * mvb + yh * mvd + mvf;
var tx3 = xw * mva + y * mvc + mve;
var ty3 = xw * mvb + y * mvd + mvf;
2018-06-25 21:53:42 +00:00
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
tx2 |= 0;
ty2 |= 0;
tx3 |= 0;
ty3 |= 0;
}
var vertexOffset = (this.vertexCount * vertexComponentCount) - 1;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x0;
vertexViewF32[++vertexOffset] = uvs.y0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = color;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = uvs.x1;
vertexViewF32[++vertexOffset] = uvs.y1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = color;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = uvs.x2;
vertexViewF32[++vertexOffset] = uvs.y2;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = color;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x0;
vertexViewF32[++vertexOffset] = uvs.y0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = color;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = uvs.x2;
vertexViewF32[++vertexOffset] = uvs.y2;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = color;
vertexViewF32[++vertexOffset] = tx3;
vertexViewF32[++vertexOffset] = ty3;
vertexViewF32[++vertexOffset] = uvs.x3;
vertexViewF32[++vertexOffset] = uvs.y3;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = color;
this.vertexCount += 6;
if (this.vertexCount >= vertexCapacity)
{
this.flush();
this.setTexture2D(texture, 0);
}
}
particleOffset += batchSize;
aliveLength -= batchSize;
if (this.vertexCount >= vertexCapacity)
{
this.flush();
this.setTexture2D(texture, 0);
}
}
}
this.setTexture2D(texture, 0);
},
2018-02-09 19:19:21 +00:00
/**
* Batches blitter game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#drawBlitter
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
* @param {Phaser.GameObjects.Blitter} blitter - [description]
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
drawBlitter: function (blitter, camera, parentTransformMatrix)
2018-01-22 21:21:47 +00:00
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
2018-01-23 16:38:58 +00:00
this.renderer.setPipeline(this);
2018-01-22 21:21:47 +00:00
var roundPixels = camera.roundPixels;
2018-01-23 19:29:47 +00:00
var getTint = Utils.getTintAppendFloatAlpha;
2018-01-22 21:21:47 +00:00
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var list = blitter.getRenderList();
var length = list.length;
var cameraMatrix = camera.matrix.matrix;
var a = cameraMatrix[0];
var b = cameraMatrix[1];
var c = cameraMatrix[2];
var d = cameraMatrix[3];
var e = cameraMatrix[4];
var f = cameraMatrix[5];
var cameraScrollX = camera.scrollX * blitter.scrollFactorX;
var cameraScrollY = camera.scrollY * blitter.scrollFactorY;
var batchCount = Math.ceil(length / this.maxQuads);
2018-01-22 21:21:47 +00:00
var batchOffset = 0;
2018-04-05 10:01:24 +00:00
if (parentMatrix)
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var cse = -cameraScrollX;
var csf = -cameraScrollY;
var pse = cse * a + csf * c + e;
var psf = cse * b + csf * d + f;
var pca = pma * a + pmb * c;
var pcb = pma * b + pmb * d;
var pcc = pmc * a + pmd * c;
var pcd = pmc * b + pmd * d;
var pce = pme * a + pmf * c + pse;
var pcf = pme * b + pmf * d + psf;
a = pca;
b = pcb;
c = pcc;
d = pcd;
e = pce;
f = pcf;
cameraScrollX = 0.0;
cameraScrollY = 0.0;
}
var blitterX = blitter.x - cameraScrollX;
var blitterY = blitter.y - cameraScrollY;
var prevTextureSourceIndex;
var alpha = camera.alpha * blitter.alpha;
var tintEffect = false;
2018-01-22 21:21:47 +00:00
for (var batchIndex = 0; batchIndex < batchCount; ++batchIndex)
{
var batchSize = Math.min(length, this.maxQuads);
2018-01-22 21:21:47 +00:00
for (var index = 0; index < batchSize; ++index)
{
var bob = list[batchOffset + index];
var frame = bob.frame;
var bobAlpha = bob.alpha * alpha;
if (bobAlpha === 0)
{
// Nothing to see here, moving on ...
continue;
}
var tint = getTint(0xffffff, bobAlpha);
2018-01-22 21:21:47 +00:00
var uvs = frame.uvs;
var flipX = bob.flipX;
var flipY = bob.flipY;
2018-02-16 18:07:49 +00:00
var width = frame.width * (flipX ? -1.0 : 1.0);
2018-01-22 21:21:47 +00:00
var height = frame.height * (flipY ? -1.0 : 1.0);
var x = blitterX + bob.x + frame.x + (frame.width * ((flipX) ? 1.0 : 0.0));
var y = blitterY + bob.y + frame.y + (frame.height * ((flipY) ? 1.0 : 0.0));
2018-02-16 18:07:49 +00:00
var xw = x + width;
2018-01-22 21:21:47 +00:00
var yh = y + height;
var tx0 = x * a + y * c + e;
var ty0 = x * b + y * d + f;
var tx1 = xw * a + yh * c + e;
var ty1 = xw * b + yh * d + f;
// Bind texture only if the Texture Source is different from before
if (frame.sourceIndex !== prevTextureSourceIndex)
{
this.setTexture2D(frame.texture.source[frame.sourceIndex].glTexture, 0);
prevTextureSourceIndex = frame.sourceIndex;
}
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
}
var vertexOffset = this.vertexCount * this.vertexComponentCount - 1;
2018-01-22 21:21:47 +00:00
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x0;
vertexViewF32[++vertexOffset] = uvs.y0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = uvs.x1;
vertexViewF32[++vertexOffset] = uvs.y1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = uvs.x2;
vertexViewF32[++vertexOffset] = uvs.y2;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x0;
vertexViewF32[++vertexOffset] = uvs.y0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = uvs.x2;
vertexViewF32[++vertexOffset] = uvs.y2;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x3;
vertexViewF32[++vertexOffset] = uvs.y3;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
2018-01-22 21:21:47 +00:00
this.vertexCount += 6;
if (this.vertexCount >= this.vertexCapacity)
{
this.flush();
prevTextureSourceIndex = -1;
}
2018-01-22 21:21:47 +00:00
}
batchOffset += batchSize;
2018-01-22 21:21:47 +00:00
length -= batchSize;
if (this.vertexCount >= this.vertexCapacity)
{
this.flush();
}
2018-01-22 21:21:47 +00:00
}
2018-01-23 16:38:58 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Batches Sprite game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchSprite
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
* @param {Phaser.GameObjects.Sprite} sprite - [description]
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
batchSprite: function (sprite, camera, parentTransformMatrix)
2018-01-23 16:38:58 +00:00
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
{
parentMatrix = parentTransformMatrix.matrix;
}
2018-01-23 16:38:58 +00:00
this.renderer.setPipeline(this);
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
2018-01-23 16:38:58 +00:00
var frame = sprite.frame;
var texture = frame.texture.source[frame.sourceIndex].glTexture;
var getTint = Utils.getTintAppendFloatAlpha;
2018-01-23 16:38:58 +00:00
var forceFlipY = (texture.isRenderTexture ? true : false);
var flipX = sprite.flipX;
var flipY = sprite.flipY ^ forceFlipY;
var uvs = frame.uvs;
2018-01-23 16:38:58 +00:00
var scaleX = sprite.scaleX;
var scaleY = sprite.scaleY;
var rotation = sprite.rotation;
var alphaTL = camera.alpha * sprite._alphaTL;
var alphaTR = camera.alpha * sprite._alphaTR;
var alphaBL = camera.alpha * sprite._alphaBL;
var alphaBR = camera.alpha * sprite._alphaBR;
2018-01-23 16:38:58 +00:00
var tintTL = sprite._tintTL;
var tintTR = sprite._tintTR;
var tintBL = sprite._tintBL;
var tintBR = sprite._tintBR;
var roundPixels = camera.roundPixels;
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var cameraMatrix = camera.matrix.matrix;
var width = frame.width * (flipX ? -1.0 : 1.0);
var height = frame.height * (flipY ? -1.0 : 1.0);
var x = -sprite.displayOriginX + frame.x + ((frame.width) * (flipX ? 1.0 : 0.0));
var y = -sprite.displayOriginY + frame.y + ((frame.height) * (flipY ? 1.0 : 0.0));
var xw = (roundPixels ? (x | 0) : x) + width;
var yh = (roundPixels ? (y | 0) : y) + height;
2018-01-23 16:38:58 +00:00
var sr = Math.sin(rotation);
var cr = Math.cos(rotation);
var sra = cr * scaleX;
var srb = sr * scaleX;
var src = -sr * scaleY;
2018-01-23 16:38:58 +00:00
var srd = cr * scaleY;
var sre = sprite.x;
var srf = sprite.y;
2018-01-23 16:38:58 +00:00
var cma = cameraMatrix[0];
var cmb = cameraMatrix[1];
var cmc = cameraMatrix[2];
var cmd = cameraMatrix[3];
var cme = cameraMatrix[4];
var cmf = cameraMatrix[5];
2018-03-26 22:55:38 +00:00
var mva, mvb, mvc, mvd, mve, mvf;
2018-04-05 10:01:24 +00:00
if (parentMatrix)
2018-03-26 22:55:38 +00:00
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var cse = -camera.scrollX * sprite.scrollFactorX;
var csf = -camera.scrollY * sprite.scrollFactorY;
var pse = cse * cma + csf * cmc + cme;
var psf = cse * cmb + csf * cmd + cmf;
var pca = pma * cma + pmb * cmc;
var pcb = pma * cmb + pmb * cmd;
var pcc = pmc * cma + pmd * cmc;
var pcd = pmc * cmb + pmd * cmd;
var pce = pme * cma + pmf * cmc + pse;
var pcf = pme * cmb + pmf * cmd + psf;
2018-03-26 22:55:38 +00:00
mva = sra * pca + srb * pcc;
mvb = sra * pcb + srb * pcd;
mvc = src * pca + srd * pcc;
mvd = src * pcb + srd * pcd;
mve = sre * pca + srf * pcc + pce;
mvf = sre * pcb + srf * pcd + pcf;
}
2018-04-05 10:01:24 +00:00
else
{
sre -= camera.scrollX * sprite.scrollFactorX;
srf -= camera.scrollY * sprite.scrollFactorY;
2018-04-05 10:01:24 +00:00
mva = sra * cma + srb * cmc;
mvb = sra * cmb + srb * cmd;
mvc = src * cma + srd * cmc;
mvd = src * cmb + srd * cmd;
mve = sre * cma + srf * cmc + cme;
mvf = sre * cmb + srf * cmd + cmf;
}
2018-03-26 22:55:38 +00:00
2018-01-23 16:38:58 +00:00
var tx0 = x * mva + y * mvc + mve;
var ty0 = x * mvb + y * mvd + mvf;
var tx1 = x * mva + yh * mvc + mve;
var ty1 = x * mvb + yh * mvd + mvf;
var tx2 = xw * mva + yh * mvc + mve;
var ty2 = xw * mvb + yh * mvd + mvf;
var tx3 = xw * mva + y * mvc + mve;
var ty3 = xw * mvb + y * mvd + mvf;
var vTintTL = getTint(tintTL, alphaTL);
var vTintTR = getTint(tintTR, alphaTR);
var vTintBL = getTint(tintBL, alphaBL);
var vTintBR = getTint(tintBR, alphaBR);
2018-01-23 20:32:20 +00:00
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
tx2 |= 0;
ty2 |= 0;
tx3 |= 0;
ty3 |= 0;
}
this.setTexture2D(texture, 0);
2018-01-23 19:29:47 +00:00
var vertexOffset = (this.vertexCount * this.vertexComponentCount) - 1;
var tintEffect = (sprite._isTinted && sprite.tintFill);
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x0;
vertexViewF32[++vertexOffset] = uvs.y0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTL;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = uvs.x1;
vertexViewF32[++vertexOffset] = uvs.y1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = uvs.x2;
vertexViewF32[++vertexOffset] = uvs.y2;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBR;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = uvs.x0;
vertexViewF32[++vertexOffset] = uvs.y0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = uvs.x2;
vertexViewF32[++vertexOffset] = uvs.y2;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBR;
vertexViewF32[++vertexOffset] = tx3;
vertexViewF32[++vertexOffset] = ty3;
vertexViewF32[++vertexOffset] = uvs.x3;
vertexViewF32[++vertexOffset] = uvs.y3;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTR;
2018-01-23 16:38:58 +00:00
this.vertexCount += 6;
2018-01-23 19:29:47 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Batches Mesh game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchMesh
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
* @param {Phaser.GameObjects.Mesh} mesh - [description]
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
batchMesh: function (mesh, camera, parentTransformMatrix)
2018-01-23 19:29:47 +00:00
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
2018-01-23 19:29:47 +00:00
var vertices = mesh.vertices;
var length = vertices.length;
var vertexCount = (length / 2)|0;
2018-01-23 20:32:20 +00:00
this.renderer.setPipeline(this);
2018-01-23 19:29:47 +00:00
if (this.vertexCount + vertexCount > this.vertexCapacity)
{
this.flush();
}
var roundPixels = camera.roundPixels;
2018-01-23 19:29:47 +00:00
var getTint = Utils.getTintAppendFloatAlpha;
var uvs = mesh.uv;
var colors = mesh.colors;
var alphas = mesh.alphas;
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var cameraMatrix = camera.matrix.matrix;
var frame = mesh.frame;
var texture = mesh.texture.source[frame.sourceIndex].glTexture;
var translateX = mesh.x;
var translateY = mesh.y;
2018-01-23 19:29:47 +00:00
var scaleX = mesh.scaleX;
var scaleY = mesh.scaleY;
var rotation = mesh.rotation;
2018-01-23 19:29:47 +00:00
var sr = Math.sin(rotation);
var cr = Math.cos(rotation);
var sra = cr * scaleX;
var srb = sr * scaleX;
var src = -sr * scaleY;
2018-01-23 19:29:47 +00:00
var srd = cr * scaleY;
var sre = translateX;
var srf = translateY;
var cma = cameraMatrix[0];
var cmb = cameraMatrix[1];
var cmc = cameraMatrix[2];
var cmd = cameraMatrix[3];
var cme = cameraMatrix[4];
var cmf = cameraMatrix[5];
2018-01-23 20:32:20 +00:00
var vertexOffset = 0;
var mva, mvb, mvc, mvd, mve, mvf;
2018-04-05 10:01:24 +00:00
if (parentMatrix)
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var cse = -camera.scrollX * mesh.scrollFactorX;
var csf = -camera.scrollY * mesh.scrollFactorY;
var pse = cse * cma + csf * cmc + cme;
var psf = cse * cmb + csf * cmd + cmf;
var pca = pma * cma + pmb * cmc;
var pcb = pma * cmb + pmb * cmd;
var pcc = pmc * cma + pmd * cmc;
var pcd = pmc * cmb + pmd * cmd;
var pce = pme * cma + pmf * cmc + pse;
var pcf = pme * cmb + pmf * cmd + psf;
mva = sra * pca + srb * pcc;
mvb = sra * pcb + srb * pcd;
mvc = src * pca + srd * pcc;
mvd = src * pcb + srd * pcd;
mve = sre * pca + srf * pcc + pce;
mvf = sre * pcb + srf * pcd + pcf;
}
2018-04-05 10:01:24 +00:00
else
{
sre -= camera.scrollX * mesh.scrollFactorX;
srf -= camera.scrollY * mesh.scrollFactorY;
2018-04-05 10:01:24 +00:00
mva = sra * cma + srb * cmc;
mvb = sra * cmb + srb * cmd;
mvc = src * cma + srd * cmc;
mvd = src * cmb + srd * cmd;
mve = sre * cma + srf * cmc + cme;
mvf = sre * cmb + srf * cmd + cmf;
}
this.setTexture2D(texture, 0);
2018-01-23 20:32:20 +00:00
vertexOffset = this.vertexCount * this.vertexComponentCount;
2018-01-23 19:29:47 +00:00
for (var index = 0, index0 = 0; index < length; index += 2)
{
var x = vertices[index + 0];
var y = vertices[index + 1];
var tx = x * mva + y * mvc + mve;
var ty = x * mvb + y * mvd + mvf;
if (roundPixels)
{
tx |= 0;
ty |= 0;
}
2018-01-23 19:29:47 +00:00
vertexViewF32[vertexOffset + 0] = tx;
vertexViewF32[vertexOffset + 1] = ty;
vertexViewF32[vertexOffset + 2] = uvs[index + 0];
vertexViewF32[vertexOffset + 3] = uvs[index + 1];
vertexViewF32[vertexOffset + 4] = 0;
vertexViewU32[vertexOffset + 5] = getTint(colors[index0], camera.alpha * alphas[index0]);
2018-01-23 19:29:47 +00:00
vertexOffset += 6;
2018-01-23 19:29:47 +00:00
index0 += 1;
}
2018-01-23 16:38:58 +00:00
2018-01-23 19:29:47 +00:00
this.vertexCount += vertexCount;
},
2018-02-09 19:19:21 +00:00
/**
* Batches BitmapText game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchBitmapText
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-02-13 00:12:17 +00:00
* @param {Phaser.GameObjects.BitmapText} bitmapText - [description]
2018-02-09 19:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
batchBitmapText: function (bitmapText, camera, parentTransformMatrix)
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
this.renderer.setPipeline(this);
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
var roundPixels = camera.roundPixels;
var text = bitmapText.text;
var textLength = text.length;
var getTint = Utils.getTintAppendFloatAlpha;
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var cameraMatrix = camera.matrix.matrix;
2018-01-26 00:05:38 +00:00
var cameraWidth = camera.width + 50;
var cameraHeight = camera.height + 50;
var cameraX = -50;
var cameraY = -50;
var frame = bitmapText.frame;
var textureSource = bitmapText.texture.source[frame.sourceIndex];
var cameraScrollX = camera.scrollX * bitmapText.scrollFactorX;
var cameraScrollY = camera.scrollY * bitmapText.scrollFactorY;
var fontData = bitmapText.fontData;
var lineHeight = fontData.lineHeight;
var scale = (bitmapText.fontSize / fontData.size);
var chars = fontData.chars;
var alpha = camera.alpha * bitmapText.alpha;
var vTintTL = getTint(bitmapText._tintTL, alpha);
var vTintTR = getTint(bitmapText._tintTR, alpha);
var vTintBL = getTint(bitmapText._tintBL, alpha);
var vTintBR = getTint(bitmapText._tintBR, alpha);
var tintEffect = (bitmapText._isTinted && bitmapText.tintFill);
var srcX = bitmapText.x;
var srcY = bitmapText.y;
var textureX = frame.cutX;
var textureY = frame.cutY;
var textureWidth = textureSource.width;
var textureHeight = textureSource.height;
var texture = textureSource.glTexture;
var xAdvance = 0;
var yAdvance = 0;
var indexCount = 0;
var charCode = 0;
var glyph = null;
var glyphX = 0;
var glyphY = 0;
var glyphW = 0;
var glyphH = 0;
var x = 0;
var y = 0;
var xw = 0;
var yh = 0;
2018-02-16 18:07:49 +00:00
var tx0;
var ty0;
var tx1;
var ty1;
var tx2;
var ty2;
var tx3;
var ty3;
var umin = 0;
var umax = 0;
var vmin = 0;
var vmax = 0;
var lastGlyph = null;
var lastCharCode = 0;
var translateX = srcX + frame.x;
var translateY = srcY + frame.y;
var rotation = bitmapText.rotation;
var scaleX = bitmapText.scaleX;
var scaleY = bitmapText.scaleY;
var letterSpacing = bitmapText.letterSpacing;
var sr = Math.sin(rotation);
var cr = Math.cos(rotation);
var sra = cr * scaleX;
var srb = sr * scaleX;
var src = -sr * scaleY;
var srd = cr * scaleY;
var sre = translateX;
var srf = translateY;
var cma = cameraMatrix[0];
var cmb = cameraMatrix[1];
var cmc = cameraMatrix[2];
var cmd = cameraMatrix[3];
var cme = cameraMatrix[4];
var cmf = cameraMatrix[5];
var vertexOffset = 0;
var mva, mvb, mvc, mvd, mve, mvf;
2018-04-05 10:01:24 +00:00
if (parentMatrix)
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var cse = -cameraScrollX;
var csf = -cameraScrollY;
var pse = cse * cma + csf * cmc + cme;
var psf = cse * cmb + csf * cmd + cmf;
var pca = pma * cma + pmb * cmc;
var pcb = pma * cmb + pmb * cmd;
var pcc = pmc * cma + pmd * cmc;
var pcd = pmc * cmb + pmd * cmd;
var pce = pme * cma + pmf * cmc + pse;
var pcf = pme * cmb + pmf * cmd + psf;
mva = sra * pca + srb * pcc;
mvb = sra * pcb + srb * pcd;
mvc = src * pca + srd * pcc;
mvd = src * pcb + srd * pcd;
mve = sre * pca + srf * pcc + pce;
mvf = sre * pcb + srf * pcd + pcf;
}
2018-04-05 10:01:24 +00:00
else
{
sre -= cameraScrollX;
srf -= cameraScrollY;
2018-04-05 10:01:24 +00:00
mva = sra * cma + srb * cmc;
mvb = sra * cmb + srb * cmd;
mvc = src * cma + srd * cmc;
mvd = src * cmb + srd * cmd;
mve = sre * cma + srf * cmc + cme;
mvf = sre * cmb + srf * cmd + cmf;
}
this.setTexture2D(texture, 0);
for (var index = 0; index < textLength; ++index)
{
charCode = text.charCodeAt(index);
if (charCode === 10)
{
xAdvance = 0;
indexCount = 0;
yAdvance += lineHeight;
lastGlyph = null;
continue;
}
glyph = chars[charCode];
if (!glyph)
{
continue;
}
glyphX = textureX + glyph.x;
glyphY = textureY + glyph.y;
2018-01-24 18:55:23 +00:00
glyphW = glyph.width;
glyphH = glyph.height;
2018-01-24 18:55:23 +00:00
x = (indexCount + glyph.xOffset + xAdvance) * scale;
y = (glyph.yOffset + yAdvance) * scale;
if (lastGlyph !== null)
{
var kerningOffset = glyph.kerning[lastCharCode];
x += (kerningOffset !== undefined) ? kerningOffset : 0;
2018-02-16 18:07:49 +00:00
}
2018-01-24 18:55:23 +00:00
xAdvance += glyph.xAdvance + letterSpacing;
2018-01-26 00:05:38 +00:00
indexCount += 1;
lastGlyph = glyph;
lastCharCode = charCode;
// Nothing to render or a space? Then skip to the next glyph
if (glyphW === 0 || glyphH === 0 || charCode === 32)
{
continue;
}
x -= bitmapText.displayOriginX;
y -= bitmapText.displayOriginY;
xw = x + glyphW * scale;
yh = y + glyphH * scale;
tx0 = x * mva + y * mvc + mve;
ty0 = x * mvb + y * mvd + mvf;
tx1 = x * mva + yh * mvc + mve;
ty1 = x * mvb + yh * mvd + mvf;
tx2 = xw * mva + yh * mvc + mve;
ty2 = xw * mvb + yh * mvd + mvf;
tx3 = xw * mva + y * mvc + mve;
ty3 = xw * mvb + y * mvd + mvf;
2018-01-24 18:55:23 +00:00
umin = glyphX / textureWidth;
umax = (glyphX + glyphW) / textureWidth;
vmin = glyphY / textureHeight;
vmax = (glyphY + glyphH) / textureHeight;
2018-01-26 00:05:38 +00:00
if ((tx0 < cameraX || tx0 > cameraWidth || ty0 < cameraY || ty0 > cameraHeight) &&
(tx1 < cameraX || tx1 > cameraWidth || ty1 < cameraY || ty1 > cameraHeight) &&
(tx2 < cameraX || tx2 > cameraWidth || ty2 < cameraY || ty2 > cameraHeight) &&
(tx3 < cameraX || tx3 > cameraWidth || ty3 < cameraY || ty3 > cameraHeight))
2018-01-26 00:05:38 +00:00
{
continue;
}
2018-01-24 18:55:23 +00:00
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
2018-01-24 18:55:23 +00:00
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
tx2 |= 0;
ty2 |= 0;
tx3 |= 0;
ty3 |= 0;
}
vertexOffset = (this.vertexCount * this.vertexComponentCount) - 1;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = umin;
vertexViewF32[++vertexOffset] = vmin;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTL;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = umin;
vertexViewF32[++vertexOffset] = vmax;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = umax;
vertexViewF32[++vertexOffset] = vmax;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBR;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = umin;
vertexViewF32[++vertexOffset] = vmin;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = umax;
vertexViewF32[++vertexOffset] = vmax;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBR;
vertexViewF32[++vertexOffset] = tx3;
vertexViewF32[++vertexOffset] = ty3;
vertexViewF32[++vertexOffset] = umax;
vertexViewF32[++vertexOffset] = vmin;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTR;
2018-01-24 18:55:23 +00:00
this.vertexCount += 6;
}
},
2018-02-09 19:19:21 +00:00
/**
* Batches DynamicBitmapText game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchDynamicBitmapText
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-02-13 00:12:17 +00:00
* @param {Phaser.GameObjects.DynamicBitmapText} bitmapText - [description]
2018-02-09 19:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
batchDynamicBitmapText: function (bitmapText, camera, parentTransformMatrix)
2018-01-24 18:55:23 +00:00
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
2018-01-24 18:55:23 +00:00
this.renderer.setPipeline(this);
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
var roundPixels = camera.roundPixels;
2018-01-24 18:55:23 +00:00
var displayCallback = bitmapText.displayCallback;
var text = bitmapText.text;
var textLength = text.length;
var getTint = Utils.getTintAppendFloatAlpha;
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var renderer = this.renderer;
var cameraMatrix = camera.matrix.matrix;
var frame = bitmapText.frame;
var textureSource = bitmapText.texture.source[frame.sourceIndex];
var cameraScrollX = camera.scrollX * bitmapText.scrollFactorX;
var cameraScrollY = camera.scrollY * bitmapText.scrollFactorY;
var scrollX = bitmapText.scrollX;
var scrollY = bitmapText.scrollY;
var fontData = bitmapText.fontData;
var lineHeight = fontData.lineHeight;
var scale = (bitmapText.fontSize / fontData.size);
var chars = fontData.chars;
var alpha = camera.alpha * bitmapText.alpha;
var vTintTL = getTint(bitmapText._tintTL, alpha);
var vTintTR = getTint(bitmapText._tintTR, alpha);
var vTintBL = getTint(bitmapText._tintBL, alpha);
var vTintBR = getTint(bitmapText._tintBR, alpha);
var tintEffect = (bitmapText._isTinted && bitmapText.tintFill);
2018-01-24 18:55:23 +00:00
var srcX = bitmapText.x;
var srcY = bitmapText.y;
var textureX = frame.cutX;
var textureY = frame.cutY;
var textureWidth = textureSource.width;
var textureHeight = textureSource.height;
var texture = textureSource.glTexture;
var xAdvance = 0;
var yAdvance = 0;
var indexCount = 0;
var charCode = 0;
var glyph = null;
var glyphX = 0;
var glyphY = 0;
var glyphW = 0;
var glyphH = 0;
var x = 0;
var y = 0;
var xw = 0;
2018-02-16 18:07:49 +00:00
var tx0;
var ty0;
var tx1;
var ty1;
var tx2;
var ty2;
var tx3;
var ty3;
2018-01-24 18:55:23 +00:00
var yh = 0;
var umin = 0;
var umax = 0;
var vmin = 0;
var vmax = 0;
var lastGlyph = null;
var lastCharCode = 0;
var translateX = srcX + frame.x;
var translateY = srcY + frame.y;
var rotation = bitmapText.rotation;
2018-01-24 18:55:23 +00:00
var scaleX = bitmapText.scaleX;
var scaleY = bitmapText.scaleY;
var letterSpacing = bitmapText.letterSpacing;
2018-01-24 18:55:23 +00:00
var sr = Math.sin(rotation);
var cr = Math.cos(rotation);
var sra = cr * scaleX;
var srb = sr * scaleX;
var src = -sr * scaleY;
2018-01-24 18:55:23 +00:00
var srd = cr * scaleY;
var sre = translateX;
var srf = translateY;
var cma = cameraMatrix[0];
var cmb = cameraMatrix[1];
var cmc = cameraMatrix[2];
var cmd = cameraMatrix[3];
var cme = cameraMatrix[4];
var cmf = cameraMatrix[5];
var crop = (bitmapText.cropWidth > 0 || bitmapText.cropHeight > 0);
var uta, utb, utc, utd, ute, utf;
var vertexOffset = 0;
var mva, mvb, mvc, mvd, mve, mvf;
2018-04-05 10:01:24 +00:00
if (parentMatrix)
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var cse = -cameraScrollX;
var csf = -cameraScrollY;
var pse = cse * cma + csf * cmc + cme;
var psf = cse * cmb + csf * cmd + cmf;
var pca = pma * cma + pmb * cmc;
var pcb = pma * cmb + pmb * cmd;
var pcc = pmc * cma + pmd * cmc;
var pcd = pmc * cmb + pmd * cmd;
var pce = pme * cma + pmf * cmc + pse;
var pcf = pme * cmb + pmf * cmd + psf;
mva = sra * pca + srb * pcc;
mvb = sra * pcb + srb * pcd;
mvc = src * pca + srd * pcc;
mvd = src * pcb + srd * pcd;
mve = sre * pca + srf * pcc + pce;
mvf = sre * pcb + srf * pcd + pcf;
}
2018-04-05 10:01:24 +00:00
else
{
sre -= cameraScrollX;
srf -= cameraScrollY;
2018-04-05 10:01:24 +00:00
mva = sra * cma + srb * cmc;
mvb = sra * cmb + srb * cmd;
mvc = src * cma + srd * cmc;
mvd = src * cmb + srd * cmd;
mve = sre * cma + srf * cmc + cme;
mvf = sre * cmb + srf * cmd + cmf;
}
this.setTexture2D(texture, 0);
2018-01-24 18:55:23 +00:00
if (crop)
{
renderer.pushScissor(
2018-02-16 18:07:49 +00:00
bitmapText.x,
bitmapText.y,
bitmapText.cropWidth * bitmapText.scaleX,
2018-01-24 18:55:23 +00:00
bitmapText.cropHeight * bitmapText.scaleY
);
2018-02-16 18:07:49 +00:00
}
2018-01-24 18:55:23 +00:00
for (var index = 0; index < textLength; ++index)
{
scale = (bitmapText.fontSize / bitmapText.fontData.size);
rotation = 0;
charCode = text.charCodeAt(index);
if (charCode === 10)
{
2018-01-24 18:55:23 +00:00
xAdvance = 0;
indexCount = 0;
yAdvance += lineHeight;
lastGlyph = null;
continue;
}
2018-01-24 18:55:23 +00:00
glyph = chars[charCode];
if (!glyph)
{
continue;
}
glyphX = textureX + glyph.x;
glyphY = textureY + glyph.y;
glyphW = glyph.width;
glyphH = glyph.height;
x = (indexCount + glyph.xOffset + xAdvance) - scrollX;
y = (glyph.yOffset + yAdvance) - scrollY;
if (lastGlyph !== null)
{
var kerningOffset = glyph.kerning[lastCharCode];
x += (kerningOffset !== undefined) ? kerningOffset : 0;
}
xAdvance += glyph.xAdvance + letterSpacing;
indexCount += 1;
lastGlyph = glyph;
lastCharCode = charCode;
// Nothing to render or a space? Then skip to the next glyph
if (glyphW === 0 || glyphH === 0 || charCode === 32)
{
continue;
}
2018-01-24 18:55:23 +00:00
if (displayCallback)
{
2018-02-16 18:07:49 +00:00
var output = displayCallback({
color: 0,
tint: {
topLeft: vTintTL,
topRight: vTintTR,
bottomLeft: vTintBL,
bottomRight: vTintBR
2018-02-16 18:07:49 +00:00
},
index: index,
charCode: charCode,
x: x,
y: y,
scale: scale,
rotation: 0,
data: glyph.data
2018-01-24 18:55:23 +00:00
});
x = output.x;
y = output.y;
scale = output.scale;
rotation = output.rotation;
if (output.color)
{
vTintTL = output.color;
vTintTR = output.color;
vTintBL = output.color;
vTintBR = output.color;
2018-01-24 18:55:23 +00:00
}
else
{
vTintTL = output.tint.topLeft;
vTintTR = output.tint.topRight;
vTintBL = output.tint.bottomLeft;
vTintBR = output.tint.bottomRight;
2018-01-24 18:55:23 +00:00
}
vTintTL = getTint(vTintTL, alpha);
vTintTR = getTint(vTintTR, alpha);
vTintBL = getTint(vTintBL, alpha);
vTintBR = getTint(vTintBR, alpha);
2018-01-24 18:55:23 +00:00
}
x -= bitmapText.displayOriginX;
y -= bitmapText.displayOriginY;
2018-01-24 18:55:23 +00:00
x *= scale;
y *= scale;
sr = Math.sin(rotation);
cr = Math.cos(rotation);
2018-01-24 18:55:23 +00:00
uta = cr * scale;
utb = sr * scale;
utc = -sr * scale;
2018-01-24 18:55:23 +00:00
utd = cr * scale;
ute = x;
utf = y;
sra = uta * mva + utb * mvc;
srb = uta * mvb + utb * mvd;
src = utc * mva + utd * mvc;
srd = utc * mvb + utd * mvd;
sre = ute * mva + utf * mvc + mve;
srf = ute * mvb + utf * mvd + mvf;
xw = glyphW;
yh = glyphH;
tx0 = sre;
ty0 = srf;
tx1 = yh * src + sre;
ty1 = yh * srd + srf;
tx2 = xw * sra + yh * src + sre;
ty2 = xw * srb + yh * srd + srf;
tx3 = xw * sra + sre;
ty3 = xw * srb + srf;
umin = glyphX / textureWidth;
umax = (glyphX + glyphW) / textureWidth;
vmin = glyphY / textureHeight;
vmax = (glyphY + glyphH) / textureHeight;
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
tx2 |= 0;
ty2 |= 0;
tx3 |= 0;
ty3 |= 0;
}
vertexOffset = (this.vertexCount * this.vertexComponentCount) - 1;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = umin;
vertexViewF32[++vertexOffset] = vmin;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTL;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = umin;
vertexViewF32[++vertexOffset] = vmax;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = umax;
vertexViewF32[++vertexOffset] = vmax;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBR;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = umin;
vertexViewF32[++vertexOffset] = vmin;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = umax;
vertexViewF32[++vertexOffset] = vmax;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintBR;
vertexViewF32[++vertexOffset] = tx3;
vertexViewF32[++vertexOffset] = ty3;
vertexViewF32[++vertexOffset] = umax;
vertexViewF32[++vertexOffset] = vmin;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = vTintTR;
this.vertexCount += 6;
}
2018-01-24 18:55:23 +00:00
if (crop)
{
renderer.popScissor();
}
2018-01-30 03:38:31 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Batches Text game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchText
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-02-13 00:12:17 +00:00
* @param {Phaser.GameObjects.Text} text - [description]
2018-02-09 19:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
batchText: function (text, camera, parentTransformMatrix)
2018-01-30 03:38:31 +00:00
{
var getTint = Utils.getTintAppendFloatAlpha;
this.batchTexture(
text,
text.canvasTexture,
text.canvasTexture.width, text.canvasTexture.height,
text.x, text.y,
text.canvasTexture.width, text.canvasTexture.height,
text.scaleX, text.scaleY,
text.rotation,
text.flipX, text.flipY,
text.scrollFactorX, text.scrollFactorY,
text.displayOriginX, text.displayOriginY,
0, 0, text.canvasTexture.width, text.canvasTexture.height,
getTint(text._tintTL, camera.alpha * text._alphaTL),
getTint(text._tintTR, camera.alpha * text._alphaTR),
getTint(text._tintBL, camera.alpha * text._alphaBL),
getTint(text._tintBR, camera.alpha * text._alphaBR),
(text._isTinted && text.tintFill),
2018-01-30 03:38:31 +00:00
0, 0,
2018-03-26 22:55:38 +00:00
camera,
parentTransformMatrix
2018-01-30 03:38:31 +00:00
);
},
2018-02-09 19:19:21 +00:00
/**
* Batches DynamicTilemapLayer game object
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchDynamicTilemapLayer
2018-02-09 19:19:21 +00:00
* @since 3.0.0
*
2018-02-13 00:12:17 +00:00
* @param {Phaser.Tilemaps.DynamicTilemapLayer} tilemapLayer - [description]
2018-02-09 19:19:21 +00:00
* @param {Phaser.Cameras.Scene2D.Camera} camera - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-02-09 19:19:21 +00:00
*/
batchDynamicTilemapLayer: function (tilemapLayer, camera, parentTransformMatrix)
2018-01-30 03:38:31 +00:00
{
var renderTiles = tilemapLayer.culledTiles;
var length = renderTiles.length;
var texture = tilemapLayer.tileset.image.get().source.glTexture;
var tileset = tilemapLayer.tileset;
var scrollFactorX = tilemapLayer.scrollFactorX;
var scrollFactorY = tilemapLayer.scrollFactorY;
var alpha = camera.alpha * tilemapLayer.alpha;
2018-01-30 03:38:31 +00:00
var x = tilemapLayer.x;
var y = tilemapLayer.y;
var sx = tilemapLayer.scaleX;
var sy = tilemapLayer.scaleY;
var getTint = Utils.getTintAppendFloatAlpha;
for (var index = 0; index < length; ++index)
{
var tile = renderTiles[index];
var tileTexCoords = tileset.getTileTextureCoordinates(tile.index);
if (tileTexCoords === null) { continue; }
2018-02-07 16:14:52 +00:00
var frameWidth = tile.width;
var frameHeight = tile.height;
var frameX = tileTexCoords.x;
var frameY = tileTexCoords.y;
2018-01-30 03:38:31 +00:00
var tint = getTint(tile.tint, alpha * tile.alpha);
this.batchTexture(
tilemapLayer,
texture,
texture.width, texture.height,
2018-02-07 19:30:18 +00:00
(tile.width / 2) + x + tile.pixelX * sx, (tile.height / 2) + y + tile.pixelY * sy,
tile.width * sx, tile.height * sy,
1, 1,
2018-02-07 16:14:52 +00:00
tile.rotation,
tile.flipX, tile.flipY,
2018-01-30 03:38:31 +00:00
scrollFactorX, scrollFactorY,
2018-02-07 19:30:18 +00:00
(tile.width / 2), (tile.height / 2),
2018-01-30 03:38:31 +00:00
frameX, frameY, frameWidth, frameHeight,
tint, tint, tint, tint, false,
2018-01-30 03:38:31 +00:00
0, 0,
2018-03-26 22:55:38 +00:00
camera,
parentTransformMatrix
2018-01-30 03:38:31 +00:00
);
2018-02-16 18:07:49 +00:00
}
2018-01-30 03:38:31 +00:00
},
2018-02-09 19:19:21 +00:00
/**
* Generic function for batching a textured quad
2018-02-09 19:19:21 +00:00
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#batchTexture
2018-02-09 19:19:21 +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} 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
2018-02-09 19:19:21 +00:00
*/
2018-01-30 03:38:31 +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,
2018-01-30 03:38:31 +00:00
uOffset, vOffset,
2018-04-05 08:02:36 +00:00
camera,
parentTransformMatrix)
2018-01-30 03:38:31 +00:00
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
2018-01-30 03:38:31 +00:00
this.renderer.setPipeline(this);
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
flipY = flipY ^ (texture.isRenderTexture ? 1 : 0);
var roundPixels = camera.roundPixels;
2018-01-30 03:38:31 +00:00
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var cameraMatrix = camera.matrix.matrix;
var width = srcWidth * (flipX ? -1.0 : 1.0);
var height = srcHeight * (flipY ? -1.0 : 1.0);
var x = -displayOriginX + ((srcWidth) * (flipX ? 1.0 : 0.0));
var y = -displayOriginY + ((srcHeight) * (flipY ? 1.0 : 0.0));
2018-06-28 12:07:59 +00:00
// var x = -displayOriginX + frameX + ((frameWidth) * (flipX ? 1.0 : 0.0));
// var y = -displayOriginY + frameY + ((frameHeight) * (flipY ? 1.0 : 0.0));
2018-06-28 12:07:59 +00:00
var xw = (roundPixels ? (x | 0) : x) + width;
var yh = (roundPixels ? (y | 0) : y) + height;
2018-01-30 03:38:31 +00:00
var sr = Math.sin(rotation);
var cr = Math.cos(rotation);
var sra = cr * scaleX;
var srb = sr * scaleX;
var src = -sr * scaleY;
2018-01-30 03:38:31 +00:00
var srd = cr * scaleY;
var sre = srcX;
var srf = srcY;
2018-01-30 03:38:31 +00:00
var cma = cameraMatrix[0];
var cmb = cameraMatrix[1];
var cmc = cameraMatrix[2];
var cmd = cameraMatrix[3];
var cme = cameraMatrix[4];
var cmf = cameraMatrix[5];
var mva, mvb, mvc, mvd, mve, mvf;
2018-04-05 10:01:24 +00:00
if (parentMatrix)
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var cse = -camera.scrollX * scrollFactorX;
var csf = -camera.scrollY * scrollFactorY;
var pse = cse * cma + csf * cmc + cme;
var psf = cse * cmb + csf * cmd + cmf;
var pca = pma * cma + pmb * cmc;
var pcb = pma * cmb + pmb * cmd;
var pcc = pmc * cma + pmd * cmc;
var pcd = pmc * cmb + pmd * cmd;
var pce = pme * cma + pmf * cmc + pse;
var pcf = pme * cmb + pmf * cmd + psf;
mva = sra * pca + srb * pcc;
mvb = sra * pcb + srb * pcd;
mvc = src * pca + srd * pcc;
mvd = src * pcb + srd * pcd;
mve = sre * pca + srf * pcc + pce;
mvf = sre * pcb + srf * pcd + pcf;
}
2018-04-05 10:01:24 +00:00
else
{
sre -= camera.scrollX * scrollFactorX;
srf -= camera.scrollY * scrollFactorY;
2018-04-05 10:01:24 +00:00
mva = sra * cma + srb * cmc;
mvb = sra * cmb + srb * cmd;
mvc = src * cma + srd * cmc;
mvd = src * cmb + srd * cmd;
mve = sre * cma + srf * cmc + cme;
mvf = sre * cmb + srf * cmd + cmf;
}
2018-01-30 03:38:31 +00:00
var tx0 = x * mva + y * mvc + mve;
var ty0 = x * mvb + y * mvd + mvf;
var tx1 = x * mva + yh * mvc + mve;
var ty1 = x * mvb + yh * mvd + mvf;
var tx2 = xw * mva + yh * mvc + mve;
var ty2 = xw * mvb + yh * mvd + mvf;
var tx3 = xw * mva + y * mvc + mve;
var ty3 = xw * mvb + y * mvd + mvf;
var u0 = (frameX / textureWidth) + uOffset;
var v0 = (frameY / textureHeight) + vOffset;
var u1 = (frameX + frameWidth) / textureWidth + uOffset;
var v1 = (frameY + frameHeight) / textureHeight + vOffset;
2018-01-30 03:38:31 +00:00
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
tx2 |= 0;
ty2 |= 0;
tx3 |= 0;
ty3 |= 0;
}
this.setTexture2D(texture, 0);
var vertexOffset = (this.vertexCount * this.vertexComponentCount) - 1;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = u0;
vertexViewF32[++vertexOffset] = v0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tintTL;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = u0;
vertexViewF32[++vertexOffset] = v1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tintTR;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = u1;
vertexViewF32[++vertexOffset] = v1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tintBL;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = u0;
vertexViewF32[++vertexOffset] = v0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tintTL;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = u1;
vertexViewF32[++vertexOffset] = v1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tintBL;
vertexViewF32[++vertexOffset] = tx3;
vertexViewF32[++vertexOffset] = ty3;
vertexViewF32[++vertexOffset] = u1;
vertexViewF32[++vertexOffset] = v0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tintBR;
2018-01-30 03:38:31 +00:00
this.vertexCount += 6;
},
2018-03-05 21:49:08 +00:00
/**
* Immediately draws a texture with no batching.
*
2018-04-18 11:13:49 +00:00
* @method Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline#drawTexture
2018-03-05 21:49:08 +00:00
* @since 3.2.0
*
* @param {WebGLTexture} texture [description]
* @param {number} srcX - [description]
* @param {number} srcY - [description]
* @param {number} tint - [description]
* @param {number} alpha - [description]
* @param {number} frameX - [description]
* @param {number} frameY - [description]
* @param {number} frameWidth - [description]
* @param {number} frameHeight - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} transformMatrix - [description]
* @param {Phaser.GameObjects.Components.TransformMatrix} parentTransformMatrix - [description]
2018-03-05 21:49:08 +00:00
*
2018-04-18 11:13:49 +00:00
* @return {Phaser.Renderer.WebGL.Pipelines.TextureTintPipeline} This Pipeline.
2018-03-05 21:49:08 +00:00
*/
drawTexture: function (
texture,
srcX, srcY,
tint, alpha,
frameX, frameY, frameWidth, frameHeight,
2018-04-05 08:02:36 +00:00
transformMatrix,
parentTransformMatrix
)
{
2018-03-26 22:55:38 +00:00
var parentMatrix = null;
2018-04-05 10:01:24 +00:00
if (parentTransformMatrix)
2018-03-26 22:55:38 +00:00
{
parentMatrix = parentTransformMatrix.matrix;
}
this.renderer.setPipeline(this);
if (this.vertexCount + 6 > this.vertexCapacity)
{
this.flush();
}
2018-06-27 14:27:16 +00:00
var roundPixels = this.renderer.config.roundPixels;
var vertexViewF32 = this.vertexViewF32;
var vertexViewU32 = this.vertexViewU32;
var width = frameWidth;
var height = frameHeight;
var x = srcX;
var y = srcY;
var xw = x + width;
var yh = y + height;
var mva = transformMatrix[0];
var mvb = transformMatrix[1];
var mvc = transformMatrix[2];
var mvd = transformMatrix[3];
var mve = transformMatrix[4];
var mvf = transformMatrix[5];
2018-04-05 10:01:24 +00:00
if (parentMatrix)
{
var pma = parentMatrix[0];
var pmb = parentMatrix[1];
var pmc = parentMatrix[2];
var pmd = parentMatrix[3];
var pme = parentMatrix[4];
var pmf = parentMatrix[5];
var pca = mva * pma + mvb * pmc;
var pcb = mva * pmb + mvb * pmd;
var pcc = mvc * pma + mvd * pmc;
var pcd = mvc * pmb + mvd * pmd;
var pce = mve * pma + mvf * pmc + pme;
var pcf = mve * pmb + mvf * pmd + pmf;
mva = pca;
mvb = pcb;
mvc = pcc;
mvd = pcd;
mve = pce;
mvf = pcf;
}
var tx0 = x * mva + y * mvc + mve;
var ty0 = x * mvb + y * mvd + mvf;
var tx1 = x * mva + yh * mvc + mve;
var ty1 = x * mvb + yh * mvd + mvf;
var tx2 = xw * mva + yh * mvc + mve;
var ty2 = xw * mvb + yh * mvd + mvf;
var tx3 = xw * mva + y * mvc + mve;
var ty3 = xw * mvb + y * mvd + mvf;
var textureWidth = texture.width;
var textureHeight = texture.height;
var u0 = (frameX / textureWidth);
var v0 = (frameY / textureHeight);
var u1 = (frameX + frameWidth) / textureWidth;
var v1 = (frameY + frameHeight) / textureHeight;
var tintEffect = 0;
2018-03-05 21:49:08 +00:00
tint = Utils.getTintAppendFloatAlpha(tint, alpha);
this.setTexture2D(texture, 0);
if (roundPixels)
{
tx0 |= 0;
ty0 |= 0;
tx1 |= 0;
ty1 |= 0;
tx2 |= 0;
ty2 |= 0;
tx3 |= 0;
ty3 |= 0;
}
var vertexOffset = (this.vertexCount * this.vertexComponentCount) - 1;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = u0;
vertexViewF32[++vertexOffset] = v0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx1;
vertexViewF32[++vertexOffset] = ty1;
vertexViewF32[++vertexOffset] = u0;
vertexViewF32[++vertexOffset] = v1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = u1;
vertexViewF32[++vertexOffset] = v1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx0;
vertexViewF32[++vertexOffset] = ty0;
vertexViewF32[++vertexOffset] = u0;
vertexViewF32[++vertexOffset] = v0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx2;
vertexViewF32[++vertexOffset] = ty2;
vertexViewF32[++vertexOffset] = u1;
vertexViewF32[++vertexOffset] = v1;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
vertexViewF32[++vertexOffset] = tx3;
vertexViewF32[++vertexOffset] = ty3;
vertexViewF32[++vertexOffset] = u1;
vertexViewF32[++vertexOffset] = v0;
vertexViewF32[++vertexOffset] = tintEffect;
vertexViewU32[++vertexOffset] = tint;
this.vertexCount += 6;
// Force an immediate draw
this.flush();
2018-01-22 21:21:47 +00:00
}
2018-01-24 22:29:57 +00:00
2018-01-22 21:21:47 +00:00
});
2018-01-22 22:51:15 +00:00
module.exports = TextureTintPipeline;