trail.js 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. /**
  2. * With love.
  3. * http://hakim.se/experiments/
  4. * http://twitter.com/hakimel
  5. */
  6. var SCREEN_WIDTH = window.innerWidth;
  7. var SCREEN_HEIGHT = window.innerHeight;
  8. var RADIUS = 60;
  9. var RADIUS_SCALE = 1;
  10. var RADIUS_SCALE_MIN = 1;
  11. var RADIUS_SCALE_MAX = 1.5;
  12. // The number of particles that are used to generate the trail
  13. var QUANTITY = 25;
  14. var canvas;
  15. var context;
  16. var particles;
  17. var mouseX = SCREEN_WIDTH * 0.5;
  18. var mouseY = SCREEN_HEIGHT * 0.5;
  19. var mouseIsDown = false;
  20. init();
  21. function init() {
  22. canvas = document.getElementById( 'world' );
  23. if (canvas && canvas.getContext) {
  24. context = canvas.getContext('2d');
  25. document.addEventListener('mousemove', documentMouseMoveHandler, false);
  26. document.addEventListener('mousedown', documentMouseDownHandler, false);
  27. document.addEventListener('mouseup', documentMouseUpHandler, false);
  28. document.addEventListener('touchstart', documentTouchStartHandler, false);
  29. document.addEventListener('touchmove', documentTouchMoveHandler, false);
  30. window.addEventListener('resize', windowResizeHandler, false);
  31. createParticles();
  32. windowResizeHandler();
  33. setInterval( loop, 1000 / 60 );
  34. }
  35. }
  36. function createParticles() {
  37. particles = [];
  38. for (var i = 0; i < QUANTITY; i++) {
  39. var particle = {
  40. position: { x: mouseX, y: mouseY },
  41. shift: { x: mouseX, y: mouseY },
  42. size: 1,
  43. angle: 0,
  44. speed: 0.01+Math.random()*0.04,
  45. targetSize: 1,
  46. fillColor: '#' + (Math.random() * 0x404040 + 0xaaaaaa | 0).toString(16),
  47. orbit: RADIUS*.5 + (RADIUS * .5 * Math.random())
  48. };
  49. particles.push( particle );
  50. }
  51. }
  52. function documentMouseMoveHandler(event) {
  53. mouseX = event.clientX;
  54. mouseY = event.clientY;
  55. }
  56. function documentMouseDownHandler(event) {
  57. mouseIsDown = true;
  58. }
  59. function documentMouseUpHandler(event) {
  60. mouseIsDown = false;
  61. }
  62. function documentTouchStartHandler(event) {
  63. if(event.touches.length == 1) {
  64. event.preventDefault();
  65. mouseX = event.touches[0].pageX - (window.innerWidth - SCREEN_WIDTH) * .5;
  66. mouseY = event.touches[0].pageY - (window.innerHeight - SCREEN_HEIGHT) * .5;
  67. }
  68. }
  69. function documentTouchMoveHandler(event) {
  70. if(event.touches.length == 1) {
  71. event.preventDefault();
  72. mouseX = event.touches[0].pageX - (window.innerWidth - SCREEN_WIDTH) * .5;
  73. mouseY = event.touches[0].pageY - (window.innerHeight - SCREEN_HEIGHT) * .5;
  74. }
  75. }
  76. function windowResizeHandler() {
  77. SCREEN_WIDTH = window.innerWidth;
  78. SCREEN_HEIGHT = window.innerHeight;
  79. canvas.width = SCREEN_WIDTH;
  80. canvas.height = SCREEN_HEIGHT;
  81. }
  82. function loop() {
  83. context.shadowBlur = 3;
  84. if( mouseIsDown ) {
  85. // Scale upward to the max scale
  86. RADIUS_SCALE += ( RADIUS_SCALE_MAX - RADIUS_SCALE ) * (0.02);
  87. }
  88. else {
  89. // Scale downward to the min scale
  90. RADIUS_SCALE -= ( RADIUS_SCALE - RADIUS_SCALE_MIN ) * (0.02);
  91. }
  92. RADIUS_SCALE = Math.min( RADIUS_SCALE, RADIUS_SCALE_MAX );
  93. for (i = 0, len = particles.length; i < len; i++) {
  94. var particle = particles[i];
  95. // Rotation
  96. particle.angle += particle.speed;
  97. // Follow mouse with some lag
  98. particle.shift.x += ( mouseX - particle.shift.x) * (particle.speed);
  99. particle.shift.y += ( mouseY - particle.shift.y) * (particle.speed);
  100. // Apply position
  101. particle.position.x = particle.shift.x + Math.cos(i + particle.angle) * (particle.orbit*RADIUS_SCALE);
  102. particle.position.y = particle.shift.y + Math.sin(i + particle.angle) * (particle.orbit*RADIUS_SCALE);
  103. // Limit to screen bounds
  104. particle.position.x = Math.max( Math.min( particle.position.x, SCREEN_WIDTH ), 0 );
  105. particle.position.y = Math.max( Math.min( particle.position.y, SCREEN_HEIGHT ), 0 );
  106. particle.size += ( particle.targetSize - particle.size ) * 0.05;
  107. if( Math.round( particle.size ) == Math.round( particle.targetSize ) ) {
  108. particle.targetSize = 1 + Math.random() * 7;
  109. }
  110. context.beginPath();
  111. context.fillStyle = particle.fillColor;;
  112. context.moveTo(particle.position.x, particle.position.y);
  113. context.arc(particle.position.x, particle.position.y, particle.size, 0, Math.PI*2, true);
  114. context.fill();
  115. }
  116. }