header#header
  h1#title
  #controls
    .btn#btn-wave
      i.fa.fa-exchange
    .btn#btn-lights
      i.fa.fa-sun-o
    .btn#btn-prev
      i.fa.fa-backward
    .btn#btn-play
      i.fa.fa-play
    .btn#btn-next
      i.fa.fa-forward
    .btn#btn-volume
      i.fa.fa-volume-up
#background
#loader
  #spinner
  h2#loading-text Loading...
canvas#glow-layer
canvas#canvas
audio#audio
#progress-bar
a.make-google-happy-button Click to Start
a#codepen-link(href='https://www.codepen.io/seanfree' target='_blank')
View Compiled
body
  background-color: #030303
  overflow: hidden

#header
  position: fixed
  z-index: 100
  width: 100%
  padding: 20px
  color: #69cdd7
  display: flex
  align-items: center
  justify-content: space-between
  flex-direction: column
  box-sizing: border-box
  
#title
  font-family: 'Josefin Sans', sans-serif
  font-size: 1.2em
  text-align: center
  
#controls
  display: flex
  padding: 15px
  .btn
    position: relative
    margin: 0 10px
    cursor: pointer
    transition: opacity 0.5s ease-in-out
    .fa
      width: 16px
      pointer-events: none
      &.off
        color: #efefef
        opacity: 0.4
    &:active
      transform: scale(0.9)
    &.disabled
      pointer-events: none
      opacity: 0.2
#flex-pad
  flex: 1

#background
  position: absolute
  top: 0
  left: 0
  z-index: 0
  width: 100vw
  height: 100vh
  background-image: radial-gradient(rgba(105,205,215,0.5),rgba(10,20,30,0.5))
  background-position: 100% 100%
  background-size: 200% 200%
  opacity: 1
  transition: opacity 2s ease-in-out
  
  &.hidden
    opacity: 0
    
  &.loading
    opacity: 0.2
    
    + #loader
      opacity: 1
    
      #spinner
        animation: spinner-anim 2s ease-out infinite
        
        &::before
          animation: spinner-anim 2s ease-out infinite
        
        &::after
          animation: spinner-anim 1s ease-out infinite
        
#loader
  position: absolute
  bottom: 20px
  left: 20px
  z-index: 3
  display: flex
  align-items: center
  justify-content: flex-start
  width: 100vw
  opacity: 0
  transition: opacity 0.5s ease-in-out
  
  #loading-text
    font-size: 1.8em
    font-family: 'Josefin Sans', sans-serif
    color: #69cdd7
    transition: all 0.5s ease-in-out
    
  #spinner
    height: 40px
    width: 40px
    border: 2px solid #efefef
    border-color: transparent transparent #69cdd7 #69cdd7
    border-radius: 50%
    margin-right: 10px
    transform: rotate(0deg)
    
    &::before, &::after
      position: absolute
      content: ''
      display: block
      border: 2px solid #efefef
      border-color: transparent transparent #69cdd7 #69cdd7
      border-radius: 50%
    
    &::before
      top: 3px
      left: 3px
      height: 30px
      width: 30px
    
    &::after
      top: 8px
      left: 8px
      height: 20px
      width: 20px
    
#canvas, #glow-layer
  position: absolute
  top: 0
  left: 0
  overflow: hidden
  height: 100vh
  width: 100vw

#canvas
  z-index: 1

#glow-layer
  z-index: 0
  
#codepen-link
  position: absolute
  bottom: 20px
  right: 20px
  height: 40px
  width: 40px
  z-index: 10
  border-radius: 50%
  box-sizing: border-box
  background-image: url('https://s3-us-west-2.amazonaws.com/s.cdpn.io/544318/logo.jpg')
  background-position: center center
  background-size: cover
  opacity: 0.4
  transition: all 0.25s
  
  &:hover
    opacity: 0.8
    box-shadow: 0 0 6px #efefef
  
#progress-bar
  position: absolute
  left: 0
  top: 0
  z-index: 3
  width: 100%
  height: 4px
  background: #69cdd7
  transform: scaleX(0)
  transform-origin: center left
  transition: transform 0.5s linear

.make-google-happy-button
  position: absolute
  padding: 1rem 1rem 0.8rem 1rem
  top: 50%
  left: 50%
  transform: translateX(-50%) translateY(-50%)
  z-index: 4
  border: 1px solid #69cdd7
  color: #69cdd7
  font-family: 'Josefin Sans', sans-serif
  font-size: 1.2em
  cursor: pointer
  transition: all 0.5s

  &:hover
    background: #69cdd7
    color: #333

  &.hidden
    opacity: 0
    transform: translateX(-50%) translateY(-50%)
  
@media screen and (min-width: 600px)
  #header
    flex-direction: row
  #controls
    padding: 0
  
@keyframes spinner-anim
  to
    transform: rotate(360deg)
