Daily Creative Coding

元「30 min. Processing」。毎日、Creative Codingします。

降る雪

/**
* snowfall
*
* @author aa_debdeb
* @date 2016/08/16
*/

float MIN_PARTICLE_DIAMETER = 2;
float MAX_PARTICLE_DIAMETER = 7;
float MAX_PARTICLE_VELOCITY_X = 0.5;
float MAX_PARTICLE_VEL_VARIANT_X = 0.1;
float MAX_PARTICLE_VELOCITY_Y = 0.5;
float MAX_PARTICLE_VEL_VARIANT_Y = 0.1;
float MAX_BLINK_SPEED = PI / 128;

ArrayList<Particle> particles;

void setup(){
  size(640, 480);
  noStroke();
  particles = new ArrayList<Particle>();
  background(0);
}

void draw(){
  fill(0, 70);
  rect(0, 0, width, height);
  ArrayList<Particle> nextParticles = new ArrayList<Particle>();
  for(Particle p: particles){
    p.display();
    p.update();
    if(!p.isGoneOut()){
      nextParticles.add(p);
    }
  }
  particles = nextParticles;
  if(random(1) < 0.3){
    particles.add(new Particle());
  }
}

class Particle{
  
  PVector loc, vel;
  float blinkAngle;
  
  Particle(){
    loc = new PVector(random(width), - MAX_PARTICLE_DIAMETER / 2);
    vel = new PVector(random(-MAX_PARTICLE_VELOCITY_X, MAX_PARTICLE_VELOCITY_X), random(MAX_PARTICLE_VELOCITY_Y));
    blinkAngle = random(TWO_PI);
  }
  
  void display(){
    float diameter = map(sin(blinkAngle), -1, 1, MIN_PARTICLE_DIAMETER, MAX_PARTICLE_DIAMETER);
    for(float v = 1.0; v >= 0.0; v -= 0.2){
      float d = map(v, 0, 1, 0, diameter);
      float r = map(pow(v, 5), 0, 1, 255, 128);
      float g = map(pow(v, 5), 0, 1, 255, 255);
      float b = map(pow(v, 5), 0, 1, 255, 255);
      fill(r, g, b, 30);
      ellipse(loc.x, loc.y, d, d);
    }
  }
  
  void update(){
    vel.x += random(-MAX_PARTICLE_VEL_VARIANT_X, MAX_PARTICLE_VEL_VARIANT_X);
    vel.x = constrain(vel.x, -MAX_PARTICLE_VELOCITY_X, MAX_PARTICLE_VELOCITY_X);
    vel.y += random(-MAX_PARTICLE_VEL_VARIANT_Y, MAX_PARTICLE_VEL_VARIANT_Y);
    vel.y = constrain(vel.y, 0, MAX_PARTICLE_VELOCITY_Y);
    loc.add(vel);
    blinkAngle += random(MAX_BLINK_SPEED);
    if(blinkAngle >= TWO_PI){
      blinkAngle -= TWO_PI;
    }
  }
  
  boolean isGoneOut(){
    return loc.y > height + MAX_PARTICLE_DIAMETER / 2;
  }
  
}
f:id:aa_debdeb:20160809083417j:plain