Blob Blame Raw

#include <stdlib.h>
#include <math.h>
#include <helianthus.h>




int seed;
double t;

const double segmentLength = 5;
const double segmentWidth = 0.05;
const double minWidth = 0.5;
const double branchesPerSegment = 0.2;
const double leafSize = 200;
const double leafPerBranch = 1;
const double leafThreshold = 1 + 200/5*0.05/4*0.5;


double randomRange(double min, double max)
  { return randomFloat()*(max - min) + min; }

double randomIncrement(double deviation)
  { return randomRange(-deviation, deviation); }

double randomAmplifier(double deviation)
  { return randomRange(1 - deviation, 1 + deviation); }


void leaf(double x, double y, double size) {
  double r = 0.5*size;//*randomAmplifier(0.5);
  saveState();
  noStroke();
  circle(x, y, r);
  restoreState();
}


void branch(double x, double y, double angle, double width) {
  double da = sin(t*randomRange(0.5, 1) + randomIncrement(PI));

  saveState();
  while(TRUE) {
    double length = segmentLength * randomAmplifier(0.1);

    double s = sin(angle/180*PI);
    double c = cos(angle/180*PI);
    double nx = x + c*length;
    double ny = y + s*length;
    double nw = width - segmentWidth;
    strokeWidth(width);
    line(x, y, nx, ny);

    if (width > leafThreshold && nw <= leafThreshold && randomFloat() < leafPerBranch && y > leafSize/2) {
      double dl = randomRange(0, length);
      leaf(x + c*dl, y + s*dl, leafSize);
    }

    if (width < minWidth || ny < 0)
      break;

    x = nx;
    y = ny;
    s += randomRange(0, 0.2);
    c += randomIncrement(0.15);
    angle = atan2(s, c)/PI*180;
    width -= segmentWidth;

    translate(x, y);
    rotate(da*pow(minWidth/width, 2));
    translate(-x, -y);

    if (randomFloat() < branchesPerSegment) {
      double k = randomRange(0.1, 0.3);
      double w = sqrt(width*width*k);
      if (w > minWidth) {
        width = sqrt(width*width*(1-k));
        double a = randomNumber(0, 1) ? angle - 90 + randomRange(0, 45)
                                      : angle + 90 - randomRange(0, 45);
        branch(x, y, a, w);
      }
    }
  }
  restoreState();
}


void tree(int seed, double x, double y, double width) {
  srand(seed);
  double angle = 90 + randomIncrement(30);
  width *= randomAmplifier(0.5);
  int s = randomNumber(0, 1000000);

  saveState();
  noStroke();
  srand(s);
  branch(x, y, angle, width);
  restoreState();

  saveState();
  noFill();
  srand(s);
  branch(x, y, angle, width);
  restoreState();
}


void init() {
  seed = randomNumber(0, 1000000);
}


void draw() {
  double w = worldGetWidth();
  double h = worldGetHeight();
  t += worldGetFrameTime();

  if (keyWentDown("space")) seed = randomNumber(0, 1000000);

  saveState();
  translate(w/2, h*0.9);
  scale(1, -1);

  fill(colorByName("green"));
  stroke(colorByName("black"));
  tree(seed, 0, 0, 20);

  restoreState();
}


int main() {
  worldSetVariableFrameRate();
  worldSetResizable(TRUE);
  worldSetInit(&init);
  worldSetDraw(&draw);
  worldRun();
}