View Compiled
class Particle {
  constructor(index, parent) {
    this.index = index;
    this.parent = parent;
    this.minSize = 5;
    this.init();
  }
  init() {
    this.freqVal = this.parent.freqData[this.index] * 0.01;
    this.size =
      this.freqVal *
        ((this.parent.dimensions.x + this.parent.dimensions.y) * 0.5) *
        0.0125 +
      this.minSize;
    this.position = new Vector2(
      Math.random() * this.parent.dimensions.x,
      this.parent.dimensions.y + this.size + Math.random() * 50
    );
    this.velocity = new Vector2(2 - Math.random() * 4, 0);
  }
  update() {
    this.freqVal = this.parent.freqData[this.index] * 0.01;

    this.size = this.freqVal * 20 + this.minSize;

    this.hue =
      this.index / this.parent.analyser.frequencyBinCount * 360 + 120 + this.parent.tick / 6;
    this.saturation = this.freqVal * 50;
    this.alpha = this.freqVal * 0.3;

    this.fill = `hsla(${this.hue}, ${this.saturation}%, 50%, ${this.alpha})`;
    this.lift = Math.pow(this.freqVal, 3);

    this.position.subY(this.lift + 0.5);
    this.position.add(this.velocity);

    this.checkBounds();
  }
  checkBounds() {
    if (
      this.position.y < -this.size ||
      this.position.x < -this.parent.dimensions.x * 0.15 ||
      this.position.x > this.parent.dimensions.x * 1.15
    ) {
      this.init();
    }
  }
}

