diff --git "a/assets/player/gl/KNWebGLParticleObjects.js" "b/assets/player/gl/KNWebGLParticleObjects.js" --- "a/assets/player/gl/KNWebGLParticleObjects.js" +++ "b/assets/player/gl/KNWebGLParticleObjects.js" @@ -1,3 +1,1482 @@ -version https://git-lfs.github.com/spec/v1 -oid sha256:5bd1a1ef5a329c3eda73b56e103728b9dfaafe54bdbeda18c86b9412c99002ea -size 163580 +/* + * 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; + + // 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; + + 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.a += a; + } + } + } + } + + 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); + + 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; + } +});