25개 이상의 토픽을 선택하실 수 없습니다. Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1496 lines
160KB

  1. /*
  2. * KNWebGLParticleObjects.js
  3. * Keynote HTML Player
  4. *
  5. * Created by Tungwei Cheng
  6. * Copyright (c) 2016-2018 Apple Inc. All rights reserved.
  7. */
  8. var kNumCameraShakePoints = 10;
  9. var kParticleSize = 16;
  10. var smokeImage = new Image();
  11. smokeImage.src = "";
  12. var speckImage = new Image();
  13. speckImage.src = "";
  14. var flameImage = new Image();
  15. flameImage.src = "";
  16. var fireworksImage = new Image();
  17. fireworksImage.src = "";
  18. var fireworksCenterBurstImage = new Image();
  19. fireworksCenterBurstImage.src = "";
  20. var shimmerImage = new Image();
  21. shimmerImage.src = "";
  22. var sparkleImage = new Image();
  23. sparkleImage.src = "";
  24. var KNWebGLParticleSystem = Class.create({
  25. initialize: function(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) {
  26. this.renderer = renderer;
  27. this.program = program;
  28. var gl = this.gl = this.renderer.gl;
  29. var uniforms = this.uniforms = program.uniforms;
  30. var attribs = this.attribs = program.attribs;
  31. // enables attribs and uniforms
  32. KNWebGLUtil.enableAttribs(gl, program)
  33. this.texture = texture;
  34. this.objectSize = objectSize;
  35. this.slideSize = slideSize;
  36. this.duration = duration;
  37. this.particleSize = particleSize;
  38. this.particleSystemSize = particleSystemSize;
  39. this.particleCount = particleSystemSize.width * particleSystemSize.height;
  40. this.particlesWide = particleSystemSize.width;
  41. this.particlesHigh = particleSystemSize.height;
  42. // array to store individual particle visibility within the particle system
  43. this.particleSystemVisibilities = [];
  44. // set to true by default
  45. this.shouldDraw = true;
  46. this.percentfinished = 0.0;
  47. },
  48. animationWillBeginWithContext: function() {
  49. var gl = this.gl;
  50. var uniforms = this.uniforms;
  51. var attribs = this.attribs;
  52. if (uniforms["SpeedMax"] !== undefined) {
  53. this._speedMax = this.speedMax();
  54. gl.uniform1f(uniforms["SpeedMax"], this._speedMax);
  55. }
  56. if (uniforms["RotationMax"] !== undefined) {
  57. this._rotationMax = this.rotationMax();
  58. gl.uniform1f(uniforms["RotationMax"], this._rotationMax);
  59. }
  60. if (uniforms["Duration"] !== undefined) {
  61. this._duration = this.duration / 1000;
  62. gl.uniform1f(uniforms["Duration"], this._duration);
  63. }
  64. var particleSize = this.particleSize;
  65. var objectSize = this.objectSize;
  66. var objectBoundsRect = CGRectMake(0, 0, objectSize.width, objectSize.height);
  67. var texWidth = objectSize.width;
  68. var texHeight = objectSize.height;
  69. var pixels;
  70. if (this.pixels) {
  71. // create framebuffer to read gl texture
  72. var fb = gl.createFramebuffer();
  73. pixels = new Uint8Array(texWidth * texHeight * 4);
  74. // bind the framebuffer for reading pixels
  75. gl.bindFramebuffer(gl.FRAMEBUFFER, fb);
  76. gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0);
  77. if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) == gl.FRAMEBUFFER_COMPLETE) {
  78. gl.readPixels(0, 0, texWidth, texHeight, gl.RGBA, gl.UNSIGNED_BYTE, pixels);
  79. }
  80. gl.bindFramebuffer(gl.FRAMEBUFFER, null);
  81. }
  82. var isLookingAtObjectTexture = attribs["TexCoord"] !== undefined;
  83. gl.useProgram(this.program.shaderProgram);
  84. var attributeBuffers = this.attributeBuffers = {};
  85. attributeBuffers["Position"] = [];
  86. attributeBuffers["Center"] = [];
  87. attributeBuffers["TexCoord"] = [];
  88. attributeBuffers["ParticleTexCoord"] = [];
  89. attributeBuffers["Speed"] = [];
  90. attributeBuffers["Rotation"] = [];
  91. attributeBuffers["Scale"] = [];
  92. attributeBuffers["LifeSpan"] = [];
  93. attributeBuffers["Color"] = [];
  94. var numberOfVerticesPerParticle = this.numberOfVerticesPerParticle;
  95. for (var y = 0, particlesHigh = this.particlesHigh; y < particlesHigh; y++) {
  96. for (var x = 0, particlesWide = this.particlesWide; x < particlesWide; x++) {
  97. var indexPoint = WebGraphics.makePoint(x, y);
  98. var index = this.particlesWide * indexPoint.y + indexPoint.x;
  99. var vertIndex = index * numberOfVerticesPerParticle;
  100. var particleRect = CGRectMake(x * particleSize.width, y * particleSize.height, particleSize.width, particleSize.height);
  101. if (isLookingAtObjectTexture && (x === particlesWide - 1 || y === particlesHigh - 1)) {
  102. // Make sure we clip last row/column to actual bounds of object
  103. particleRect = CGRectIntersection(particleRect, objectBoundsRect);
  104. }
  105. var pixelParticleRect = particleRect;
  106. var visible = true;
  107. if (this.pixels) {
  108. var bytesPerRow = texWidth * 4;
  109. // add up color values in region
  110. var addingColor = WebGraphics.makePoint4D(0, 0, 0, 0);
  111. var attenuationCounter = 0;
  112. var maxXX = pixelParticleRect.origin.x + pixelParticleRect.size.width;
  113. var maxYY = pixelParticleRect.origin.y + pixelParticleRect.size.height;
  114. for (var yy = pixelParticleRect.origin.y; yy < maxYY; ++yy) {
  115. for (var xx = pixelParticleRect.origin.x; xx < maxXX; ++xx) {
  116. if (xx < texWidth && yy < texHeight) {
  117. // find pixel index from top because the object is flipped
  118. var index = (texHeight - yy - 1) * texWidth * 4 + xx * 4;
  119. // Make sure we're within the OpenGL texture bounds
  120. // (our total size can be a little bigger than the actual texture)
  121. if (attribs["Color"] !== undefined) {
  122. // attenuate color value by distance from center
  123. var attenuation = 1.0;
  124. if (pixelParticleRect.size.width > 1) {
  125. var xPos = 2 * (xx - pixelParticleRect.origin.x) / pixelParticleRect.size.width - 1;
  126. var yPos = 2 * (yy - pixelParticleRect.origin.y) / pixelParticleRect.size.height - 1;
  127. attenuation = (1 - xPos * xPos) + (1 - yPos * yPos);
  128. }
  129. var r = pixels[index];
  130. var g = pixels[index + 1];
  131. var b = pixels[index + 2];
  132. var a = pixels[index + 3];
  133. var thisPixel = WebGraphics.makePoint4D(r/255, g/255, b/255, a/255);
  134. if (thisPixel.w !== 0.0) {
  135. addingColor.x += attenuation * thisPixel.x / thisPixel.w;
  136. addingColor.y += attenuation * thisPixel.y / thisPixel.w;
  137. addingColor.z += attenuation * thisPixel.z / thisPixel.w;
  138. addingColor.w += attenuation * thisPixel.w;
  139. attenuationCounter += attenuation;
  140. }
  141. } else {
  142. // only care about alpha
  143. var a = pixels[index + 3] / 255;
  144. addingColor.w += a;
  145. }
  146. }
  147. }
  148. }
  149. // set visibility for the particle
  150. visible = addingColor.w > 0;
  151. this.particleSystemVisibilities.push(visible);
  152. if (attribs["Color"] !== undefined) {
  153. // set the color
  154. if (attenuationCounter == 0) {
  155. // If we never saw any pixels, we're just copying all zeroes into color
  156. attenuationCounter = 1;
  157. }
  158. var invAttenuationCounter = 1.0 / attenuationCounter;
  159. var theColor = WebGraphics.multiplyPoint4DByScalar(addingColor, invAttenuationCounter)
  160. var attributeBuffersColor = attributeBuffers["Color"];
  161. KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex, attributeBuffersColor);
  162. KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex + 1, attributeBuffersColor);
  163. KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex + 2, attributeBuffersColor);
  164. KNWebGLUtil.setPoint4DAtIndexForAttribute(theColor, vertIndex + 3, attributeBuffersColor);
  165. }
  166. }
  167. if (this.willOverrideStartingPoints) {
  168. particleRect.origin = WebGraphics.setOrigin(particleRect.origin, this.startingPointAtIndexPoint(indexPoint));
  169. }
  170. var vertices = [];
  171. vertices[0] = WebGraphics.makePoint(particleRect.origin.x, particleRect.origin.y);
  172. if (numberOfVerticesPerParticle > 1) {
  173. vertices[1] = WebGraphics.makePoint(particleRect.origin.x + particleRect.size.width, particleRect.origin.y);
  174. vertices[2] = WebGraphics.makePoint(particleRect.origin.x + particleRect.size.width, particleRect.origin.y + particleRect.size.height);
  175. vertices[3] = WebGraphics.makePoint(particleRect.origin.x, particleRect.origin.y + particleRect.size.height);
  176. }
  177. for (var i = 0; i < numberOfVerticesPerParticle; i++) {
  178. KNWebGLUtil.setPoint2DAtIndexForAttribute(vertices[i], vertIndex + i, attributeBuffers["Position"]);
  179. }
  180. if (attribs["Center"] !== undefined) {
  181. var midX = particleRect.origin.x + (particleRect.size.width / 2);
  182. var midY = particleRect.origin.y + (particleRect.size.height / 2);
  183. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  184. WebGraphics.makePoint(midX, midY), vertIndex, attributeBuffers["Center"]);
  185. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  186. WebGraphics.makePoint(midX, midY), vertIndex + 1, attributeBuffers["Center"]);
  187. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  188. WebGraphics.makePoint(midX, midY), vertIndex + 2, attributeBuffers["Center"]);
  189. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  190. WebGraphics.makePoint(midX, midY), vertIndex + 3, attributeBuffers["Center"]);
  191. }
  192. if (attribs["TexCoord"] !== undefined) {
  193. for (var i = 0; i < numberOfVerticesPerParticle; i++) {
  194. var texCoord = WebGraphics.makePoint(vertices[i].x / objectSize.width, vertices[i].y / objectSize.height);
  195. KNWebGLUtil.setPoint2DAtIndexForAttribute(texCoord, vertIndex + i, attributeBuffers["TexCoord"]);
  196. }
  197. }
  198. if (attribs["ParticleTexCoord"] !== undefined) {
  199. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  200. WebGraphics.makePoint(0, 0), vertIndex, attributeBuffers["ParticleTexCoord"]);
  201. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  202. WebGraphics.makePoint(1, 0), vertIndex + 1, attributeBuffers["ParticleTexCoord"]);
  203. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  204. WebGraphics.makePoint(1, 1), vertIndex + 2, attributeBuffers["ParticleTexCoord"]);
  205. KNWebGLUtil.setPoint2DAtIndexForAttribute(
  206. WebGraphics.makePoint(0, 1), vertIndex + 3, attributeBuffers["ParticleTexCoord"]);
  207. }
  208. if (attribs["Speed"] !== undefined) {
  209. var speed = this.speedAtIndexPoint(indexPoint);
  210. KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex, attributeBuffers["Speed"]);
  211. KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex + 1, attributeBuffers["Speed"]);
  212. KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex + 2, attributeBuffers["Speed"]);
  213. KNWebGLUtil.setPoint3DAtIndexForAttribute(speed, vertIndex + 3, attributeBuffers["Speed"]);
  214. }
  215. if (attribs["Rotation"] !== undefined) {
  216. var rotation = this.rotationAtIndexPoint(indexPoint);
  217. KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex, attributeBuffers["Rotation"]);
  218. KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex + 1, attributeBuffers["Rotation"]);
  219. KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex + 2, attributeBuffers["Rotation"]);
  220. KNWebGLUtil.setPoint3DAtIndexForAttribute(rotation, vertIndex + 3, attributeBuffers["Rotation"]);
  221. }
  222. if (attribs["Scale"] !== undefined) {
  223. var scale = this.scaleAtIndexPoint(indexPoint);
  224. KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex, attributeBuffers["Scale"]);
  225. KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex + 1, attributeBuffers["Scale"]);
  226. KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex + 2, attributeBuffers["Scale"]);
  227. KNWebGLUtil.setFloatAtIndexForAttribute(scale, vertIndex + 3, attributeBuffers["Scale"]);
  228. }
  229. if (attribs["LifeSpan"] !== undefined) {
  230. var lifeSpan = this.lifeSpanAtIndexPoint(indexPoint);
  231. KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex, attributeBuffers["LifeSpan"]);
  232. KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex + 1, attributeBuffers["LifeSpan"]);
  233. KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex + 2, attributeBuffers["LifeSpan"]);
  234. KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, vertIndex + 3, attributeBuffers["LifeSpan"]);
  235. }
  236. if (attribs["Color"] !== undefined && this.willOverrideColors) {
  237. var color = this.colorAtIndexPoint(indexPoint);
  238. KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex, attributeBuffers["Color"]);
  239. KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex + 1, attributeBuffers["Color"]);
  240. KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex + 2, attributeBuffers["Color"]);
  241. KNWebGLUtil.setPoint4DAtIndexForAttribute(color, vertIndex + 3, attributeBuffers["Color"]);
  242. }
  243. }
  244. }
  245. var indexCounter = 0;
  246. var elementArray = this.elementArray = [];
  247. for (var i = 0; i < this.particleCount; i++) {
  248. elementArray[indexCounter++] = 4 * i + 0;
  249. elementArray[indexCounter++] = 4 * i + 1;
  250. elementArray[indexCounter++] = 4 * i + 2;
  251. // second triangle
  252. elementArray[indexCounter++] = 4 * i + 0;
  253. elementArray[indexCounter++] = 4 * i + 2;
  254. elementArray[indexCounter++] = 4 * i + 3;
  255. }
  256. this.buffer = {};
  257. KNWebGLUtil.bindAllAvailableAttributesToBuffers(gl, attribs, attributeBuffers, {
  258. "LifeSpan": 2,
  259. "Scale": 1,
  260. "Rotation": 3,
  261. "Speed": 3,
  262. "ParticleTexCoord": 2,
  263. "TexCoord": 2,
  264. "Center": 2,
  265. "Position": 2,
  266. "Color": 4
  267. }, this.buffer,
  268. gl.DYNAMIC_DRAW);
  269. this.elementArrayBuffer = gl.createBuffer();
  270. gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.elementArrayBuffer);
  271. gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(elementArray), gl.DYNAMIC_DRAW);
  272. if (this.uniforms["Texture"]) {
  273. gl.uniform1i(this.uniforms["Texture"], 0);
  274. }
  275. if (this.uniforms["ParticleTexture"]) {
  276. gl.uniform1i(this.uniforms["ParticleTexture"], 0);
  277. }
  278. },
  279. drawFrame: function(percent, opacity) {
  280. var gl = this.gl;
  281. var program = this.program;
  282. var uniforms = this.uniforms;
  283. this.percentfinished = percent;
  284. gl.useProgram(program.shaderProgram);
  285. gl.bindTexture(gl.TEXTURE_2D, this.texture);
  286. gl.uniform1f(uniforms["Percent"], percent);
  287. gl.uniform1f(uniforms["Opacity"], opacity);
  288. KNWebGLUtil.bindAllAvailableAttributesToBuffers(gl, this.attribs, this.attributeBuffers, {
  289. "LifeSpan": 2,
  290. "Scale": 1,
  291. "Rotation": 3,
  292. "Speed": 3,
  293. "ParticleTexCoord": 2,
  294. "TexCoord": 2,
  295. "Center": 2,
  296. "Position": 2,
  297. "Color": 4
  298. }, this.buffer,
  299. gl.DYNAMIC_DRAW);
  300. gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.elementArrayBuffer);
  301. gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(this.elementArray), gl.DYNAMIC_DRAW);
  302. this.draw();
  303. },
  304. draw: function() {
  305. var gl = this.gl;
  306. gl.drawElements(gl.TRIANGLES, this.elementArray.length, gl.UNSIGNED_SHORT, 0);
  307. },
  308. setColor: function(color) {
  309. var gl = this.gl;
  310. gl.useProgram(this.program.shaderProgram);
  311. gl.uniform4fv(this.uniforms["Color"], color);
  312. },
  313. setMVPMatrix: function(MVPMatrix) {
  314. var gl = this.gl;
  315. gl.useProgram(this.program.shaderProgram);
  316. gl.uniformMatrix4fv(this.uniforms["MVPMatrix"], false, MVPMatrix);
  317. },
  318. indexFromPoint: function(point) {
  319. return this.particlesWide * point.y + point.x;
  320. },
  321. particleSystemSizeWithRequestedNumber: function(requestedNumParticles, objectSize) {
  322. var maxNumParticles = 65535 / this.numberOfVerticesPerParticle;
  323. requestedNumParticles = Math.min(requestedNumParticles, maxNumParticles);
  324. if ((objectSize.width === 0 && objectSize.height === 0) || this.willOverrideStartingPoints || requestedNumParticles === 1) {
  325. return WebGraphics.makeSize(requestedNumParticles, 1);
  326. }
  327. if (requestedNumParticles >= objectSize.width * objectSize.height) {
  328. return objectSize;
  329. }
  330. if (requestedNumParticles < 1) {
  331. requestedNumParticles = 1;
  332. return WebGraphics.makeSize(requestedNumParticles, 1);
  333. }
  334. var currentNumParticles = 0, prevNumParticles = 0;
  335. var particleSize = Math.round(Math.sqrt(objectSize.width * objectSize.height));
  336. //find best fit for number of particles
  337. currentNumParticles = Math.ceil(objectSize.width / particleSize) * Math.ceil(objectSize.height / particleSize);
  338. if (currentNumParticles === requestedNumParticles) {
  339. return WebGraphics.makeSize(particleSize, particleSize);
  340. }
  341. if (currentNumParticles < requestedNumParticles) {
  342. do {
  343. prevNumParticles = currentNumParticles;
  344. particleSize--;
  345. currentNumParticles = Math.ceil(objectSize.width/particleSize) * Math.ceil(objectSize.height/particleSize);
  346. } while (currentNumParticles < requestedNumParticles && particleSize > 2);
  347. if (particleSize <= 2.0) {
  348. return WebGraphics.makeSize(Math.ceil(objectSize.width / 2), Math.ceil(objectSize.height / 2));
  349. }
  350. if (Math.abs(currentNumParticles - requestedNumParticles) < Math.abs(prevNumParticles - requestedNumParticles)) {
  351. return WebGraphics.makeSize(Math.ceil(objectSize.width / particleSize), Math.ceil(objectSize.height / particleSize));
  352. } else {
  353. return WebGraphics.makeSize(Math.ceil(objectSize.width / (particleSize + 1)), Math.ceil(objectSize.height / (particleSize + 1)));
  354. }
  355. } else {
  356. do {
  357. prevNumParticles = currentNumParticles;
  358. particleSize++;
  359. currentNumParticles = Math.ceil(objectSize.width / particleSize) * Math.ceil(objectSize.height / particleSize);
  360. } while (currentNumParticles > requestedNumParticles && particleSize > 2);
  361. if (particleSize <= 2.0) {
  362. return WebGraphics.makeSize(Math.ceil(objectSize.width / 2), Math.ceil(objectSize.height / 2));
  363. }
  364. if (Math.abs(currentNumParticles - requestedNumParticles) < Math.abs(prevNumParticles - requestedNumParticles)) {
  365. return WebGraphics.makeSize(Math.ceil(objectSize.width / particleSize), Math.ceil(objectSize.height / particleSize));
  366. } else {
  367. return WebGraphics.makeSize(Math.ceil(objectSize.width / (particleSize + 1)), Math.ceil(objectSize.height / (particleSize + 1)));
  368. }
  369. }
  370. },
  371. point3DRandomDirection: function() {
  372. var u = WebGraphics.randomBetween(-1.0, 1.0);
  373. var theta = WebGraphics.randomBetween(0, 2.0 * Math.PI);
  374. var prefix = Math.sqrt(1.0 - u * u);
  375. var result = WebGraphics.makePoint3D(prefix * Math.cos(theta), prefix * Math.sin(theta), u);
  376. return result;
  377. }
  378. });
  379. var KNWebGLBuildAnvilSmokeSystem = Class.create(KNWebGLParticleSystem, {
  380. initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) {
  381. this.willOverrideStartingPoints = true;
  382. // number of vertices per particle
  383. this.numberOfVerticesPerParticle = 4;
  384. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  385. this.setupWithTexture();
  386. },
  387. setupWithTexture: function() {
  388. this.animationWillBeginWithContext();
  389. },
  390. startingPointAtIndexPoint: function(indexPoint) {
  391. var index = this.indexFromPoint(indexPoint);
  392. return WebGraphics.makePoint(index / this.particleCount * this.objectSize.width - kParticleSize / 2, kParticleSize / 2);
  393. },
  394. speedAtIndexPoint: function(point) {
  395. var xDirection = (2 * this.indexFromPoint(point)) / (this.particleCount) -1;
  396. xDirection = (xDirection < 0 ? -1 : 1) * Math.sqrt(Math.abs(xDirection));
  397. //we've adjusted these values specifically for WebGL
  398. var yDirection = (Math.abs(xDirection) + 0.25) * WebGraphics.randomBetween(-1.0, 0.1);
  399. xDirection *= WebGraphics.randomBetween(0.25, 1);
  400. var speed = WebGraphics.makePoint3D(
  401. this.p_anvilGlobalScale() * 5.6 * xDirection,
  402. this.p_anvilGlobalScale() * 5 * -yDirection, 0);
  403. return speed;
  404. },
  405. rotationAtIndexPoint: function(point) {
  406. var xDirection = (2 * this.indexFromPoint(point)) / (this.particleCount) - 1;
  407. var thisRotation = xDirection * WebGraphics.randomBetween(0.5, 1) * Math.PI;
  408. thisRotation *= this.duration / 1000;
  409. return WebGraphics.makePoint3D(0, 0, thisRotation);
  410. },
  411. scaleAtIndexPoint: function(indexPoint) {
  412. var scale = Math.abs(((2.0 * this.indexFromPoint(indexPoint)) / this.particleCount) - 1.0);
  413. scale += 1.25;
  414. var randVal = WebGraphics.randomBetween(0, 1);
  415. randVal *= randVal * randVal;
  416. scale *= WebGraphics.mix(1, 2.5, randVal);
  417. scale *= (this.p_anvilGlobalScale() / kParticleSize);
  418. return scale;
  419. },
  420. lifeSpanAtIndexPoint: function(indexPoint) {
  421. return WebGraphics.makePoint(0, (this.duration / 1000) * WebGraphics.randomBetween(0.15, 1));
  422. },
  423. p_anvilGlobalScale: function() {
  424. return WebGraphics.mix(1.25, 0.75, this.objectSize.width / this.slideSize.width) * (this.objectSize.width/7);
  425. }
  426. });
  427. var KNWebGLBuildAnvilSpeckSystem = Class.create(KNWebGLParticleSystem, {
  428. initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) {
  429. this.willOverrideStartingPoints = true;
  430. // number of vertices per particle
  431. this.numberOfVerticesPerParticle = 4;
  432. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  433. this.setupWithTexture();
  434. },
  435. setupWithTexture: function() {
  436. this.animationWillBeginWithContext();
  437. },
  438. startingPointAtIndexPoint: function(indexPoint) {
  439. var index = this.indexFromPoint(indexPoint);
  440. return WebGraphics.makePoint(index/this.particleCount * this.objectSize.width - kParticleSize/2, kParticleSize/2);
  441. },
  442. speedAtIndexPoint: function(point) {
  443. var speed = WebGraphics.makePoint3D(0, 0, 0);
  444. var index = this.indexFromPoint(point);
  445. speed.x = 2 * index / this.particleCount - 1;
  446. speed.y = WebGraphics.randomBetween(0.2, 1);
  447. if (index % 3 != 1) {
  448. speed.z = 0.2;
  449. speed.y *= 0.025 * (WebGraphics.randomBetween(0,1) < 0.5 ? -1 : 1);
  450. speed.x *= 10;
  451. speed.y *= 10;
  452. } else {
  453. speed.z = 1.0;
  454. }
  455. speed.x *= this.p_anvilGlobalScale() * 5.25;
  456. 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
  457. return speed;
  458. },
  459. rotationAtIndexPoint: function(point) {
  460. var xDirection = (2 * this.indexFromPoint(point)) / (this.particleCount) - 1;
  461. var thisRotation = xDirection * WebGraphics.randomBetween(0.5, 1) * Math.PI;
  462. thisRotation *= this.duration / 1000;
  463. return WebGraphics.makePoint3D(0, 0, thisRotation);
  464. },
  465. scaleAtIndexPoint: function(indexPoint) {
  466. var index = this.indexFromPoint(indexPoint);
  467. var scale = WebGraphics.randomBetween(3, 5);
  468. if (index % 3 != 1) {
  469. scale *= 3;
  470. }
  471. scale *= (this.p_anvilGlobalScale() / kParticleSize);
  472. return scale;
  473. },
  474. lifeSpanAtIndexPoint: function(indexPoint) {
  475. var index = this.indexFromPoint(indexPoint);
  476. var lifeSpan = WebGraphics.makePoint(0, Math.min(1, this.duration/1000) * WebGraphics.randomBetween(0.2, 0.5));
  477. if (index % 3 != 1) {
  478. lifeSpan.y *= 10;
  479. }
  480. return lifeSpan;
  481. },
  482. p_anvilGlobalScale: function() {
  483. return WebGraphics.mix(1.25, 0.75, this.objectSize.width / this.slideSize.width)*(this.objectSize.width/7);
  484. }
  485. });
  486. var KNWebGLBuildFlameSystem = Class.create(KNWebGLParticleSystem, {
  487. initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture) {
  488. // overwrite starting point
  489. this.willOverrideStartingPoints = true;
  490. // number of vertices per particle
  491. this.numberOfVerticesPerParticle = 4;
  492. var width = objectSize.width;
  493. var height = objectSize.height;
  494. var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height));
  495. var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height));
  496. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  497. },
  498. p_setupParticleDataWithTexture: function(textureInfo) {
  499. var gl = this.gl;
  500. var width = textureInfo.width;
  501. var height = textureInfo.height;
  502. // create framebuffer to read gl texture
  503. var fb = gl.createFramebuffer();
  504. var pixels = new Uint8Array(width * height * 4);
  505. // bind the framebuffer for reading pixels
  506. gl.bindFramebuffer(gl.FRAMEBUFFER, fb);
  507. gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, textureInfo.texture, 0);
  508. if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) == gl.FRAMEBUFFER_COMPLETE) {
  509. gl.readPixels(0, 0, width, height, gl.RGBA, gl.UNSIGNED_BYTE, pixels);
  510. }
  511. var minPoint = {
  512. "x": width,
  513. "y": height
  514. };
  515. var maxPoint = {
  516. "x": 0,
  517. "y": 0
  518. };
  519. // Want to start the flames at the alpha-bottom of the image, not the absolute bottom
  520. var bottomRow = [];
  521. var bottomRowCount = width;
  522. var actualSize = {};
  523. for (var i = 0; i < width * 4; i += 4) {
  524. var x = i / 4;
  525. var minY = Number.MAX_VALUE;
  526. var maxY = 0;
  527. var hasNonTransparentPixels = false;
  528. var thisPoint = {
  529. "x": x / width,
  530. "y": -1
  531. };
  532. // search from top because the object is flipped
  533. for (var j = height - 1; j >= 0; j--) {
  534. var alphaIndex = x * 4 + j * (width) * 4 + 3;
  535. var y = height - j;
  536. if (pixels[alphaIndex]/255 > 0.1) {
  537. minY = Math.min(y, minY);
  538. maxY = Math.max(y, maxY);
  539. hasNonTransparentPixels = true;
  540. }
  541. }
  542. if (hasNonTransparentPixels) {
  543. thisPoint.y = maxY / height;
  544. minPoint.x = Math.min(minPoint.x, x);
  545. maxPoint.x = Math.max(maxPoint.x, x);
  546. minPoint.y = Math.min(minPoint.y, minY);
  547. maxPoint.y = Math.max(maxPoint.y, maxY);
  548. }
  549. bottomRow[x] = thisPoint;
  550. }
  551. actualSize = {
  552. width: maxPoint.x - minPoint.x,
  553. height: maxPoint.y - minPoint.y
  554. };
  555. this._actualSize = actualSize;
  556. this._bottomRow = bottomRow;
  557. this._bottomRowCount = bottomRowCount;
  558. // unbind the framebuffer
  559. gl.bindFramebuffer(gl.FRAMEBUFFER, null);
  560. this.animationWillBeginWithContext();
  561. var positionBuffer = this.attributeBuffers["Position"];
  562. var centerBuffer = this.attributeBuffers["Center"];
  563. // adjust vertices according to starting scale
  564. var minSide = this._actualSize.height / 2.0;
  565. var numberOfVerticesPerParticle = this.numberOfVerticesPerParticle;
  566. for (var index = 0, length = this.particleCount; index < length; index++) {
  567. var vertIndex = index * numberOfVerticesPerParticle;
  568. var origin = {
  569. "x": positionBuffer[vertIndex*2],
  570. "y": positionBuffer[vertIndex*2 + 1]
  571. };
  572. origin.x -= minSide / 2;
  573. origin.y -= minSide / 2;
  574. var newRect = WebGraphics.makeRect(origin.x, origin.y, minSide, minSide);
  575. var center = WebGraphics.makePoint(origin.x + minSide/2, origin.y + minSide/2);
  576. KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x, newRect.y), vertIndex, positionBuffer);
  577. KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x + newRect.width, newRect.y), vertIndex + 1, positionBuffer);
  578. KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x + newRect.width, newRect.y + newRect.height), vertIndex + 2, positionBuffer);
  579. KNWebGLUtil.setPoint2DAtIndexForAttribute(WebGraphics.makePoint(newRect.x, newRect.y + newRect.height), vertIndex + 3, positionBuffer);
  580. for (var i = 0; i < 4; i++) {
  581. KNWebGLUtil.setPoint2DAtIndexForAttribute(center, vertIndex + i, centerBuffer);
  582. }
  583. }
  584. },
  585. startingPointAtIndexPoint: function(indexPoint) {
  586. var x = 0;
  587. var y = 0;
  588. var foundY = false;
  589. for (var i = 0; i < this._bottomRowCount; i++) {
  590. // at least one row has y value not equal to -1
  591. if (this._bottomRow[i].y !== -1) {
  592. foundY = true;
  593. break;
  594. }
  595. }
  596. if (this._bottomRow && foundY) {
  597. do {
  598. var index = WebGraphics.randomBetween(0, this._bottomRowCount - 1);
  599. index = Math.round(index);
  600. x = this._bottomRow[index].x;
  601. y = this._bottomRow[index].y;
  602. } while (y === -1);
  603. }
  604. var positionPoint = {
  605. "x": x * this.objectSize.width,
  606. "y": y * this.objectSize.height
  607. };
  608. return positionPoint;
  609. },
  610. speedAtIndexPoint: function(indexPoint) {
  611. var index = this.indexFromPoint(indexPoint) * 4;
  612. var yPos = this.attributeBuffers["Position"][index*2 + 1];
  613. yPos = Math.min(yPos, this._actualSize.height);
  614. var maxUp = (this._actualSize.height * 0.2 + 0.9 * yPos) * (1.0 + WebGraphics.randomBetween(0.0, 0.1));
  615. var result = WebGraphics.makePoint3D(0, maxUp, 0);
  616. result = WebGraphics.multiplyPoint3DByScalar(result, 1.0 / this.speedMax());
  617. return result;
  618. },
  619. speedMax: function() {
  620. return this._actualSize.height * 1.1 * 1.1;
  621. },
  622. lifeSpanAtIndexPoint: function(indexPoint) {
  623. // CONSTANTS
  624. var maxParticleLife = Math.min(1.0, 1.0 / Math.max(2, this._duration));
  625. var thisParticleLife = maxParticleLife * WebGraphics.randomBetween(0.8, 1.0);
  626. // more at beginning
  627. var time = this.indexFromPoint(indexPoint) / this.particleCount;
  628. time = time * time * 0.25 + time * 0.75;
  629. time *= 1.0 - thisParticleLife;
  630. var lifeSpan = WebGraphics.makePoint(time, thisParticleLife);
  631. return lifeSpan;
  632. },
  633. rotationAtIndexPoint: function(indexPoint) {
  634. var rotation = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1.0, 1.0), 0, WebGraphics.randomBetween(-1.0, 1.0));
  635. return rotation;
  636. },
  637. rotationMax: function() {
  638. return Math.PI * 2;
  639. }
  640. });
  641. var KNWebGLBuildConfettiSystem = Class.create(KNWebGLParticleSystem, {
  642. initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture) {
  643. this.willOverrideStartingPoints = false;
  644. // number of vertices per particle
  645. this.numberOfVerticesPerParticle = 4;
  646. var width = objectSize.width;
  647. var height = objectSize.height;
  648. var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height));
  649. var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height));
  650. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  651. this.setupWithTexture();
  652. },
  653. setupWithTexture: function() {
  654. this.animationWillBeginWithContext();
  655. },
  656. startingPointAtIndexPoint: function(indexPoint) {
  657. var index = this.indexFromPoint(indexPoint);
  658. return WebGraphics.makePoint(index / this.particleCount * this.objectSize.width - kParticleSize / 2, kParticleSize / 2);
  659. },
  660. speedAtIndexPoint: function(point) {
  661. var speedMax = 0.025;
  662. var speedRandMax = speedMax * 20;
  663. var speedAdjust = this.objectSize.width / this.slideSize.width * this.objectSize.height / this.slideSize.height;
  664. speedAdjust = Math.sqrt(speedAdjust);
  665. speedAdjust = 1.0 - speedAdjust * 0.75;
  666. speedMax *= speedAdjust;
  667. speedRandMax *= speedAdjust;
  668. var minSide = Math.min(this.objectSize.height, this.objectSize.width);
  669. var theRandSpeed = minSide * speedRandMax;
  670. var theSpeed = minSide * speedMax;
  671. var maxRadiusSquared = (this.particleSystemSize.width * this.particleSystemSize.width + this.particleSystemSize.height * this.particleSystemSize.height) / 4.0;
  672. var randSpeed = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1));
  673. randSpeed.z = -Math.abs(randSpeed.z);
  674. randSpeed = WebGraphics.multiplyPoint3DByScalar(randSpeed, theRandSpeed);
  675. var vector = WebGraphics.makePoint3D(point.x - this.particlesWide / 2.0, point.y - this.particlesHigh / 2.0, 0);
  676. var radiusSquared = (vector.x * vector.x + vector.y * vector.y);
  677. vector.z = Math.sqrt(maxRadiusSquared - radiusSquared);
  678. var speed = WebGraphics.multiplyPoint3DByScalar(vector, theSpeed);
  679. speed = WebGraphics.addPoint3DToPoint3D(speed, randSpeed);
  680. return speed;
  681. },
  682. rotationAtIndexPoint: function(point) {
  683. var rotationMax = 8.0 * Math.PI;
  684. var rotation = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1));
  685. return WebGraphics.multiplyPoint3DByScalar(rotation, rotationMax);
  686. },
  687. scaleAtIndexPoint: function(indexPoint) {
  688. return 1.0;
  689. }
  690. });
  691. var KNWebGLBuildDiffuseSystem = Class.create(KNWebGLParticleSystem, {
  692. initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture, l2r) {
  693. this.willOverrideStartingPoints = false;
  694. // number of vertices per particle
  695. this.numberOfVerticesPerParticle = 4;
  696. // set direction
  697. this.l2r = l2r;
  698. var width = objectSize.width;
  699. var height = objectSize.height;
  700. var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height));
  701. var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height));
  702. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  703. this.setupWithTexture();
  704. },
  705. setupWithTexture: function() {
  706. this.animationWillBeginWithContext();
  707. },
  708. speedAtIndexPoint: function(indexPoint) {
  709. var l2r = this.l2r;
  710. //more random towards center
  711. var randDirection = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1));
  712. var vertical = 2 * indexPoint.y / this.particleSystemSize.height - 1.0;
  713. randDirection = WebGraphics.multiplyPoint3DByScalar(randDirection, (1.1 - vertical * vertical) * 8);
  714. //actual direction based on position of fragment height
  715. var startDirection = WebGraphics.makePoint3D((2 - Math.abs(vertical)) * (l2r ? -1: 1), vertical * 2, 0);
  716. startDirection = WebGraphics.multiplyPoint3DByScalar(startDirection, 5.0);
  717. var theDirection = WebGraphics.addPoint3DToPoint3D(startDirection, randDirection);
  718. theDirection = WebGraphics.point3DNormalize(theDirection);
  719. return theDirection;
  720. },
  721. speedMax: function() {
  722. var speed = this.objectSize.height * 1.5 / Math.sqrt(this.duration / 1000);
  723. return speed;
  724. },
  725. rotationAtIndexPoint: function(indexPoint) {
  726. var rotation = WebGraphics.makePoint3D(WebGraphics.randomBetween(-1, 1), WebGraphics.randomBetween(-1, 1), 0);
  727. return rotation;
  728. },
  729. rotationMax: function() {
  730. var rotationMax = 8.0 * Math.PI;
  731. return rotationMax;
  732. },
  733. lifeSpanAtIndexPoint: function(indexPoint) {
  734. var l2r = this.l2r;
  735. var width = this.particleSystemSize.width;
  736. var maxParticleLife = WebGraphics.clamp(0.8 / (this.duration / 1000), 0.1, 0.9);
  737. var time = (l2r ? (width - indexPoint.x): indexPoint.x) / width;
  738. time *= (1 - maxParticleLife);
  739. var lifeSpan = WebGraphics.makePoint(time, maxParticleLife);
  740. return lifeSpan;
  741. }
  742. });
  743. var KNWebGLBuildFireworksSystem = Class.create(KNWebGLParticleSystem, {
  744. initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) {
  745. // override starting point
  746. this.willOverrideStartingPoints = true;
  747. // override colors
  748. this.willOverrideColors = true;
  749. // number of vertices per particle
  750. this.numberOfVerticesPerParticle = 4;
  751. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  752. },
  753. setupWithTexture: function(textureInfo) {
  754. var gl = this.gl;
  755. var width = textureInfo.width;
  756. var height = textureInfo.height;
  757. var val1 = -Math.min(this.objectSize.height, this.slideSize.height / 10);
  758. var val2 = WebGraphics.randomBetween(val1, this.objectSize.height)
  759. this._startingPoint = WebGraphics.makePoint(this.fireworkStartingPositionX * this.objectSize.width, val2);
  760. // Initialize random color
  761. var startColor = WebGraphics.colorWithHSBA(WebGraphics.randomBetween(0, 1), 1, 1, 1);
  762. this._startingColorRGB = WebGraphics.makePoint3D(startColor.red, startColor.green, startColor.blue);
  763. this.animationWillBeginWithContext();
  764. },
  765. startingPointAtIndexPoint: function(indexPoint) {
  766. return this._startingPoint;
  767. },
  768. colorAtIndexPoint: function(indexPoint) {
  769. var randomColor = this.point3DRandomDirection();
  770. var randomColorResult = WebGraphics.multiplyPoint3DByScalar(randomColor, this.colorRandomness);
  771. var color = WebGraphics.addPoint3DToPoint3D(this._startingColorRGB, randomColorResult);
  772. color.x = WebGraphics.clamp(color.x, 0, 1);
  773. color.y = WebGraphics.clamp(color.y, 0, 1);
  774. color.z = WebGraphics.clamp(color.z, 0, 1);
  775. var result = {
  776. "x": color.x,
  777. "y": color.y,
  778. "z": color.z,
  779. "w": 1
  780. };
  781. return result;
  782. },
  783. speedAtIndexPoint: function(indexPoint) {
  784. var speed = this.point3DRandomDirection();
  785. var randomSpeed = WebGraphics.randomBetween(0.8, 1.0);
  786. var result = WebGraphics.multiplyPoint3DByScalar(speed, randomSpeed);
  787. return result;
  788. },
  789. speedMax: function() {
  790. return this.maxDistance;
  791. },
  792. scaleAtIndexPoint: function(indexPoint) {
  793. var scale = WebGraphics.randomBetween(this.randomParticleSizeMinMax.width, this.randomParticleSizeMinMax.height);
  794. return scale;
  795. },
  796. lifeSpanAtIndexPoint: function(indexPoint) {
  797. var lifeSpan = WebGraphics.makePoint(0, WebGraphics.randomBetween(this.lifeSpanMinDuration, 1.0));
  798. return lifeSpan;
  799. }
  800. });
  801. var KNWebGLBuildShimmerSystem = Class.create(KNWebGLParticleSystem, {
  802. initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture) {
  803. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  804. },
  805. speedMax: function() {
  806. var sqrtArea = Math.sqrt(this.objectSize.width * this.objectSize.height);
  807. var result = sqrtArea * 0.075;
  808. return result;
  809. },
  810. speedAtIndexPoint: function(indexPoint) {
  811. var speed = this.point3DRandomDirection();
  812. speed.z = 0;
  813. return speed;
  814. }
  815. });
  816. var KNWebGLBuildShimmerObjectSystem = Class.create(KNWebGLBuildShimmerSystem, {
  817. initialize: function($super, renderer, program, objectSize, slideSize, duration, numParticles, texture, direction) {
  818. // override starting point
  819. this.willOverrideStartingPoints = false;
  820. // override colors
  821. this.willOverrideColors = false;
  822. // number of vertices per particle
  823. this.numberOfVerticesPerParticle = 4;
  824. // require setting color pixels
  825. this.pixels = true;
  826. // set direction
  827. this.direction = direction;
  828. var width = objectSize.width;
  829. var height = objectSize.height;
  830. var particleSystemSize = this.particleSystemSizeWithRequestedNumber(numParticles, WebGraphics.makeSize(width, height));
  831. var particleSize = WebGraphics.makeSize(Math.ceil(width / particleSystemSize.width), Math.ceil(height / particleSystemSize.height));
  832. // call KNWebGLBuildShimmerSystem initialize method
  833. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture, direction);
  834. this.setupWithTexture();
  835. },
  836. setupWithTexture: function() {
  837. this.animationWillBeginWithContext();
  838. },
  839. drawGLSLWithPercent: function(percent, particleOpacity, rotation, clockwise) {
  840. var delayedPercent = Math.max(0.0, percent * 1.1 - 0.1);
  841. var origTexPercent = 1.0 - Math.min(1.0, TSDMixFloats(5.0 * percent, percent * percent, percent));
  842. // Rotation Matrix
  843. var angle = delayedPercent * (clockwise ? 1 : -1) * 2;
  844. var rotMatrix = CGAffineTransformMakeRotation(angle);
  845. var mat3 = WebGraphics.makeMat3WithAffineTransform(rotMatrix);
  846. // set mat3 uniform for RotationMatrix
  847. this.gl.uniformMatrix3fv(this.uniforms["RotationMatrix"], false, mat3);
  848. // draw GLSL with percent
  849. this.drawFrame(delayedPercent, particleOpacity * origTexPercent);
  850. }
  851. });
  852. var KNWebGLBuildShimmerParticleSystem = Class.create(KNWebGLBuildShimmerSystem, {
  853. initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, objectSystem, texture, direction) {
  854. // override starting point
  855. this.willOverrideStartingPoints = true;
  856. // override colors
  857. this.willOverrideColors = true;
  858. // number of vertices per particle
  859. this.numberOfVerticesPerParticle = 4;
  860. // set direction
  861. this.direction = direction;
  862. // object system particle count
  863. this.objectSystem = objectSystem;
  864. // object system vertex count
  865. this.objectSystemVertexCount = objectSystem.particleCount * 4;
  866. var kNumHullPoints = this.kNumHullPoints = 20
  867. //beginning of width of bottom spire on x-axis
  868. var kSpireMinW = 0.48828125;
  869. //beginning of height of bottom spire on y-axis
  870. var kSpireMinH = 0.02;
  871. //top of height of bottom spire on y-axis
  872. var kSpireMaxH = 0.3;
  873. this.p_particleHullArray = [
  874. // center quad
  875. { x: kSpireMaxH, y: kSpireMaxH },
  876. { x: 1 - kSpireMaxH, y: kSpireMaxH },
  877. { x: 1 - kSpireMaxH, y: 1 - kSpireMaxH },
  878. { x: kSpireMaxH, y: 1 - kSpireMaxH },
  879. // top quad
  880. { x: kSpireMinW, y: 1 - kSpireMaxH },
  881. { x: 1 - kSpireMinW, y: 1 - kSpireMaxH },
  882. { x: 1 - kSpireMinW, y: 1 - kSpireMinH },
  883. { x: kSpireMinW, y: 1 - kSpireMinH },
  884. // right quad
  885. { x: 1 - kSpireMaxH, y: 1 - kSpireMinW },
  886. { x: 1 - kSpireMinH, y: 1 - kSpireMinW },
  887. { x: 1 - kSpireMinH, y: kSpireMinW },
  888. { x: 1 - kSpireMaxH, y: kSpireMinW },
  889. // bottom quad
  890. { x: kSpireMinW, y: kSpireMinH },
  891. { x: 1 - kSpireMinW, y: kSpireMinH },
  892. { x: 1 - kSpireMinW, y: kSpireMaxH },
  893. { x: kSpireMinW, y: kSpireMaxH },
  894. // left quad
  895. { x: kSpireMinH, y: kSpireMinW },
  896. { x: kSpireMaxH, y: kSpireMinW },
  897. { x: kSpireMaxH, y: 1 - kSpireMinW },
  898. { x: kSpireMinH, y: 1 - kSpireMinW },
  899. ];
  900. // call KNWebGLBuildShimmerSystem initialize method
  901. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture, direction);
  902. this.setupWithTexture();
  903. },
  904. setupWithTexture: function() {
  905. this.animationWillBeginWithContext();
  906. // update vertex data
  907. this.p_setupVertexData();
  908. },
  909. p_setupVertexData: function() {
  910. var texRect = CGRectMake(0, 0, 1, 1);
  911. var numVertsPP = this.numberOfVerticesPerParticle;
  912. var oldParticleCount = this.particleCount;
  913. var newParticleCount = this.particleCount * 5;
  914. // objtain a reference of current data buffer
  915. var oldAttributeBuffers = this.attributeBuffers;
  916. // Set up fresh, new data buffer attributes!
  917. var newAttributeBuffers = {};
  918. newAttributeBuffers["Position"] = [];
  919. newAttributeBuffers["Center"] = [];
  920. newAttributeBuffers["TexCoord"] = [];
  921. newAttributeBuffers["ParticleTexCoord"] = [];
  922. newAttributeBuffers["Speed"] = [];
  923. newAttributeBuffers["Rotation"] = [];
  924. newAttributeBuffers["Scale"] = [];
  925. newAttributeBuffers["LifeSpan"] = [];
  926. newAttributeBuffers["Color"] = [];
  927. // Update new buffer with old buffer values
  928. for (var i = 0; i < oldParticleCount; ++i) {
  929. var oldVertIndex = i * 4;
  930. var thisScale = this.attributeBuffers["Scale"][oldVertIndex];
  931. var thisSpeed = WebGraphics.makePoint3D(
  932. this.attributeBuffers["Speed"][oldVertIndex * 3],
  933. this.attributeBuffers["Speed"][oldVertIndex * 3 + 1],
  934. this.attributeBuffers["Speed"][oldVertIndex * 3 + 2]
  935. );
  936. var thisColor = WebGraphics.makePoint4D(
  937. this.attributeBuffers["Color"][oldVertIndex * 4],
  938. this.attributeBuffers["Color"][oldVertIndex * 4 + 1],
  939. this.attributeBuffers["Color"][oldVertIndex * 4 + 2],
  940. this.attributeBuffers["Color"][oldVertIndex * 4 + 3]
  941. );
  942. var vertMin = WebGraphics.makePoint(
  943. this.attributeBuffers["Position"][oldVertIndex * 2],
  944. this.attributeBuffers["Position"][oldVertIndex * 2 + 1]
  945. );
  946. var vertMax = WebGraphics.makePoint(
  947. this.attributeBuffers["Position"][oldVertIndex * 2 + 4],
  948. this.attributeBuffers["Position"][oldVertIndex * 2 + 5]
  949. );
  950. var vertRect = TSDRectWithPoints(vertMin, vertMax);
  951. // set vertRect to empty if the particle from the object system is not visible
  952. if (this.objectSystem.particleSystemVisibilities[i] === false) {
  953. vertRect = TSDRectWithPoints(WebGraphics.makePoint(0, 0), WebGraphics.makePoint(0, 0));
  954. }
  955. var center = WebGraphics.makePoint(
  956. this.attributeBuffers["Center"][oldVertIndex * 2],
  957. this.attributeBuffers["Center"][oldVertIndex * 2 + 1]
  958. );
  959. var lifeSpan = WebGraphics.makePoint(
  960. this.attributeBuffers["LifeSpan"][oldVertIndex * 2],
  961. this.attributeBuffers["LifeSpan"][oldVertIndex * 2 + 1]
  962. );
  963. var p_particleHullArray = this.p_particleHullArray;
  964. var kNumHullPoints = this.kNumHullPoints;
  965. for (var v = 0; v < kNumHullPoints; ++v) {
  966. var newVertIndex = i * kNumHullPoints + v;
  967. var thisHullPoint = p_particleHullArray[v];
  968. var newVertexPoint = this.p_hullPoint(thisHullPoint, vertRect);
  969. KNWebGLUtil.setPoint2DAtIndexForAttribute(newVertexPoint, newVertIndex, newAttributeBuffers["Position"]);
  970. var newTexCoord = this.p_hullPoint(thisHullPoint, texRect);
  971. KNWebGLUtil.setPoint2DAtIndexForAttribute(newTexCoord, newVertIndex, newAttributeBuffers["ParticleTexCoord"]);
  972. KNWebGLUtil.setPoint2DAtIndexForAttribute(center, newVertIndex, newAttributeBuffers["Center"]);
  973. KNWebGLUtil.setPoint4DAtIndexForAttribute(thisColor, newVertIndex, newAttributeBuffers["Color"]);
  974. KNWebGLUtil.setPoint3DAtIndexForAttribute(thisSpeed, newVertIndex, newAttributeBuffers["Speed"]);
  975. KNWebGLUtil.setFloatAtIndexForAttribute(thisScale, newVertIndex, newAttributeBuffers["Scale"]);
  976. KNWebGLUtil.setPoint2DAtIndexForAttribute(lifeSpan, newVertIndex, newAttributeBuffers["LifeSpan"]);
  977. }
  978. }
  979. // update data buffer with new data buffer
  980. this.attributeBuffers = newAttributeBuffers;
  981. // update element array
  982. var elementArray = this.elementArray = [];
  983. var indexCounter = 0;
  984. for (var i = 0; i < newParticleCount; i++) {
  985. elementArray[indexCounter++] = 4 * i + 0;
  986. elementArray[indexCounter++] = 4 * i + 1;
  987. elementArray[indexCounter++] = 4 * i + 2;
  988. // second triangle
  989. elementArray[indexCounter++] = 4 * i + 0;
  990. elementArray[indexCounter++] = 4 * i + 2;
  991. elementArray[indexCounter++] = 4 * i + 3;
  992. }
  993. },
  994. p_hullPoint: function(hullPoint, vertexRect) {
  995. var point = WebGraphics.makePoint(vertexRect.origin.x + vertexRect.size.width * hullPoint.x, vertexRect.origin.y + vertexRect.size.height * hullPoint.y);
  996. return point;
  997. },
  998. startingPointAtIndexPoint: function(indexPoint) {
  999. var index = this.indexFromPoint(indexPoint) * 4;
  1000. var objectSystemVertexCount = this.objectSystemVertexCount;
  1001. var attributeBuffers = this.objectSystem.attributeBuffers["Position"];
  1002. if (index < objectSystemVertexCount) {
  1003. // Copy value from object particle system so this sparkle exactly matches up with an existing object particle
  1004. var result = WebGraphics.makePoint(
  1005. attributeBuffers[index * 2],
  1006. attributeBuffers[index * 2 + 1]
  1007. );
  1008. return result;
  1009. }
  1010. // else, it's an extra sparkle at the end, in a random location!
  1011. var halfWidth = this.objectSize.width / 2;
  1012. var halfHeight = this.objectSize.height / 2;
  1013. var midPoint = CGPointMake(halfWidth, halfHeight);
  1014. // attenuate points towards the middle
  1015. var r = WebGraphics.randomBetween(0, 1);
  1016. var angle = WebGraphics.doubleBetween(0, 2.0 * Math.PI);
  1017. var randLoc = CGPointMake(midPoint.x + halfWidth * r * Math.cos(angle), midPoint.y + halfHeight * r * Math.sin(angle));
  1018. return randLoc;
  1019. },
  1020. speedAtIndexPoint: function(indexPoint) {
  1021. var index = this.indexFromPoint(indexPoint) * 4;
  1022. var objectSystemVertexCount = this.objectSystemVertexCount;
  1023. var attributeBuffers = this.objectSystem.attributeBuffers["Speed"];
  1024. if (index < objectSystemVertexCount) {
  1025. // Copy value from object particle system so this sparkle exactly matches up with an existing object particle
  1026. var result = WebGraphics.makePoint3D(
  1027. attributeBuffers[index * 3],
  1028. attributeBuffers[index * 3 + 1],
  1029. attributeBuffers[index * 3 + 2]
  1030. );
  1031. return result;
  1032. }
  1033. // else, it's an extra sparkle at the end; stay in place
  1034. var speed = WebGraphics.makePoint3D(0, 0, 0);
  1035. return speed;
  1036. },
  1037. scaleAtIndexPoint: function(indexPoint) {
  1038. var minScale = 1.0;
  1039. var maxScale = 25.0;
  1040. var randNum = WebGraphics.randomBetween(0, 1);
  1041. // Most particles will be smaller
  1042. var result = TSUMix(minScale, maxScale, randNum * randNum);
  1043. return result;
  1044. },
  1045. lifeSpanAtIndexPoint: function(indexPoint) {
  1046. var index = this.indexFromPoint(indexPoint) * 4;
  1047. var objectSystemVertexCount = this.objectSystemVertexCount;
  1048. if (index < objectSystemVertexCount) {
  1049. // This is an existing object
  1050. return WebGraphics.makePoint(0, 1);
  1051. }
  1052. // else, it's an extra sparkle at the end, with a random life span!
  1053. // 2 second or 90% of total duration, whichever is shorter
  1054. var lifeDuration = Math.min(2.0 / (this.duration / 1000), 0.9);
  1055. var lifeStart = TSDMixFloats(0.01, 0.99, 1.0 - TSUReverseSquare(WebGraphics.randomBetween(0, 1)));
  1056. // Since percent is non-uniform, make duration less at later part of anim
  1057. lifeDuration *= TSUReverseSquare(lifeStart);
  1058. // Make sure lifespan ends before animation completes
  1059. lifeStart = Math.min(lifeStart, 0.99 - lifeDuration);
  1060. var result = WebGraphics.makePoint(lifeStart, lifeDuration);
  1061. return result;
  1062. },
  1063. colorAtIndexPoint: function(indexPoint) {
  1064. var index = this.indexFromPoint(indexPoint) * 4;
  1065. var objectSystemVertexCount = this.objectSystemVertexCount;
  1066. var attributeBuffers = this.objectSystem.attributeBuffers["Color"];
  1067. if (index < objectSystemVertexCount) {
  1068. // Copy value from object particle system so this sparkle exactly matches up with an existing object particle
  1069. var result = WebGraphics.makePoint4D(
  1070. attributeBuffers[index * 4],
  1071. attributeBuffers[index * 4 + 1],
  1072. attributeBuffers[index * 4 + 2],
  1073. attributeBuffers[index * 4 + 3]
  1074. );
  1075. return result;
  1076. }
  1077. // else it's an extra sparkle at the end
  1078. // white
  1079. var result = WebGraphics.makePoint4D(1, 1, 1, 1);
  1080. return result;
  1081. },
  1082. drawGLSLWithPercent: function(percent, opacity, rotation, clockwise) {
  1083. // Rotation Matrix
  1084. var angle = percent * (clockwise ? 1 : -1) * 2;
  1085. var rotMatrix = CGAffineTransformMakeRotation(angle);
  1086. var mat3 = WebGraphics.makeMat3WithAffineTransform(rotMatrix);
  1087. // set mat3 uniform for RotationMatrix
  1088. this.gl.uniformMatrix3fv(this.uniforms[kShimmerUniformRotationMatrix], false, mat3);
  1089. // Particle Scale Percent
  1090. var invPercent = 1.0 - percent;
  1091. var powPercent = Math.pow(invPercent, 15.0);
  1092. var particleScalePercent = TSUMix(invPercent * invPercent, 25. * percent, powPercent);
  1093. this.gl.uniform1f(this.uniforms[kShimmerUniformParticleScalePercent], particleScalePercent);
  1094. // Draw object
  1095. this.drawFrame(percent, opacity);
  1096. }
  1097. });
  1098. var KNWebGLBuildSparkleSystem = Class.create(KNWebGLParticleSystem, {
  1099. initialize: function($super, renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture, direction) {
  1100. this.willOverrideStartingPoints = true;
  1101. // number of vertices per particle
  1102. this.numberOfVerticesPerParticle = 4;
  1103. // set direction
  1104. this.direction = direction;
  1105. $super(renderer, program, objectSize, slideSize, duration, particleSystemSize, particleSize, texture);
  1106. // cache p_globalScale for faster access
  1107. this.cachedGlobalScale = this.p_globalScale();
  1108. this.setupWithTexture();
  1109. },
  1110. setupWithTexture: function() {
  1111. this.animationWillBeginWithContext();
  1112. },
  1113. p_globalScale: function() {
  1114. var objectSize = this.objectSize;
  1115. var slideSize = this.slideSize;
  1116. var minSide = Math.min(objectSize.width, objectSize.height);
  1117. var minSideRatio = minSide / (Math.min(slideSize.width, slideSize.height));
  1118. minSide = minSide / Math.sqrt(Math.sqrt(minSideRatio)) * 0.25;
  1119. return minSide;
  1120. },
  1121. startingPointAtIndexPoint: function(indexPoint) {
  1122. // CONSTANTS
  1123. var maxOffset = 0.1 / (this.duration / 1000);
  1124. var yAxis = (this.direction == KNDirection.kKNDirectionTopToBottom || this.direction == KNDirection.kKNDirectionBottomToTop);
  1125. var reverse = (this.direction == KNDirection.kKNDirectionRightToLeft || this.direction == KNDirection.kKNDirectionTopToBottom);
  1126. var index = this.indexFromPoint(indexPoint);
  1127. var x = index / this.particleCount; // position along line
  1128. x = reverse ? 1 - x : x;
  1129. var axis1 = x + maxOffset * WebGraphics.doubleBetween(-1.0, 1.0);
  1130. var axis2 = Math.random();
  1131. // skew position towards center vertically
  1132. axis2 = 2. * axis2 - 1.;
  1133. axis2 *= Math.abs(axis2);
  1134. axis2 = (axis2 + 1.) / 2.;
  1135. var newX = yAxis ? axis2 : axis1;
  1136. var newY = yAxis ? axis1 : axis2;
  1137. var position = WebGraphics.makePoint(
  1138. newX * this.objectSize.width - this.particleSize.width / 2.0,
  1139. newY * this.objectSize.height - this.particleSize.height / 2.0
  1140. );
  1141. return position;
  1142. },
  1143. speedAtIndexPoint: function(point) {
  1144. var speed = this.point3DRandomDirection();
  1145. // reduce z speed
  1146. speed.z *= 0.01;
  1147. var randomMultiplier = Math.random();
  1148. var result = WebGraphics.multiplyPoint3DByScalar(speed, randomMultiplier);
  1149. return result;
  1150. },
  1151. speedMax: function() {
  1152. var minSide = Math.min(this.objectSize.width, this.objectSize.height);
  1153. var minSideRatio = minSide / (Math.min(this.slideSize.width, this.slideSize.height));
  1154. minSide = minSide / Math.pow(minSideRatio, 0.667) * 0.25 * 1.5;
  1155. return minSide;
  1156. },
  1157. scaleAtIndexPoint: function(indexPoint) {
  1158. var minSide = this.cachedGlobalScale;
  1159. var result = minSide / this.particleSize.width;
  1160. return result;
  1161. },
  1162. lifeSpanAtIndexPoint: function(indexPoint) {
  1163. var index = this.indexFromPoint(indexPoint);
  1164. var timeStart = index / this.particleCount;
  1165. var timeDuration = KNSparkleMaxParticleLife / Math.max(0.75, this.duration/1000);
  1166. timeStart *= 1 - timeDuration;
  1167. var lifeSpan = WebGraphics.makePoint(timeStart, timeDuration);
  1168. return lifeSpan;
  1169. }
  1170. });