class App {
  constructor() {
    this.globalMovement = new Vector2();
    this.initCanvas();
    this.initUI();
    this.initAudio();
    this.btnMakeGoogleHappy = document.querySelector(".make-google-happy-button");
    this.btnMakeGoogleHappy.addEventListener("click", () => {
      this.audioCtx.resume();
      this.loadAudio();
      this.btnMakeGoogleHappy.classList.add("hidden");
    });
    this.populate();
    this.render();
    window.onresize = () => {
      this.resize();
    };
  }
  initCanvas() {
    this.tick = 0;
    this.dark = false;
    this.wave = true;
    this.canvas = document.getElementById("canvas");
    this.ctx = this.canvas.getContext("2d");
    this.glowLayer = document.getElementById("glow-layer");
    this.glowCtx = this.glowLayer.getContext("2d");
    this.dimensions = {};
    this.resize();
  }
  resize() {
    this.canvas.width = this.glowLayer.width = this.dimensions.x = window.innerWidth;
    this.canvas.height = this.glowLayer.height = this.dimensions.y = window.innerHeight;
  }
  initUI() {
    this.progressBar = document.querySelector("#progress-bar");
    this.controls = {
      wave: document.querySelector("#btn-wave"),
      lights: document.querySelector("#btn-lights"),
      prev: document.querySelector("#btn-prev"),
      next: document.querySelector("#btn-next"),
      play: document.querySelector("#btn-play"),
      volume: document.querySelector("#btn-volume")
    };
    this.controls.wave.onclick = () => {
      let i = this.controls.wave.getElementsByTagName("i")[0];
      if (this.wave) {
        i.classList.add("off");
        this.wave = false;
      } else if (!this.wave) {
        i.classList.remove("off");
        this.wave = true;
      }
    };
    this.controls.lights.onclick = () => {
      let i = this.controls.lights.getElementsByTagName("i")[0];
      if (this.dark) {
        i.classList.remove("off");
        this.background.classList.remove("hidden");
        this.dark = false;
      } else if (!this.dark) {
        i.classList.add("off");
        this.background.classList.add("hidden");
        this.dark = true;
      }
    };
    this.controls.prev.onclick = () => {
      this.currentSong = this.currentSong > 1
        ? this.currentSong - 1
        : this.fileNames.length;
      this.loadAudio();
    };
    this.controls.next.onclick = () => {
      this.currentSong = this.currentSong < this.fileNames.length
        ? this.currentSong + 1
        : 1;
      this.loadAudio();
    };
    this.controls.play.onclick = () => {
      let i = this.controls.play.getElementsByTagName("i")[0];
      if (this.playing && this.audioReady) {
        i.classList.remove("fa-pause");
        i.classList.add("fa-play");
        this.playing = false;
        this.audio.pause();
      } else if (!this.playing && this.audioReady) {
        i.classList.remove("fa-play");
        i.classList.add("fa-pause");
        this.playing = true;
        this.audio.play();
      }
    };
    this.controls.volume.onclick = () => {
      let i = this.controls.volume.getElementsByTagName("i")[0];
      this.volume = this.volume > 0 ? this.volume - 0.5 : 1;
      switch (this.volume) {
        case 1:
          i.classList.remove("fa-volume-off");
          i.classList.add("fa-volume-up");
          break;
        case 0.5:
          i.classList.remove("fa-volume-up");
          i.classList.add("fa-volume-down");
          break;
        case 0:
          i.classList.remove("fa-volume-down");
          i.classList.add("fa-volume-off");
          break;
        default:
          break;
      }
      this.gainNode.gain.value = this.volume;
    };
    this.background = document.getElementById("background");
    this.title = document.getElementById("title");
  }
  initAudio() {
    this.currentSong = 1;
    this.volume = 1;
    this.baseURL = "https://res.cloudinary.com/sf-cloudinary/video/upload/v1525440046/";
    this.fileNames = [
      "dmwaltz.mp3",
      "nocturne92.mp3",
      "mozart25.mp3",
      "trista.mp3",
      "waltzflowers.mp3"
    ];
    this.songTitles = [
      "Dmitri Shostakovich - Waltz No. 2",
      "Frederic Chopin - Nocturne op. 9 no. 2",
      "Mozart - Symphony no. 25",
      "Heitor Villa-Lobos - Tristorosa",
      "Pyotr Tchaikovsky - Waltz of the Flowers"
    ];

    this.audio = document.getElementById("audio");
    this.audio.addEventListener("ended", () => {
      this.audio.currentTime = 0;
      this.audio.pause();
      this.currentSong = this.currentSong < this.fileNames.length
        ? this.currentSong + 1
        : 1;
      this.loadAudio();
    });
    this.audio.addEventListener("timeupdate", () => {
      this.progressBar.style = `transform: scaleX(${this.audio.currentTime / this.audio.duration})`;
    });
    
    this.audioCtx = new AudioContext();
    
    this.source = this.audioCtx.createMediaElementSource(this.audio);
    this.gainNode = this.audioCtx.createGain();
    
    this.analyser = this.audioCtx.createAnalyser();
    this.analyser.smoothingTimeConstant = 0.92;
    this.analyser.fftSize = 2048;
    this.analyser.minDecibels = -125;
    this.analyser.maxDecibels = -10;

    this.source.connect(this.gainNode);
    this.gainNode.connect(this.analyser);
    this.analyser.connect(this.audioCtx.destination);

    this.gainNode.gain.value = this.volume;
    this.freqData = new Uint8Array(this.analyser.frequencyBinCount);
  }
  loadAudio() {
    let request = new XMLHttpRequest();

    this.audioReady = false;
    this.playing = false;
    this.background.classList.add("loading");

    this.controls.prev.classList.add("disabled");
    this.controls.next.classList.add("disabled");
    this.controls.play.classList.add("disabled");

    request.open(
      "GET",
      this.baseURL + this.fileNames[this.currentSong - 1],
      true
    );
    request.responseType = "blob";

    request.onprogress = () => {
      if (request.response)
        this.playAudio(request.response);
    };

    request.send();
  }
  playAudio(data) {
    this.audioReady = true;
    this.playing = true;

    this.background.classList.remove("loading");
    this.title.innerHTML = this.songTitles[this.currentSong - 1];

    this.controls.prev.classList.remove("disabled");
    this.controls.next.classList.remove("disabled");
    this.controls.play.classList.remove("disabled");

    this.controls.play.getElementsByTagName("i")[0].classList.remove("fa-play");
    this.controls.play.getElementsByTagName("i")[0].classList.add("fa-pause");

    this.audio.src = window.URL.createObjectURL(data);
    this.audio.play();
  }
  populate() {
    this.particles = [];
    for (let i = 0; i < 625; i++) {
      this.particles.push(new Particle(i, this));
    }
  }
  update() {
    this.ctx.clearRect(0, 0, this.dimensions.x, this.dimensions.y);
    this.ctx.save();
    this.ctx.globalCompositeOperation = "lighten";
    for (let i = this.particles.length - 1; i >= 0; i--) {
      let particle = this.particles[i];
      if (this.freqData[i] > 0) {
        particle.update();
        if (this.wave) particle.position.add(this.globalMovement);
        this.ctx.beginPath();
        this.ctx.fillStyle = particle.fill;
        this.ctx.beginPath();
        this.ctx.arc(
          particle.position.x,
          particle.position.y,
          particle.size,
          0,
          2 * Math.PI
        );
        this.ctx.fill();
        this.ctx.closePath();
      }
    }
    this.ctx.restore();
    this.glowCtx.clearRect(0, 0, this.dimensions.x, this.dimensions.y);
    this.glowCtx.filter = "blur(8px) saturate(150%) brightness(150%)";
    this.glowCtx.drawImage(this.canvas, 0, 0);
  }
  render() {
    this.tick++;
    if (this.wave) this.globalMovement.x = Math.sin(this.tick * 0.01) * 2;
    this.analyser.getByteFrequencyData(this.freqData);
    this.update();
    window.requestAnimationFrame(this.render.bind(this));
  }
}

window.requestAnimationFrame = (() => {
  return (
    window.requestAnimationFrame ||
    window.webkitRequestAnimationFrame ||
    window.mozRequestAnimationFrame ||
    window.oRequestAnimationFrame ||
    window.msRequestAnimationFrame ||
    function(callback) {
      window.setTimeout(callback, 1000 / 60);
    }
  );
})();

window.onload = () => {
  let app = new App();
};
View Compiled

External CSS

  1. https://maxcdn.bootstrapcdn.com/font-awesome/4.7.0/css/font-awesome.min.css

External JavaScript

  1. https://cdn.jsdelivr.net/gh/SeanFree/Vector2@master/Vector2.min.js
  2. https://cdnjs.cloudflare.com/ajax/libs/dat-gui/0.6.1/dat.gui.min.js