/* * KNWebGLParticleObjects.js * Keynote HTML Player * * Created by Tungwei Cheng * Copyright (c) 2016-2018 Apple Inc. All rights reserved. */ var kNumCameraShakePoints = 10; var kParticleSize = 16; var smokeImage = new Image(); smokeImage.src = ""; var speckImage = new Image(); speckImage.src = ""; var flameImage = new Image(); flameImage.src = ""; var fireworksImage = new Image(); fireworksImage.src = ""; var fireworksCenterBurstImage = new Image(); fireworksCenterBurstImage.src = ""; var shimmerImage = new Image(); shimmerImage.src = ""; var sparkleImage = new Image(); sparkleImage.src = ""; var KNWebGLParticleSystem = Class.create({ initialize: function(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) { this.renderer = renderer; this.program = program; var gl = this.gl = this.renderer.gl; var uniforms = this.uniforms = program.uniforms; var attribs = this.attribs = program.attribs; // enables attribs and uniforms KNWebGLUtil.enableAttribs(gl, program) this.texture = texture; this.objectSize = objectSize; this.slideSize = slideSize; this.duration = duration; this.particleSize = particleSize; this.particleSystemSize = particleSystemSize; this.particleCount = particleSystemSize.width * particleSystemSize.height; this.particlesWide = particleSystemSize.width; this.particlesHigh = particleSystemSize.height; // array to store individual particle visibility within the particle system this.particleSystemVisibilities = []; // set to true by default this.shouldDraw = true; this.percentfinished = 0.0; }, animationWillBeginWithContext: function() { var gl = this.gl; var uniforms = this.uniforms; var attribs = this.attribs; if (uniforms["SpeedMax"] !== undefined) { this._speedMax = this.speedMax(); gl.uniform1f(uniforms["SpeedMax"], this._speedMax); } if (uniforms["RotationMax"] !== undefined) { this._rotationMax = this.rotationMax(); gl.uniform1f(uniforms["RotationMax"], this._rotationMax); } if (uniforms["Duration"] !== undefined) { this._duration = this.duration / 1000; gl.uniform1f(uniforms["Duration"], this._duration); } var particleSize = this.particleSize; var objectSize = this.objectSize; var objectBoundsRect = CGRectMake(0, 0, objectSize.width, objectSize.height); var texWidth = objectSize.width; var texHeight = objectSize.height; var pixels; if (this.pixels) { // create framebuffer to read gl texture var fb = gl.createFramebuffer(); pixels = new Uint8Array(texWidth * texHeight * 4); // bind the framebuffer for reading pixels gl.bindFramebuffer(gl.FRAMEBUFFER, fb); gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0); if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) == gl.FRAMEBUFFER_COMPLETE) { gl.readPixels(0, 0, texWidth, texHeight, gl.RGBA, gl.UNSIGNED_BYTE, pixels); } gl.bindFramebuffer(gl.FRAMEBUFFER, null); } var isLookingAtObjectTexture = attribs["TexCoord"] !== undefined; gl.useProgram(this.program.shaderProgram); var attributeBuffers = this.attributeBuffers = {}; attributeBuffers["Position"] = []; attributeBuffers["Center"] = []; attributeBuffers["TexCoord"] = []; attributeBuffers["ParticleTexCoord"] = []; attributeBuffers["Speed"] = []; attributeBuffers["Rotation"] = []; attributeBuffers["Scale"] = []; attributeBuffers["LifeSpan"] = []; attributeBuffers["Color"] = []; var numberOfVerticesPerParticle = this.numberOfVerticesPerParticle; for (var y = 0, particlesHigh = this.particlesHigh; y < particlesHigh; y++) { for (var x = 0, particlesWide = this.particlesWide; x < particlesWide; x++) { var indexPoint = WebGraphics.makePoint(x, y); var index = this.particlesWide * indexPoint.y + indexPoint.x; var vertIndex = index * numberOfVerticesPerParticle; var particleRect = CGRectMake(x * particleSize.width, y * particleSize.height, particleSize.width, particleSize.height); if (isLookingAtObjectTexture && (x === particlesWide - 1 || y === particlesHigh - 1)) { // Make sure we clip last row/column to actual bounds of object particleRect = CGRectIntersection(particleRect, objectBoundsRect); } var pixelParticleRect = particleRect; var visible = true; if (this.pixels) { var bytesPerRow = texWidth * 4; // add up color values in region var addingColor = WebGraphics.makePoint4D(0, 0, 0, 0); var attenuationCounter = 0; var maxXX = pixelParticleRect.origin.x + pixelParticleRect.size.width; var maxYY = pixelParticleRect.origin.y + pixelParticleRect.size.height; for (var yy = pixelParticleRect.origin.y; yy < maxYY; ++yy) { for (var xx = pixelParticleRect.origin.x; xx < maxXX; ++xx) { if (xx < texWidth && yy < texHeight) { // find pixel index from top because the object is flipped var index = (texHeight - yy - 1) * texWidth * 4 + xx * 4; // Make sure we're within the OpenGL texture bounds // (our total size can be a little bigger than the actual texture) if (attribs["Color"] !== undefined) { // attenuate color value by distance from center var attenuation = 1.0; if (pixelParticleRect.size.width > 1) { var xPos = 2 * (xx - pixelParticleRect.origin.x) / pixelParticleRect.size.width - 1; var yPos = 2 * (yy - pixelParticleRect.origin.y) / pixelParticleRect.size.height - 1; attenuation = (1 - xPos * xPos) + (1 - yPos * yPos); } var r = pixels[index]; var g = pixels[index + 1]; var b = pixels[index + 2]; var a = pixels[index + 3]; var thisPixel = WebGraphics.makePoint4D(r/255, g/255, b/255, a/255); if (thisPixel.w !== 0.0) { addingColor.x += attenuation * thisPixel.x / thisPixel.w; addingColor.y += attenuation * thisPixel.y / thisPixel.w; addingColor.z += attenuation * thisPixel.z / thisPixel.w; addingColor.w += attenuation * thisPixel.w; attenuationCounter += attenuation; } } else { // only care about alpha var a = pixels[index + 3] / 255; addingColor.w += a; } } } } // set visibility for the particle visible = addingColor.w > 0; this.particleSystemVisibilities.push(visible); if (attribs["Color"] !== undefined) { // set the color if (attenuationCounter == 0) { // If we never saw any pixels, we're just copying all zeroes into color attenuationCounter = 1; } var invAttenuationCounter = 1.0 / attenuationCounter; var theColor = WebGraphics.multiplyPoint4DByScalar(addingColor, invAttenuationCounter) var attributeBuffersColor = attributeBuffers["Color"]; KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex, attributeBuffersColor); KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex + 1, attributeBuffersColor); KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex + 2, attributeBuffersColor); KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex + 3, attributeBuffersColor); } } if (this.willOverrideStartingPoints) { particleRect.origin = WebGraphics.setOrigin(particleRect.origin, this.startingPointAtIndexPoint(indexPoint)); } var vertices = []; vertices[0] = WebGraphics.makePoint(particleRect.origin.x, particleRect.origin.y); if (numberOfVerticesPerParticle > 1) { vertices[1] = WebGraphics.makePoint(particleRect.origin.x + particleRect.size.width, particleRect.origin.y); vertices[2] = WebGraphics.makePoint(particleRect.origin.x + particleRect.size.width, particleRect.origin.y + particleRect.size.height); vertices[3] = WebGraphics.makePoint(particleRect.origin.x, particleRect.origin.y + particleRect.size.height); } for (var i = 0; i < numberOfVerticesPerParticle; i++) { KNWebGLUtil.setPoint2DAtIndexForAttribute(vertices[i], vertIndex + i, attributeBuffers["Position"]); } if (attribs["Center"] !== undefined) { var midX = particleRect.origin.x + (particleRect.size.width / 2); var midY = particleRect.origin.y + (particleRect.size.height / 2); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(midX, midY), vertIndex, attributeBuffers["Center"]); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(midX, midY), vertIndex + 1, attributeBuffers["Center"]); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(midX, midY), vertIndex + 2, attributeBuffers["Center"]); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(midX, midY), vertIndex + 3, attributeBuffers["Center"]); } if (attribs["TexCoord"] !== undefined) { for (var i = 0; i < numberOfVerticesPerParticle; i++) { var texCoord = WebGraphics.makePoint(vertices[i].x / objectSize.width, vertices[i].y / objectSize.height); KNWebGLUtil.setPoint2DAtIndexForAttribute(texCoord, vertIndex + i, attributeBuffers["TexCoord"]); } } if (attribs["ParticleTexCoord"] !== undefined) { KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(0, 0), vertIndex, attributeBuffers["ParticleTexCoord"]); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(1, 0), vertIndex + 1, attributeBuffers["ParticleTexCoord"]); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(1, 1), vertIndex + 2, attributeBuffers["ParticleTexCoord"]); KNWebGLUtil.setPoint2DAtIndexForAttribute( WebGraphics.makePoint(0, 1), vertIndex + 3, attributeBuffers["ParticleTexCoord"]); } if (attribs["Speed"] !== undefined) { var speed = this.speedAtIndexPoint(indexPoint); KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex, attributeBuffers["Speed"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex + 1, attributeBuffers["Speed"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex + 2, attributeBuffers["Speed"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex + 3, attributeBuffers["Speed"]); } if (attribs["Rotation"] !== undefined) { var rotation = this.rotationAtIndexPoint(indexPoint); KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex, attributeBuffers["Rotation"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex + 1, attributeBuffers["Rotation"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex + 2, attributeBuffers["Rotation"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex + 3, attributeBuffers["Rotation"]); } if (attribs["Scale"] !== undefined) { var scale = this.scaleAtIndexPoint(indexPoint); KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex, attributeBuffers["Scale"]); KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex + 1, attributeBuffers["Scale"]); KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex + 2, attributeBuffers["Scale"]); KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex + 3, attributeBuffers["Scale"]); } if (attribs["LifeSpan"] !== undefined) { var lifeSpan = this.lifeSpanAtIndexPoint(indexPoint); KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex, attributeBuffers["LifeSpan"]); KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex + 1, attributeBuffers["LifeSpan"]); KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex + 2, attributeBuffers["LifeSpan"]); KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex + 3, attributeBuffers["LifeSpan"]); } if (attribs["Color"] !== undefined && this.willOverrideColors) { var color = this.colorAtIndexPoint(indexPoint); KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex, attributeBuffers["Color"]); KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex + 1, attributeBuffers["Color"]); KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex + 2, attributeBuffers["Color"]); KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex + 3, attributeBuffers["Color"]); } } } var indexCounter = 0; var elementArray = this.elementArray = []; for (var i = 0; i < this.particleCount; i++) { elementArray[indexCounter++] = 4 * i + 0; elementArray[indexCounter++] = 4 * i + 1; elementArray[indexCounter++] = 4 * i + 2; // second triangle elementArray[indexCounter++] = 4 * i + 0; elementArray[indexCounter++] = 4 * i + 2; elementArray[indexCounter++] = 4 * i + 3; } this.buffer = {}; KNWebGLUtil.bindAllAvailableAttributesToBuffers(gl, attribs, attributeBuffers, { "LifeSpan": 2, "Scale": 1, "Rotation": 3, "Speed": 3, "ParticleTexCoord": 2, "TexCoord": 2, "Center": 2, "Position": 2, "Color": 4 }, this.buffer, gl.DYNAMIC_DRAW); this.elementArrayBuffer = gl.createBuffer(); gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.elementArrayBuffer); gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(elementArray), gl.DYNAMIC_DRAW); if (this.uniforms["Texture"]) { gl.uniform1i(this.uniforms["Texture"], 0); } if (this.uniforms["ParticleTexture"]) { gl.uniform1i(this.uniforms["ParticleTexture"], 0); } }, drawFrame: function(percent, opacity) { var gl = this.gl; var program = this.program; var uniforms = this.uniforms; this.percentfinished = percent; gl.useProgram(program.shaderProgram); gl.bindTexture(gl.TEXTURE_2D, this.texture); gl.uniform1f(uniforms["Percent"], percent); gl.uniform1f(uniforms["Opacity"], opacity); KNWebGLUtil.bindAllAvailableAttributesToBuffers(gl, this.attribs, this.attributeBuffers, { "LifeSpan": 2, "Scale": 1, "Rotation": 3, "Speed": 3, "ParticleTexCoord": 2, "TexCoord": 2, "Center": 2, "Position": 2, "Color": 4 }, this.buffer, gl.DYNAMIC_DRAW); gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.elementArrayBuffer); gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(this.elementArray), gl.DYNAMIC_DRAW); this.draw(); }, draw: function() { var gl = this.gl; gl.drawElements(gl.TRIANGLES, this.elementArray.length, gl.UNSIGNED_SHORT, 0); }, setColor: function(color) { var gl = this.gl; gl.useProgram(this.program.shaderProgram); gl.uniform4fv(this.uniforms["Color"], color); }, setMVPMatrix: function(MVPMatrix) { var gl = this.gl; gl.useProgram(this.program.shaderProgram); gl.uniformMatrix4fv(this.uniforms["MVPMatrix"], false, MVPMatrix); }, indexFromPoint: function(point) { return this.particlesWide * point.y + point.x; }, particleSystemSizeWithRequestedNumber: function(requestedNumParticles, objectSize) { var maxNumParticles = 65535 / this.numberOfVerticesPerParticle; requestedNumParticles = Math.min(requestedNumParticles, maxNumParticles); if ((objectSize.width === 0 && objectSize.height === 0) || this.willOverrideStartingPoints || requestedNumParticles === 1) { return WebGraphics.makeSize(requestedNumParticles, 1); } if (requestedNumParticles >= objectSize.width * objectSize.height) { return objectSize; } if (requestedNumParticles < 1) { requestedNumParticles = 1; return WebGraphics.makeSize(requestedNumParticles, 1); } var currentNumParticles = 0, prevNumParticles = 0; var particleSize = Math.round(Math.sqrt(objectSize.width * objectSize.height)); //find best fit for number of particles currentNumParticles = Math.ceil(objectSize.width / particleSize) * Math.ceil(objectSize.height / particleSize); if (currentNumParticles === requestedNumParticles) { return WebGraphics.makeSize(particleSize, particleSize); } if (currentNumParticles < requestedNumParticles) { do { prevNumParticles = currentNumParticles; particleSize--; currentNumParticles = Math.ceil(objectSize.width/particleSize) * Math.ceil(objectSize.height/particleSize); } while (currentNumParticles < requestedNumParticles && particleSize > 2); if (particleSize <= 2.0) { return WebGraphics.makeSize(Math.ceil(objectSize.width / 2), Math.ceil(objectSize.height / 2)); } if (Math.abs(currentNumParticles - requestedNumParticles) < Math.abs(prevNumParticles - requestedNumParticles)) { return WebGraphics.makeSize(Math.ceil(objectSize.width / particleSize), Math.ceil(objectSize.height / particleSize)); } else { return WebGraphics.makeSize(Math.ceil(objectSize.width / (particleSize + 1)), Math.ceil(objectSize.height / (particleSize + 1))); } } else { do { prevNumParticles = currentNumParticles; particleSize++; currentNumParticles = Math.ceil(objectSize.width / particleSize) * Math.ceil(objectSize.height / particleSize); } while (currentNumParticles > requestedNumParticles && particleSize > 2); if (particleSize <= 2.0) { return WebGraphics.makeSize(Math.ceil(objectSize.width / 2), Math.ceil(objectSize.height / 2)); } if (Math.abs(currentNumParticles - requestedNumParticles) < Math.abs(prevNumParticles - requestedNumParticles)) { return WebGraphics.makeSize(Math.ceil(objectSize.width / particleSize), Math.ceil(objectSize.height / particleSize)); } else { return WebGraphics.makeSize(Math.ceil(objectSize.width / (particleSize + 1)), Math.ceil(objectSize.height / (particleSize + 1))); } } }, point3DRandomDirection: function() { var u = WebGraphics.randomBetween(-1.0, 1.0); var theta = WebGraphics.randomBetween(0, 2.0 * Math.PI); var prefix = Math.sqrt(1.0 - u * u); var result = WebGraphics.makePoint3D(prefix * Math.cos(theta), prefix * Math.sin(theta), u); return result; } }); var KNWebGLBuildAnvilSmokeSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) { this.willOverrideStartingPoints = true; // number of vertices per particle this.numberOfVerticesPerParticle = 4; $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); }, startingPointAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint); return WebGraphics.makePoint(index / this.particleCount * this.objectSize.width - kParticleSize / 2, kParticleSize / 2); }, speedAtIndexPoint: function(point) { var xDirection = (2 * this.indexFromPoint(point)) / (this.particleCount) -1; xDirection = (xDirection < 0 ? -1 : 1) * Math.sqrt(Math.abs(xDirection)); //we've adjusted these values specifically for WebGL var yDirection = (Math.abs(xDirection) + 0.25) * WebGraphics.randomBetween(-1.0, 0.1); xDirection *= WebGraphics.randomBetween(0.25, 1); var speed = WebGraphics.makePoint3D( this.p_anvilGlobalScale() * 5.6 * xDirection, this.p_anvilGlobalScale() * 5 * -yDirection, 0); return speed; }, rotationAtIndexPoint: function(point) { var xDirection = (2 * this.indexFromPoint(point)) / (this.particleCount) - 1; var thisRotation = xDirection * WebGraphics.randomBetween(0.5, 1) * Math.PI; thisRotation *= this.duration / 1000; return WebGraphics.makePoint3D(0, 0, thisRotation); }, scaleAtIndexPoint: function(indexPoint) { var scale = Math.abs(((2.0 * this.indexFromPoint(indexPoint)) / this.particleCount) - 1.0); scale += 1.25; var randVal = WebGraphics.randomBetween(0, 1); randVal *= randVal * randVal; scale *= WebGraphics.mix(1, 2.5, randVal); scale *= (this.p_anvilGlobalScale() / kParticleSize); return scale; }, lifeSpanAtIndexPoint: function(indexPoint) { return WebGraphics.makePoint(0, (this.duration / 1000) * WebGraphics.randomBetween(0.15, 1)); }, p_anvilGlobalScale: function() { return WebGraphics.mix(1.25, 0.75, this.objectSize.width / this.slideSize.width) * (this.objectSize.width/7); } }); var KNWebGLBuildAnvilSpeckSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) { this.willOverrideStartingPoints = true; // number of vertices per particle this.numberOfVerticesPerParticle = 4; $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); }, startingPointAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint); return WebGraphics.makePoint(index/this.particleCount * this.objectSize.width - kParticleSize/2, kParticleSize/2); }, speedAtIndexPoint: function(point) { var speed = WebGraphics.makePoint3D(0, 0, 0); var index = this.indexFromPoint(point); speed.x = 2 * index / this.particleCount - 1; speed.y = WebGraphics.randomBetween(0.2, 1); if (index % 3 != 1) { speed.z = 0.2; speed.y *= 0.025 * (WebGraphics.randomBetween(0,1) < 0.5 ? -1 : 1); speed.x *= 10; speed.y *= 10; } else { speed.z = 1.0; } speed.x *= this.p_anvilGlobalScale() * 5.25; speed.y *= this.p_anvilGlobalScale() * 4; //only difference is our coordinate system is setup reversed to the desktop, so we need to multiply by -4 instead of 4 return speed; }, rotationAtIndexPoint: function(point) { var xDirection = (2 * this.indexFromPoint(point)) / (this.particleCount) - 1; var thisRotation = xDirection * WebGraphics.randomBetween(0.5, 1) * Math.PI; thisRotation *= this.duration / 1000; return WebGraphics.makePoint3D(0, 0, thisRotation); }, scaleAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint); var scale = WebGraphics.randomBetween(3, 5); if (index % 3 != 1) { scale *= 3; } scale *= (this.p_anvilGlobalScale() / kParticleSize); return scale; }, lifeSpanAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint); var lifeSpan = WebGraphics.makePoint(0, Math.min(1, this.duration/1000) * WebGraphics.randomBetween(0.2, 0.5)); if (index % 3 != 1) { lifeSpan.y *= 10; } return lifeSpan; }, p_anvilGlobalScale: function() { return WebGraphics.mix(1.25, 0.75, this.objectSize.width / this.slideSize.width)*(this.objectSize.width/7); } }); var KNWebGLBuildFlameSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture) { // overwrite starting point this.willOverrideStartingPoints = true; // number of vertices per particle this.numberOfVerticesPerParticle = 4; var width = objectSize.width; var height = objectSize.height; var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height)); var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height)); $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); }, p_setupParticleDataWithTexture: function(textureInfo) { var gl = this.gl; var width = textureInfo.width; var height = textureInfo.height; // create framebuffer to read gl texture var fb = gl.createFramebuffer(); var pixels = new Uint8Array(width * height * 4); // bind the framebuffer for reading pixels gl.bindFramebuffer(gl.FRAMEBUFFER, fb); gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, textureInfo.texture, 0); if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) == gl.FRAMEBUFFER_COMPLETE) { gl.readPixels(0, 0, width, height, gl.RGBA, gl.UNSIGNED_BYTE, pixels); } var minPoint = { "x": width, "y": height }; var maxPoint = { "x": 0, "y": 0 }; // Want to start the flames at the alpha-bottom of the image, not the absolute bottom var bottomRow = []; var bottomRowCount = width; var actualSize = {}; for (var i = 0; i < width * 4; i += 4) { var x = i / 4; var minY = Number.MAX_VALUE; var maxY = 0; var hasNonTransparentPixels = false; var thisPoint = { "x": x / width, "y": -1 }; // search from top because the object is flipped for (var j = height - 1; j >= 0; j--) { var alphaIndex = x * 4 + j * (width) * 4 + 3; var y = height - j; if (pixels[alphaIndex]/255 > 0.1) { minY = Math.min(y, minY); maxY = Math.max(y, maxY); hasNonTransparentPixels = true; } } if (hasNonTransparentPixels) { thisPoint.y = maxY / height; minPoint.x = Math.min(minPoint.x, x); maxPoint.x = Math.max(maxPoint.x, x); minPoint.y = Math.min(minPoint.y, minY); maxPoint.y = Math.max(maxPoint.y, maxY); } bottomRow[x] = thisPoint; } actualSize = { width: maxPoint.x - minPoint.x, height: maxPoint.y - minPoint.y }; this._actualSize = actualSize; this._bottomRow = bottomRow; this._bottomRowCount = bottomRowCount; // unbind the framebuffer gl.bindFramebuffer(gl.FRAMEBUFFER, null); this.animationWillBeginWithContext(); var positionBuffer = this.attributeBuffers["Position"]; var centerBuffer = this.attributeBuffers["Center"]; // adjust vertices according to starting scale var minSide = this._actualSize.height / 2.0; var numberOfVerticesPerParticle = this.numberOfVerticesPerParticle; for (var index = 0, length = this.particleCount; index < length; index++) { var vertIndex = index * numberOfVerticesPerParticle; var origin = { "x": positionBuffer[vertIndex*2], "y": positionBuffer[vertIndex*2 + 1] }; origin.x -= minSide / 2; origin.y -= minSide / 2; var newRect = WebGraphics.makeRect(origin.x, origin.y, minSide, minSide); var center = WebGraphics.makePoint(origin.x + minSide/2, origin.y + minSide/2); KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x, newRect.y), vertIndex, positionBuffer); KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x + newRect.width, newRect.y), vertIndex + 1, positionBuffer); KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x + newRect.width, newRect.y + newRect.height), vertIndex + 2, positionBuffer); KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x, newRect.y + newRect.height), vertIndex + 3, positionBuffer); for (var i = 0; i < 4; i++) { KNWebGLUtil.setPoint2DAtIndexForAttribute(center, vertIndex + i, centerBuffer); } } }, startingPointAtIndexPoint: function(indexPoint) { var x = 0; var y = 0; var foundY = false; for (var i = 0; i < this._bottomRowCount; i++) { // at least one row has y value not equal to -1 if (this._bottomRow[i].y !== -1) { foundY = true; break; } } if (this._bottomRow && foundY) { do { var index = WebGraphics.randomBetween(0, this._bottomRowCount - 1); index = Math.round(index); x = this._bottomRow[index].x; y = this._bottomRow[index].y; } while (y === -1); } var positionPoint = { "x": x * this.objectSize.width, "y": y * this.objectSize.height }; return positionPoint; }, speedAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint) * 4; var yPos = this.attributeBuffers["Position"][index*2 + 1]; yPos = Math.min(yPos, this._actualSize.height); var maxUp = (this._actualSize.height * 0.2 + 0.9 * yPos) * (1.0 + WebGraphics.randomBetween(0.0, 0.1)); var result = WebGraphics.makePoint3D(0, maxUp, 0); result = WebGraphics.multiplyPoint3DByScalar(result, 1.0 / this.speedMax()); return result; }, speedMax: function() { return this._actualSize.height * 1.1 * 1.1; }, lifeSpanAtIndexPoint: function(indexPoint) { // CONSTANTS var maxParticleLife = Math.min(1.0, 1.0 / Math.max(2, this._duration)); var thisParticleLife = maxParticleLife * WebGraphics.randomBetween(0.8, 1.0); // more at beginning var time = this.indexFromPoint(indexPoint) / this.particleCount; time = time * time * 0.25 + time * 0.75; time *= 1.0 - thisParticleLife; var lifeSpan = WebGraphics.makePoint(time, thisParticleLife); return lifeSpan; }, rotationAtIndexPoint: function(indexPoint) { var rotation = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1.0, 1.0), 0, WebGraphics.randomBetween(-1.0, 1.0)); return rotation; }, rotationMax: function() { return Math.PI * 2; } }); var KNWebGLBuildConfettiSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture) { this.willOverrideStartingPoints = false; // number of vertices per particle this.numberOfVerticesPerParticle = 4; var width = objectSize.width; var height = objectSize.height; var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height)); var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height)); $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); }, startingPointAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint); return WebGraphics.makePoint(index / this.particleCount * this.objectSize.width - kParticleSize / 2, kParticleSize / 2); }, speedAtIndexPoint: function(point) { var speedMax = 0.025; var speedRandMax = speedMax * 20; var speedAdjust = this.objectSize.width / this.slideSize.width * this.objectSize.height / this.slideSize.height; speedAdjust = Math.sqrt(speedAdjust); speedAdjust = 1.0 - speedAdjust * 0.75; speedMax *= speedAdjust; speedRandMax *= speedAdjust; var minSide = Math.min(this.objectSize.height, this.objectSize.width); var theRandSpeed = minSide * speedRandMax; var theSpeed = minSide * speedMax; var maxRadiusSquared = (this.particleSystemSize.width * this.particleSystemSize.width + this.particleSystemSize.height * this.particleSystemSize.height) / 4.0; var randSpeed = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1)); randSpeed.z = -Math.abs(randSpeed.z); randSpeed = WebGraphics.multiplyPoint3DByScalar(randSpeed, theRandSpeed); var vector = WebGraphics.makePoint3D(point.x - this.particlesWide / 2.0, point.y - this.particlesHigh / 2.0, 0); var radiusSquared = (vector.x * vector.x + vector.y * vector.y); vector.z = Math.sqrt(maxRadiusSquared - radiusSquared); var speed = WebGraphics.multiplyPoint3DByScalar(vector, theSpeed); speed = WebGraphics.addPoint3DToPoint3D(speed, randSpeed); return speed; }, rotationAtIndexPoint: function(point) { var rotationMax = 8.0 * Math.PI; var rotation = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1)); return WebGraphics.multiplyPoint3DByScalar(rotation, rotationMax); }, scaleAtIndexPoint: function(indexPoint) { return 1.0; } }); var KNWebGLBuildDiffuseSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture, l2r) { this.willOverrideStartingPoints = false; // number of vertices per particle this.numberOfVerticesPerParticle = 4; // set direction this.l2r = l2r; var width = objectSize.width; var height = objectSize.height; var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height)); var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height)); $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); }, speedAtIndexPoint: function(indexPoint) { var l2r = this.l2r; //more random towards center var randDirection = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1)); var vertical = 2 * indexPoint.y / this.particleSystemSize.height - 1.0; randDirection = WebGraphics.multiplyPoint3DByScalar(randDirection, (1.1 - vertical * vertical) * 8); //actual direction based on position of fragment height var startDirection = WebGraphics.makePoint3D((2 - Math.abs(vertical)) * (l2r ? -1: 1), vertical * 2, 0); startDirection = WebGraphics.multiplyPoint3DByScalar(startDirection, 5.0); var theDirection = WebGraphics.addPoint3DToPoint3D(startDirection, randDirection); theDirection = WebGraphics.point3DNormalize(theDirection); return theDirection; }, speedMax: function() { var speed = this.objectSize.height * 1.5 / Math.sqrt(this.duration / 1000); return speed; }, rotationAtIndexPoint: function(indexPoint) { var rotation = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), 0); return rotation; }, rotationMax: function() { var rotationMax = 8.0 * Math.PI; return rotationMax; }, lifeSpanAtIndexPoint: function(indexPoint) { var l2r = this.l2r; var width = this.particleSystemSize.width; var maxParticleLife = WebGraphics.clamp(0.8 / (this.duration / 1000), 0.1, 0.9); var time = (l2r ? (width - indexPoint.x): indexPoint.x) / width; time *= (1 - maxParticleLife); var lifeSpan = WebGraphics.makePoint(time, maxParticleLife); return lifeSpan; } }); var KNWebGLBuildFireworksSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) { // override starting point this.willOverrideStartingPoints = true; // override colors this.willOverrideColors = true; // number of vertices per particle this.numberOfVerticesPerParticle = 4; $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); }, setupWithTexture: function(textureInfo) { var gl = this.gl; var width = textureInfo.width; var height = textureInfo.height; var val1 = -Math.min(this.objectSize.height, this.slideSize.height / 10); var val2 = WebGraphics.randomBetween(val1, this.objectSize.height) this._startingPoint = WebGraphics.makePoint(this.fireworkStartingPositionX * this.objectSize.width, val2); // Initialize random color var startColor = WebGraphics.colorWithHSBA(WebGraphics.randomBetween(0, 1), 1, 1, 1); this._startingColorRGB = WebGraphics.makePoint3D(startColor.red, startColor.green, startColor.blue); this.animationWillBeginWithContext(); }, startingPointAtIndexPoint: function(indexPoint) { return this._startingPoint; }, colorAtIndexPoint: function(indexPoint) { var randomColor = this.point3DRandomDirection(); var randomColorResult = WebGraphics.multiplyPoint3DByScalar(randomColor, this.colorRandomness); var color = WebGraphics.addPoint3DToPoint3D(this._startingColorRGB, randomColorResult); color.x = WebGraphics.clamp(color.x, 0, 1); color.y = WebGraphics.clamp(color.y, 0, 1); color.z = WebGraphics.clamp(color.z, 0, 1); var result = { "x": color.x, "y": color.y, "z": color.z, "w": 1 }; return result; }, speedAtIndexPoint: function(indexPoint) { var speed = this.point3DRandomDirection(); var randomSpeed = WebGraphics.randomBetween(0.8, 1.0); var result = WebGraphics.multiplyPoint3DByScalar(speed, randomSpeed); return result; }, speedMax: function() { return this.maxDistance; }, scaleAtIndexPoint: function(indexPoint) { var scale = WebGraphics.randomBetween(this.randomParticleSizeMinMax.width, this.randomParticleSizeMinMax.height); return scale; }, lifeSpanAtIndexPoint: function(indexPoint) { var lifeSpan = WebGraphics.makePoint(0, WebGraphics.randomBetween(this.lifeSpanMinDuration, 1.0)); return lifeSpan; } }); var KNWebGLBuildShimmerSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) { $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); }, speedMax: function() { var sqrtArea = Math.sqrt(this.objectSize.width * this.objectSize.height); var result = sqrtArea * 0.075; return result; }, speedAtIndexPoint: function(indexPoint) { var speed = this.point3DRandomDirection(); speed.z = 0; return speed; } }); var KNWebGLBuildShimmerObjectSystem = Class.create(KNWebGLBuildShimmerSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture, direction) { // override starting point this.willOverrideStartingPoints = false; // override colors this.willOverrideColors = false; // number of vertices per particle this.numberOfVerticesPerParticle = 4; // require setting color pixels this.pixels = true; // set direction this.direction = direction; var width = objectSize.width; var height = objectSize.height; var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height)); var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height)); // call KNWebGLBuildShimmerSystem initialize method $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture, direction); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); }, drawGLSLWithPercent: function(percent, particleOpacity, rotation, clockwise) { var delayedPercent = Math.max(0.0, percent * 1.1 - 0.1); var origTexPercent = 1.0 - Math.min(1.0, TSDMixFloats(5.0 * percent, percent * percent, percent)); // Rotation Matrix var angle = delayedPercent * (clockwise ? 1 : -1) * 2; var rotMatrix = CGAffineTransformMakeRotation(angle); var mat3 = WebGraphics.makeMat3WithAffineTransform(rotMatrix); // set mat3 uniform for RotationMatrix this.gl.uniformMatrix3fv(this.uniforms["RotationMatrix"], false, mat3); // draw GLSL with percent this.drawFrame(delayedPercent, particleOpacity * origTexPercent); } }); var KNWebGLBuildShimmerParticleSystem = Class.create(KNWebGLBuildShimmerSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, objectSystem, texture, direction) { // override starting point this.willOverrideStartingPoints = true; // override colors this.willOverrideColors = true; // number of vertices per particle this.numberOfVerticesPerParticle = 4; // set direction this.direction = direction; // object system particle count this.objectSystem = objectSystem; // object system vertex count this.objectSystemVertexCount = objectSystem.particleCount * 4; var kNumHullPoints = this.kNumHullPoints = 20 //beginning of width of bottom spire on x-axis var kSpireMinW = 0.48828125; //beginning of height of bottom spire on y-axis var kSpireMinH = 0.02; //top of height of bottom spire on y-axis var kSpireMaxH = 0.3; this.p_particleHullArray = [ // center quad { x: kSpireMaxH, y: kSpireMaxH }, { x: 1 - kSpireMaxH, y: kSpireMaxH }, { x: 1 - kSpireMaxH, y: 1 - kSpireMaxH }, { x: kSpireMaxH, y: 1 - kSpireMaxH }, // top quad { x: kSpireMinW, y: 1 - kSpireMaxH }, { x: 1 - kSpireMinW, y: 1 - kSpireMaxH }, { x: 1 - kSpireMinW, y: 1 - kSpireMinH }, { x: kSpireMinW, y: 1 - kSpireMinH }, // right quad { x: 1 - kSpireMaxH, y: 1 - kSpireMinW }, { x: 1 - kSpireMinH, y: 1 - kSpireMinW }, { x: 1 - kSpireMinH, y: kSpireMinW }, { x: 1 - kSpireMaxH, y: kSpireMinW }, // bottom quad { x: kSpireMinW, y: kSpireMinH }, { x: 1 - kSpireMinW, y: kSpireMinH }, { x: 1 - kSpireMinW, y: kSpireMaxH }, { x: kSpireMinW, y: kSpireMaxH }, // left quad { x: kSpireMinH, y: kSpireMinW }, { x: kSpireMaxH, y: kSpireMinW }, { x: kSpireMaxH, y: 1 - kSpireMinW }, { x: kSpireMinH, y: 1 - kSpireMinW }, ]; // call KNWebGLBuildShimmerSystem initialize method $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture, direction); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); // update vertex data this.p_setupVertexData(); }, p_setupVertexData: function() { var texRect = CGRectMake(0, 0, 1, 1); var numVertsPP = this.numberOfVerticesPerParticle; var oldParticleCount = this.particleCount; var newParticleCount = this.particleCount * 5; // objtain a reference of current data buffer var oldAttributeBuffers = this.attributeBuffers; // Set up fresh, new data buffer attributes! var newAttributeBuffers = {}; newAttributeBuffers["Position"] = []; newAttributeBuffers["Center"] = []; newAttributeBuffers["TexCoord"] = []; newAttributeBuffers["ParticleTexCoord"] = []; newAttributeBuffers["Speed"] = []; newAttributeBuffers["Rotation"] = []; newAttributeBuffers["Scale"] = []; newAttributeBuffers["LifeSpan"] = []; newAttributeBuffers["Color"] = []; // Update new buffer with old buffer values for (var i = 0; i < oldParticleCount; ++i) { var oldVertIndex = i * 4; var thisScale = this.attributeBuffers["Scale"][oldVertIndex]; var thisSpeed = WebGraphics.makePoint3D( this.attributeBuffers["Speed"][oldVertIndex * 3], this.attributeBuffers["Speed"][oldVertIndex * 3 + 1], this.attributeBuffers["Speed"][oldVertIndex * 3 + 2] ); var thisColor = WebGraphics.makePoint4D( this.attributeBuffers["Color"][oldVertIndex * 4], this.attributeBuffers["Color"][oldVertIndex * 4 + 1], this.attributeBuffers["Color"][oldVertIndex * 4 + 2], this.attributeBuffers["Color"][oldVertIndex * 4 + 3] ); var vertMin = WebGraphics.makePoint( this.attributeBuffers["Position"][oldVertIndex * 2], this.attributeBuffers["Position"][oldVertIndex * 2 + 1] ); var vertMax = WebGraphics.makePoint( this.attributeBuffers["Position"][oldVertIndex * 2 + 4], this.attributeBuffers["Position"][oldVertIndex * 2 + 5] ); var vertRect = TSDRectWithPoints(vertMin, vertMax); // set vertRect to empty if the particle from the object system is not visible if (this.objectSystem.particleSystemVisibilities[i] === false) { vertRect = TSDRectWithPoints(WebGraphics.makePoint(0, 0), WebGraphics.makePoint(0, 0)); } var center = WebGraphics.makePoint( this.attributeBuffers["Center"][oldVertIndex * 2], this.attributeBuffers["Center"][oldVertIndex * 2 + 1] ); var lifeSpan = WebGraphics.makePoint( this.attributeBuffers["LifeSpan"][oldVertIndex * 2], this.attributeBuffers["LifeSpan"][oldVertIndex * 2 + 1] ); var p_particleHullArray = this.p_particleHullArray; var kNumHullPoints = this.kNumHullPoints; for (var v = 0; v < kNumHullPoints; ++v) { var newVertIndex = i * kNumHullPoints + v; var thisHullPoint = p_particleHullArray[v]; var newVertexPoint = this.p_hullPoint(thisHullPoint, vertRect); KNWebGLUtil.setPoint2DAtIndexForAttribute(newVertexPoint, newVertIndex, newAttributeBuffers["Position"]); var newTexCoord = this.p_hullPoint(thisHullPoint, texRect); KNWebGLUtil.setPoint2DAtIndexForAttribute(newTexCoord, newVertIndex, newAttributeBuffers["ParticleTexCoord"]); KNWebGLUtil.setPoint2DAtIndexForAttribute(center, newVertIndex, newAttributeBuffers["Center"]); KNWebGLUtil.setPoint4DAtIndexForAttribute(thisColor, newVertIndex, newAttributeBuffers["Color"]); KNWebGLUtil.setPoint3DAtIndexForAttribute(thisSpeed, newVertIndex, newAttributeBuffers["Speed"]); KNWebGLUtil.setFloatAtIndexForAttribute(thisScale, newVertIndex, newAttributeBuffers["Scale"]); KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, newVertIndex, newAttributeBuffers["LifeSpan"]); } } // update data buffer with new data buffer this.attributeBuffers = newAttributeBuffers; // update element array var elementArray = this.elementArray = []; var indexCounter = 0; for (var i = 0; i < newParticleCount; i++) { elementArray[indexCounter++] = 4 * i + 0; elementArray[indexCounter++] = 4 * i + 1; elementArray[indexCounter++] = 4 * i + 2; // second triangle elementArray[indexCounter++] = 4 * i + 0; elementArray[indexCounter++] = 4 * i + 2; elementArray[indexCounter++] = 4 * i + 3; } }, p_hullPoint: function(hullPoint, vertexRect) { var point = WebGraphics.makePoint(vertexRect.origin.x + vertexRect.size.width * hullPoint.x, vertexRect.origin.y + vertexRect.size.height * hullPoint.y); return point; }, startingPointAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint) * 4; var objectSystemVertexCount = this.objectSystemVertexCount; var attributeBuffers = this.objectSystem.attributeBuffers["Position"]; if (index < objectSystemVertexCount) { // Copy value from object particle system so this sparkle exactly matches up with an existing object particle var result = WebGraphics.makePoint( attributeBuffers[index * 2], attributeBuffers[index * 2 + 1] ); return result; } // else, it's an extra sparkle at the end, in a random location! var halfWidth = this.objectSize.width / 2; var halfHeight = this.objectSize.height / 2; var midPoint = CGPointMake(halfWidth, halfHeight); // attenuate points towards the middle var r = WebGraphics.randomBetween(0, 1); var angle = WebGraphics.doubleBetween(0, 2.0 * Math.PI); var randLoc = CGPointMake(midPoint.x + halfWidth * r * Math.cos(angle), midPoint.y + halfHeight * r * Math.sin(angle)); return randLoc; }, speedAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint) * 4; var objectSystemVertexCount = this.objectSystemVertexCount; var attributeBuffers = this.objectSystem.attributeBuffers["Speed"]; if (index < objectSystemVertexCount) { // Copy value from object particle system so this sparkle exactly matches up with an existing object particle var result = WebGraphics.makePoint3D( attributeBuffers[index * 3], attributeBuffers[index * 3 + 1], attributeBuffers[index * 3 + 2] ); return result; } // else, it's an extra sparkle at the end; stay in place var speed = WebGraphics.makePoint3D(0, 0, 0); return speed; }, scaleAtIndexPoint: function(indexPoint) { var minScale = 1.0; var maxScale = 25.0; var randNum = WebGraphics.randomBetween(0, 1); // Most particles will be smaller var result = TSUMix(minScale, maxScale, randNum * randNum); return result; }, lifeSpanAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint) * 4; var objectSystemVertexCount = this.objectSystemVertexCount; if (index < objectSystemVertexCount) { // This is an existing object return WebGraphics.makePoint(0, 1); } // else, it's an extra sparkle at the end, with a random life span! // 2 second or 90% of total duration, whichever is shorter var lifeDuration = Math.min(2.0 / (this.duration / 1000), 0.9); var lifeStart = TSDMixFloats(0.01, 0.99, 1.0 - TSUReverseSquare(WebGraphics.randomBetween(0, 1))); // Since percent is non-uniform, make duration less at later part of anim lifeDuration *= TSUReverseSquare(lifeStart); // Make sure lifespan ends before animation completes lifeStart = Math.min(lifeStart, 0.99 - lifeDuration); var result = WebGraphics.makePoint(lifeStart, lifeDuration); return result; }, colorAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint) * 4; var objectSystemVertexCount = this.objectSystemVertexCount; var attributeBuffers = this.objectSystem.attributeBuffers["Color"]; if (index < objectSystemVertexCount) { // Copy value from object particle system so this sparkle exactly matches up with an existing object particle var result = WebGraphics.makePoint4D( attributeBuffers[index * 4], attributeBuffers[index * 4 + 1], attributeBuffers[index * 4 + 2], attributeBuffers[index * 4 + 3] ); return result; } // else it's an extra sparkle at the end // white var result = WebGraphics.makePoint4D(1, 1, 1, 1); return result; }, drawGLSLWithPercent: function(percent, opacity, rotation, clockwise) { // Rotation Matrix var angle = percent * (clockwise ? 1 : -1) * 2; var rotMatrix = CGAffineTransformMakeRotation(angle); var mat3 = WebGraphics.makeMat3WithAffineTransform(rotMatrix); // set mat3 uniform for RotationMatrix this.gl.uniformMatrix3fv(this.uniforms[kShimmerUniformRotationMatrix], false, mat3); // Particle Scale Percent var invPercent = 1.0 - percent; var powPercent = Math.pow(invPercent, 15.0); var particleScalePercent = TSUMix(invPercent * invPercent, 25. * percent, powPercent); this.gl.uniform1f(this.uniforms[kShimmerUniformParticleScalePercent], particleScalePercent); // Draw object this.drawFrame(percent, opacity); } }); var KNWebGLBuildSparkleSystem = Class.create(KNWebGLParticleSystem, { initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture, direction) { this.willOverrideStartingPoints = true; // number of vertices per particle this.numberOfVerticesPerParticle = 4; // set direction this.direction = direction; $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture); // cache p_globalScale for faster access this.cachedGlobalScale = this.p_globalScale(); this.setupWithTexture(); }, setupWithTexture: function() { this.animationWillBeginWithContext(); }, p_globalScale: function() { var objectSize = this.objectSize; var slideSize = this.slideSize; var minSide = Math.min(objectSize.width, objectSize.height); var minSideRatio = minSide / (Math.min(slideSize.width, slideSize.height)); minSide = minSide / Math.sqrt(Math.sqrt(minSideRatio)) * 0.25; return minSide; }, startingPointAtIndexPoint: function(indexPoint) { // CONSTANTS var maxOffset = 0.1 / (this.duration / 1000); var yAxis = (this.direction == KNDirection.kKNDirectionTopToBottom || this.direction == KNDirection.kKNDirectionBottomToTop); var reverse = (this.direction == KNDirection.kKNDirectionRightToLeft || this.direction == KNDirection.kKNDirectionTopToBottom); var index = this.indexFromPoint(indexPoint); var x = index / this.particleCount; // position along line x = reverse ? 1 - x : x; var axis1 = x + maxOffset * WebGraphics.doubleBetween(-1.0, 1.0); var axis2 = Math.random(); // skew position towards center vertically axis2 = 2. * axis2 - 1.; axis2 *= Math.abs(axis2); axis2 = (axis2 + 1.) / 2.; var newX = yAxis ? axis2 : axis1; var newY = yAxis ? axis1 : axis2; var position = WebGraphics.makePoint( newX * this.objectSize.width - this.particleSize.width / 2.0, newY * this.objectSize.height - this.particleSize.height / 2.0 ); return position; }, speedAtIndexPoint: function(point) { var speed = this.point3DRandomDirection(); // reduce z speed speed.z *= 0.01; var randomMultiplier = Math.random(); var result = WebGraphics.multiplyPoint3DByScalar(speed, randomMultiplier); return result; }, speedMax: function() { var minSide = Math.min(this.objectSize.width, this.objectSize.height); var minSideRatio = minSide / (Math.min(this.slideSize.width, this.slideSize.height)); minSide = minSide / Math.pow(minSideRatio, 0.667) * 0.25 * 1.5; return minSide; }, scaleAtIndexPoint: function(indexPoint) { var minSide = this.cachedGlobalScale; var result = minSide / this.particleSize.width; return result; }, lifeSpanAtIndexPoint: function(indexPoint) { var index = this.indexFromPoint(indexPoint); var timeStart = index / this.particleCount; var timeDuration = KNSparkleMaxParticleLife / Math.max(0.75, this.duration/1000); timeStart *= 1 - timeDuration; var lifeSpan = WebGraphics.makePoint(timeStart, timeDuration); return lifeSpan; } });