<div class="title">
  <h3>PLASM.it - 2017</h3>
  <h1>ICOSAHEDRON</h1>
</div>
<div class="more-pens">
  <a target="_blank" href="https://codepen.io/plasm/" class="white-mode">VIEW OTHER PENS</a> <a target="_blank" href="https://codepen.io/collection/nZpPbz/" class="white-mode">VIEW OTHER PARTICLES</a>
</div>
@import url('https://fonts.googleapis.com/css?family=Montserrat:200,300,400,600')
.more-pens
  position: fixed
  left: 20px
  bottom: 20px
  z-index: 10
  font-family: "Montserrat"
  font-size: 12px

a.white-mode, a.white-mode:link, a.white-mode:visited, a.white-mode:active
  font-family: "Montserrat"
  font-size: 12px
  text-decoration: none
  background: #212121
  padding: 4px 8px
  color: #f7f7f7
  &:hover
    background: #edf3f8
    color: #212121

body
  margin: 0
  padding: 0
  overflow: hidden
  width: 100%
  height: 100%
  background: #080926

  
.title
  z-index: 10
  position: absolute
  left: 50%
  top: 50%
  transform: translateX(-50%) translateY(-50%)
  font-family: "Montserrat"
  text-align: center
  width: 100%
  h1
    position: relative
    color: #FFFFFF
    font-weight: 600
    font-size: 60px
    padding: 0
    margin: 0
    line-height: 1

    span
      font-weight: 600
      font-size: 60px
      padding: 0
      margin: 0
      line-height: 1
  h3
    font-weight: 200
    font-size: 20px
    padding: 0
    margin: 0
    line-height: 1
    color: #FFFFFF
    letter-spacing: 2px
canvas
  -webkit-transform: translateZ(1px)
  transform: translateZ(1px)
View Compiled
const media = [""].map(m => PIXI.Texture.from(m))


class Icosaedro{
  constructor(){
    this.progress = 0
    this.container = null
    this.sprite = null
    this.app = null
    this.bunny = null
    this.interval = null
    this.particles = []
    this.init()
  }
  
  init(){
    clearInterval(this.interval)
    this.particles = []
    
    this.app = new PIXI.Application($(window).width(), $(window).height(), {
      antialias: true, 
      transparent: true
    });
    
    let amount = (this.app.renderer instanceof PIXI.WebGLRenderer) ? 100 : 5;
    if(amount == 5){
      this.app.renderer.context.mozImageSmoothingEnabled = false
      this.app.renderer.context.webkitImageSmoothingEnabled = false;
    }

    document.body.appendChild(this.app.view);
    var container = new PIXI.Container();
    this.app.stage.addChild(container);
    this.app.view.style['transform'] = 'translatez(0)'
    this.interval = setInterval(function() {
      this.particles.push(
        new Particle(this.app, container, {
          speed: 1 + Math.random()*4,
          media: this.progress%20 == 1 ? 2 : 0
        })
      )
      this.progress++
    }.bind(this), 20)

  }
  
  update() {
    this.particles = this.particles.filter(function(p) {
      return p.move()
    })
    requestAnimationFrame(this.update.bind(this))
  }
  
  resize(){
    $("canvas").remove()
    this.init()
  }
}



class Particle{
  constructor(app, container, options){
    this.app = app;
    this.container = container
    this.destroy = false
    this.a = [0.5, 1, 1.5 ][Math.round(Math.random()*2)];
    this.steps = $(window).width()/5;
    this.scale = 0.5 * Math.random();
    this.rotation = [0.02,0.04,0.06,0.08][Math.round(Math.random()*3)];
    this.siner = 300 * Math.random();
    this.speed = options.speed;
    this.progress = 0;
    
    //this.texture = new PIXI.extras.AnimatedSprite(media);
    this.texture = new PIXI.Sprite(media[0]);
    let n = [0.344,0.4346, 0.7444, 0.992222][Math.round(Math.random()*3)]
    this.texture.tint = n * 0xFFFFFF;
    this.texture.cacheAsBitmap = true
    this.texture.anchor.set(0.5);
    //this.texture.animationSpeed = 0.5;
    //this.texture.play();
    
    this.behaviours()
    this.render();
    this.move();
  }
  
  behaviours(){
    $(window).on("resize", function(){
      this.destroy = true;
    }.bind(this))
  }
  
  render(){
    this.container.addChild(this.texture);
  }
  
  move(){
    let x = this.progress;
    let y = this.app.renderer.height/6 + this.siner +  (this.a*Math.sin(this.progress/this.steps))*130;

    this.texture.x = -50 + x
    this.texture.y = y
    this.texture.scale.x = this.scale
    this.texture.scale.y = this.scale
    this.texture.rotation += this.rotation;
    this.progress = this.progress+this.speed
    
      //Destroy
    if (x > this.app.renderer.width + 100 || this.destroy == true) {
      this.container.removeChild(this.texture);
      return false
    } else {
      return true
    }
  }
  
  radians(degrees) {
    return degrees * Math.PI / 180;
  }
}
function debounce(func, wait, immediate) {
  var timeout;
  return function() {
    var context = this, args = arguments;
    var later = function() {
      timeout = null;
      if (!immediate) func.apply(context, args);
    };
    var callNow = immediate && !timeout;
    clearTimeout(timeout);
    timeout = setTimeout(later, wait);
    if (callNow) func.apply(context, args);
  };
};


let icosaedro = new Icosaedro()
icosaedro.update()

$(window).on("resize", debounce(function(){
  icosaedro.resize()
}.bind(this), 200))
View Compiled
Run Pen

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

  1. https://cdnjs.cloudflare.com/ajax/libs/jquery/3.1.1/jquery.min.js
  2. https://cdnjs.cloudflare.com/ajax/libs/pixi.js/4.4.2/pixi.min.js