particles.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. /* -----------------------------------------------
  2. /* Author : Vincent Garreau - vincentgarreau.com
  3. /* MIT license: http://opensource.org/licenses/MIT
  4. /* GitHub : https://github.com/VincentGarreau/particles.js
  5. /* How to use? : Check the GitHub README
  6. /* v1.0.3
  7. /* ----------------------------------------------- */
  8. function launchParticlesJS(tag_id, params){
  9. var canvas_el = document.querySelector('#'+tag_id+' > canvas');
  10. /* particles.js variables with default values */
  11. pJS = {
  12. canvas: {
  13. el: canvas_el,
  14. w: canvas_el.offsetWidth,
  15. h: canvas_el.offsetHeight
  16. },
  17. particles: {
  18. color: '#fff',
  19. shape: 'circle',
  20. opacity: 1,
  21. size: 2.5,
  22. size_random: true,
  23. nb: 200,
  24. line_linked: {
  25. enable_auto: true,
  26. distance: 100,
  27. color: '#fff',
  28. opacity: 1,
  29. width: 1,
  30. condensed_mode: {
  31. enable: true,
  32. rotateX: 65000,
  33. rotateY: 65000
  34. }
  35. },
  36. anim: {
  37. enable: true,
  38. speed: 1
  39. },
  40. array: []
  41. },
  42. interactivity: {
  43. enable: true,
  44. mouse: {
  45. distance: 100
  46. },
  47. detect_on: 'canvas',
  48. mode: 'grab',
  49. line_linked: {
  50. opacity: 1
  51. },
  52. events: {
  53. onclick: {
  54. enable: true,
  55. mode: 'push',
  56. nb: 4
  57. }
  58. }
  59. },
  60. retina_detect: false,
  61. fn: {
  62. vendors:{
  63. interactivity: {}
  64. }
  65. }
  66. };
  67. /* params settings */
  68. if(params){
  69. if(params.particles){
  70. var paramsForParticles = params.particles;
  71. if(paramsForParticles.color) pJS.particles.color = paramsForParticles.color;
  72. if(paramsForParticles.shape) pJS.particles.shape = paramsForParticles.shape;
  73. if(paramsForParticles.opacity) pJS.particles.opacity = paramsForParticles.opacity;
  74. if(paramsForParticles.size) pJS.particles.size = paramsForParticles.size;
  75. if(paramsForParticles.size_random == false) pJS.particles.size_random = paramsForParticles.size_random;
  76. if(paramsForParticles.nb) pJS.particles.nb = paramsForParticles.nb;
  77. if(paramsForParticles.line_linked){
  78. var paramsForLineLinked = paramsForParticles.line_linked;
  79. if(paramsForLineLinked.enable_auto == false) pJS.particles.line_linked.enable_auto = paramsForLineLinked.enable_auto;
  80. if(paramsForLineLinked.distance) pJS.particles.line_linked.distance = paramsForLineLinked.distance;
  81. if(paramsForLineLinked.color) pJS.particles.line_linked.color = paramsForLineLinked.color;
  82. if(paramsForLineLinked.opacity) pJS.particles.line_linked.opacity = paramsForLineLinked.opacity;
  83. if(paramsForLineLinked.width) pJS.particles.line_linked.width = paramsForLineLinked.width;
  84. if(paramsForLineLinked.condensed_mode){
  85. var paramsForCondensedMode = paramsForLineLinked.condensed_mode;
  86. if(paramsForCondensedMode.enable == false) pJS.particles.line_linked.condensed_mode.enable = paramsForCondensedMode.enable;
  87. if(paramsForCondensedMode.rotateX) pJS.particles.line_linked.condensed_mode.rotateX = paramsForCondensedMode.rotateX;
  88. if(paramsForCondensedMode.rotateY) pJS.particles.line_linked.condensed_mode.rotateY = paramsForCondensedMode.rotateY;
  89. }
  90. }
  91. if(paramsForParticles.anim){
  92. var paramsForAnim = paramsForParticles.anim;
  93. if(paramsForAnim.enable == false) pJS.particles.anim.enable = paramsForAnim.enable;
  94. if(paramsForAnim.speed) pJS.particles.anim.speed = paramsForAnim.speed;
  95. }
  96. }
  97. if(params.interactivity){
  98. var paramsForInteractivity = params.interactivity;
  99. if(paramsForInteractivity.enable == false) pJS.interactivity.enable = paramsForInteractivity.enable;
  100. if(paramsForInteractivity.mouse){
  101. if(paramsForInteractivity.mouse.distance) pJS.interactivity.mouse.distance = paramsForInteractivity.mouse.distance;
  102. }
  103. if(paramsForInteractivity.detect_on) pJS.interactivity.detect_on = paramsForInteractivity.detect_on;
  104. if(paramsForInteractivity.mode) pJS.interactivity.mode = paramsForInteractivity.mode;
  105. if(paramsForInteractivity.line_linked){
  106. if(paramsForInteractivity.line_linked.opacity) pJS.interactivity.line_linked.opacity = paramsForInteractivity.line_linked.opacity;
  107. }
  108. if(paramsForInteractivity.events){
  109. var paramsForEvents = paramsForInteractivity.events;
  110. if(paramsForEvents.onclick){
  111. var paramsForOnclick = paramsForEvents.onclick;
  112. if(paramsForOnclick.enable == false) pJS.interactivity.events.onclick.enable = false;
  113. if(paramsForOnclick.mode != 'push') pJS.interactivity.events.onclick.mode = paramsForOnclick.mode;
  114. if(paramsForOnclick.nb) pJS.interactivity.events.onclick.nb = paramsForOnclick.nb;
  115. }
  116. }
  117. }
  118. pJS.retina_detect = params.retina_detect;
  119. }
  120. /* convert hex colors to rgb */
  121. pJS.particles.color_rgb = hexToRgb(pJS.particles.color);
  122. pJS.particles.line_linked.color_rgb_line = hexToRgb(pJS.particles.line_linked.color);
  123. /* detect retina */
  124. if(pJS.retina_detect && window.devicePixelRatio > 1){
  125. pJS.retina = true;
  126. pJS.canvas.pxratio = window.devicePixelRatio
  127. pJS.canvas.w = pJS.canvas.el.offsetWidth * pJS.canvas.pxratio;
  128. pJS.canvas.h = pJS.canvas.el.offsetHeight * pJS.canvas.pxratio;
  129. pJS.particles.anim.speed = pJS.particles.anim.speed * pJS.canvas.pxratio;
  130. pJS.particles.line_linked.distance = pJS.particles.line_linked.distance * pJS.canvas.pxratio;
  131. pJS.particles.line_linked.width = pJS.particles.line_linked.width * pJS.canvas.pxratio;
  132. pJS.interactivity.mouse.distance = pJS.interactivity.mouse.distance * pJS.canvas.pxratio;
  133. }
  134. /* ---------- CANVAS functions ------------ */
  135. pJS.fn.canvasInit = function(){
  136. pJS.canvas.ctx = pJS.canvas.el.getContext('2d');
  137. };
  138. pJS.fn.canvasSize = function(){
  139. pJS.canvas.el.width = pJS.canvas.w;
  140. pJS.canvas.el.height = pJS.canvas.h;
  141. window.onresize = function(){
  142. if(pJS){
  143. pJS.canvas.w = pJS.canvas.el.offsetWidth;
  144. pJS.canvas.h = pJS.canvas.el.offsetHeight;
  145. /* resize canvas */
  146. if(pJS.retina){
  147. pJS.canvas.w *= pJS.canvas.pxratio;
  148. pJS.canvas.h *= pJS.canvas.pxratio;
  149. }
  150. pJS.canvas.el.width = pJS.canvas.w;
  151. pJS.canvas.el.height = pJS.canvas.h;
  152. /* repaint canvas */
  153. pJS.fn.canvasPaint();
  154. if(!pJS.particles.anim.enable){
  155. pJS.fn.particlesRemove();
  156. pJS.fn.canvasRemove();
  157. launchParticles();
  158. }
  159. }
  160. }
  161. };
  162. pJS.fn.canvasPaint = function(){
  163. pJS.canvas.ctx.fillRect(0, 0, pJS.canvas.w, pJS.canvas.h);
  164. };
  165. pJS.fn.canvasRemove = function(){
  166. pJS.canvas.ctx.clearRect(0, 0, pJS.canvas.w, pJS.canvas.h);
  167. }
  168. /* --------- PARTICLES functions ----------- */
  169. pJS.fn.particle = function(color, opacity, position){
  170. /* position */
  171. this.x = position ? position.x : Math.random() * pJS.canvas.w;
  172. this.y = position ? position.y : Math.random() * pJS.canvas.h;
  173. /* size */
  174. this.radius = (pJS.particles.size_random ? Math.random() : 1) * pJS.particles.size;
  175. if (pJS.retina) this.radius *= pJS.canvas.pxratio;
  176. /* color */
  177. this.color = color;
  178. /* opacity */
  179. this.opacity = opacity;
  180. /* animation - velocity for speed */
  181. this.vx = -.5 + Math.random();
  182. this.vy = -.5 + Math.random();
  183. /* draw function */
  184. this.draw = function(){
  185. pJS.canvas.ctx.fillStyle = 'rgba('+this.color.r+','+this.color.g+','+this.color.b+','+this.opacity+')';
  186. pJS.canvas.ctx.beginPath();
  187. switch(pJS.particles.shape){
  188. case 'circle':
  189. pJS.canvas.ctx.arc(this.x, this.y, this.radius, 0, Math.PI * 2, false);
  190. break;
  191. case 'edge':
  192. pJS.canvas.ctx.rect(this.x, this.y, this.radius*2, this.radius*2);
  193. break;
  194. case 'triangle':
  195. pJS.canvas.ctx.moveTo(this.x,this.y-this.radius);
  196. pJS.canvas.ctx.lineTo(this.x+this.radius,this.y+this.radius);
  197. pJS.canvas.ctx.lineTo(this.x-this.radius,this.y+this.radius);
  198. pJS.canvas.ctx.closePath();
  199. break;
  200. }
  201. pJS.canvas.ctx.fill();
  202. }
  203. };
  204. pJS.fn.particlesCreate = function(){
  205. for(var i = 0; i < pJS.particles.nb; i++) {
  206. pJS.particles.array.push(new pJS.fn.particle(pJS.particles.color_rgb, pJS.particles.opacity));
  207. }
  208. };
  209. pJS.fn.particlesAnimate = function(){
  210. for(var i = 0; i < pJS.particles.array.length; i++){
  211. /* the particle */
  212. var p = pJS.particles.array[i];
  213. /* move the particle */
  214. p.x += p.vx * (pJS.particles.anim.speed/2);
  215. p.y += p.vy * (pJS.particles.anim.speed/2);
  216. /* change particle position if it is out of canvas */
  217. if(p.x - p.radius > pJS.canvas.w) p.x = p.radius;
  218. else if(p.x + p.radius < 0) p.x = pJS.canvas.w + p.radius;
  219. if(p.y - p.radius > pJS.canvas.h) p.y = p.radius;
  220. else if(p.y + p.radius < 0) p.y = pJS.canvas.h + p.radius;
  221. /* Check distance between each particle and mouse position */
  222. for(var j = i + 1; j < pJS.particles.array.length; j++){
  223. var p2 = pJS.particles.array[j];
  224. /* link particles if enable */
  225. if(pJS.particles.line_linked.enable_auto){
  226. pJS.fn.vendors.distanceParticles(p,p2);
  227. }
  228. /* set interactivity if enable */
  229. if(pJS.interactivity.enable){
  230. /* interactivity mode */
  231. switch(pJS.interactivity.mode){
  232. case 'grab':
  233. pJS.fn.vendors.interactivity.grabParticles(p,p2);
  234. break;
  235. }
  236. }
  237. }
  238. }
  239. };
  240. pJS.fn.particlesDraw = function(){
  241. /* clear canvas */
  242. pJS.canvas.ctx.clearRect(0, 0, pJS.canvas.w, pJS.canvas.h);
  243. /* move particles */
  244. pJS.fn.particlesAnimate();
  245. /* draw each particle */
  246. for(var i = 0; i < pJS.particles.array.length; i++){
  247. var p = pJS.particles.array[i];
  248. p.draw('rgba('+p.color.r+','+p.color.g+','+p.color.b+','+p.opacity+')');
  249. }
  250. };
  251. pJS.fn.particlesRemove = function(){
  252. pJS.particles.array = [];
  253. };
  254. /* ---------- VENDORS functions ------------ */
  255. pJS.fn.vendors.distanceParticles = function(p1, p2){
  256. var dx = p1.x - p2.x,
  257. dy = p1.y - p2.y,
  258. dist = Math.sqrt(dx*dx + dy*dy);
  259. /* Check distance between particle and mouse mos */
  260. if(dist <= pJS.particles.line_linked.distance) {
  261. /* draw the line */
  262. var color_line = pJS.particles.line_linked.color_rgb_line;
  263. pJS.canvas.ctx.beginPath();
  264. pJS.canvas.ctx.strokeStyle = 'rgba('+color_line.r+','+color_line.g+','+color_line.b+','+ (pJS.particles.line_linked.opacity-dist/pJS.particles.line_linked.distance) +')';
  265. pJS.canvas.ctx.moveTo(p1.x, p1.y);
  266. pJS.canvas.ctx.lineTo(p2.x, p2.y);
  267. pJS.canvas.ctx.lineWidth = pJS.particles.line_linked.width;
  268. pJS.canvas.ctx.stroke();
  269. pJS.canvas.ctx.closePath();
  270. /* condensed particles */
  271. if(pJS.particles.line_linked.condensed_mode.enable){
  272. var dx = p1.x - p2.x,
  273. dy = p1.y - p2.y,
  274. ax = dx/(pJS.particles.line_linked.condensed_mode.rotateX*1000),
  275. ay = dy/(pJS.particles.line_linked.condensed_mode.rotateY*1000);
  276. p2.vx += ax;
  277. p2.vy += ay;
  278. }
  279. }
  280. };
  281. pJS.fn.vendors.interactivity.listeners = function(){
  282. /* init el */
  283. if(pJS.interactivity.detect_on == 'window'){
  284. var detect_el = window;
  285. }else{
  286. var detect_el = pJS.canvas.el;
  287. }
  288. /* el on mousemove */
  289. detect_el.onmousemove = function(e){
  290. if(detect_el == window){
  291. var pos_x = e.clientX,
  292. pos_y = e.clientY;
  293. }
  294. else{
  295. var pos_x = e.offsetX||e.clientX,
  296. pos_y = e.offsetY||e.clientY;
  297. }
  298. if(pJS){
  299. pJS.interactivity.mouse.pos_x = pos_x;
  300. pJS.interactivity.mouse.pos_y = pos_y;
  301. if(pJS.retina){
  302. pJS.interactivity.mouse.pos_x *= pJS.canvas.pxratio;
  303. pJS.interactivity.mouse.pos_y *= pJS.canvas.pxratio;
  304. }
  305. pJS.interactivity.status = 'mousemove';
  306. }
  307. };
  308. /* el on onmouseleave */
  309. detect_el.onmouseleave = function(e){
  310. if(pJS){
  311. pJS.interactivity.mouse.pos_x = 0;
  312. pJS.interactivity.mouse.pos_y = 0;
  313. pJS.interactivity.status = 'mouseleave';
  314. }
  315. };
  316. /* el on onclick */
  317. if(pJS.interactivity.events.onclick.enable){
  318. switch(pJS.interactivity.events.onclick.mode){
  319. case 'push':
  320. detect_el.onclick = function(e){
  321. if(pJS){
  322. for(var i = 0; i < pJS.interactivity.events.onclick.nb; i++){
  323. pJS.particles.array.push(
  324. new pJS.fn.particle(
  325. pJS.particles.color_rgb,
  326. pJS.particles.opacity,
  327. {
  328. 'x': pJS.interactivity.mouse.pos_x,
  329. 'y': pJS.interactivity.mouse.pos_y
  330. }
  331. )
  332. )
  333. }
  334. }
  335. }
  336. break;
  337. case 'remove':
  338. detect_el.onclick = function(e){
  339. pJS.particles.array.splice(0, pJS.interactivity.events.onclick.nb);
  340. }
  341. break;
  342. }
  343. }
  344. };
  345. pJS.fn.vendors.interactivity.grabParticles = function(p1, p2){
  346. var dx = p1.x - p2.x,
  347. dy = p1.y - p2.y,
  348. dist = Math.sqrt(dx*dx + dy*dy);
  349. var dx_mouse = p1.x - pJS.interactivity.mouse.pos_x,
  350. dy_mouse = p1.y - pJS.interactivity.mouse.pos_y,
  351. dist_mouse = Math.sqrt(dx_mouse*dx_mouse + dy_mouse*dy_mouse);
  352. /* Check distance between 2 particles + Check distance between 1 particle and mouse position */
  353. if(dist <= pJS.particles.line_linked.distance && dist_mouse <= pJS.interactivity.mouse.distance && pJS.interactivity.status == 'mousemove'){
  354. /* Draw the line */
  355. var color_line = pJS.particles.line_linked.color_rgb_line;
  356. pJS.canvas.ctx.beginPath();
  357. pJS.canvas.ctx.strokeStyle = 'rgba('+color_line.r+','+color_line.g+','+color_line.b+','+ (pJS.interactivity.line_linked.opacity-dist_mouse/pJS.interactivity.mouse.distance) +')';
  358. pJS.canvas.ctx.moveTo(p1.x, p1.y);
  359. pJS.canvas.ctx.lineTo(pJS.interactivity.mouse.pos_x, pJS.interactivity.mouse.pos_y);
  360. pJS.canvas.ctx.lineWidth = pJS.particles.line_linked.width;
  361. pJS.canvas.ctx.stroke();
  362. pJS.canvas.ctx.closePath();
  363. }
  364. };
  365. pJS.fn.vendors.destroy = function(){
  366. cancelAnimationFrame(pJS.fn.requestAnimFrame);
  367. canvas_el.remove();
  368. delete pJS;
  369. };
  370. /* --------- LAUNCH ----------- */
  371. function launchParticles(){
  372. pJS.fn.canvasInit();
  373. pJS.fn.canvasSize();
  374. pJS.fn.canvasPaint();
  375. pJS.fn.particlesCreate();
  376. pJS.fn.particlesDraw();
  377. };
  378. function launchAnimation(){
  379. pJS.fn.particlesDraw();
  380. pJS.fn.requestAnimFrame = requestAnimFrame(launchAnimation);
  381. };
  382. launchParticles();
  383. if(pJS.particles.anim.enable){
  384. launchAnimation();
  385. }
  386. if(pJS.interactivity.enable){
  387. pJS.fn.vendors.interactivity.listeners();
  388. }
  389. };
  390. /* --- VENDORS --- */
  391. window.requestAnimFrame = (function(){
  392. return window.requestAnimationFrame ||
  393. window.webkitRequestAnimationFrame ||
  394. window.mozRequestAnimationFrame ||
  395. window.oRequestAnimationFrame ||
  396. window.msRequestAnimationFrame ||
  397. function(callback){
  398. window.setTimeout(callback, 1000 / 60);
  399. };
  400. })();
  401. window.cancelRequestAnimFrame = ( function() {
  402. return window.cancelAnimationFrame ||
  403. window.webkitCancelRequestAnimationFrame ||
  404. window.mozCancelRequestAnimationFrame ||
  405. window.oCancelRequestAnimationFrame ||
  406. window.msCancelRequestAnimationFrame ||
  407. clearTimeout
  408. } )();
  409. function hexToRgb(hex){
  410. // By Tim Down - http://stackoverflow.com/a/5624139/3493650
  411. // Expand shorthand form (e.g. "03F") to full form (e.g. "0033FF")
  412. var shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i;
  413. hex = hex.replace(shorthandRegex, function(m, r, g, b) {
  414. return r + r + g + g + b + b;
  415. });
  416. var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
  417. return result ? {
  418. r: parseInt(result[1], 16),
  419. g: parseInt(result[2], 16),
  420. b: parseInt(result[3], 16)
  421. } : null;
  422. };
  423. /* --- LAUNCH --- */
  424. window.particlesJS = function(tag_id, params){
  425. /* no string id? so it's object params, and set the id with default id */
  426. if(typeof(tag_id) != 'string'){
  427. params = tag_id;
  428. tag_id = 'particles-js';
  429. }
  430. /* no id? set the id to default id */
  431. if(!tag_id){
  432. tag_id = 'particles-js';
  433. }
  434. /* create canvas element */
  435. var canvas_el = document.createElement('canvas');
  436. /* set size canvas */
  437. canvas_el.style.width = "100%";
  438. canvas_el.style.height = "20vw";
  439. /* append canvas */
  440. var canvas = document.getElementById(tag_id).appendChild(canvas_el);
  441. /* launch particle.js */
  442. if(canvas != null){
  443. launchParticlesJS(tag_id, params);
  444. }
  445. };