sketch.js
function makePieces(records) {
var pieces = [];
for (var i = 0; i < records.length; i++) {
var r = records[i];
var x1 = r[0] * 1;
var y1 = r[1] * 1;
var x2 = r[2] * 1;
var y2 = r[3] * 1;
var a = r[4];
var t = r[5];
pieces.push(new Piece(pieces, x1, y1, x2, y2, a, t));
}
return pieces;
}
var Piece = function(parent, x1, y1, x2, y2, acceleration, duration) {
if (Array.isArray(parent)) {
parent = parent[parent.length - 1];
}
this.x1 = x1;
this.y1 = y1;
this.x2 = x2;
this.y2 = y2;
this.acceleration = acceleration;
this.duration = duration;
this.length = dist(x1, y1, x2, y2);
if (parent) {
this.t0 = parent.t1;
this.t1 = this.t0 + duration;
this.v0 = parent.v(this.t0);
this.x0 = parent.x(this.t0);
} else {
this.t0 = 0;
this.t1 = duration;
this.v0 = 0;
this.x0 = 0;
}
}
Piece.prototype.hasT = function(t) {
return t >= this.t0 && t < this.t1;
}
Piece.prototype.a = function(t) {
return this.acceleration;
}
Piece.prototype.v = function(t) {
t -= this.t0;
return this.v0 + this.acceleration * t;
}
Piece.prototype.x = function(t) {
t -= this.t0;
t = min(t, this.duration);
return this.x0 + this.v0 * t + this.acceleration * t * t / 2;
}
function evaluate(pieces, n, t) {
for (var i = 0; i < pieces.length; i++) {
if (pieces[i].hasT(t) || i == pieces.length - 1) {
switch (n) {
case 0: return pieces[i].x(t);
case 1: return pieces[i].v(t);
case 2: return pieces[i].a(t);
default: return 0;
}
}
}
return 0;
}
function computeBounds(pieces) {
var p = pieces[0];
var minx = p.x1;
var maxx = p.x1;
var miny = p.y1;
var maxy = p.y1;
for (var i = 0; i < pieces.length; i++) {
var piece = pieces[i];
minx = min(minx, piece.x1, piece.x2);
maxx = max(maxx, piece.x1, piece.x2);
miny = min(miny, piece.y1, piece.y2);
maxy = max(maxy, piece.y1, piece.y2);
}
var width = maxx - minx;
var height = maxy - miny;
return {
minx: minx, miny: miny, maxx: maxx, maxy: maxy,
width: width, height: height,
}
}
function drawPiece(dc, piece) {
dc.vertex(piece.x1, piece.y1);
dc.vertex(piece.x2, piece.y2);
}
function drawPieceLength(dc, piece, length) {
var x1 = piece.x1;
var y1 = piece.y1;
var x2 = lerp(piece.x1, piece.x2, length / piece.length);
var y2 = lerp(piece.y1, piece.y2, length / piece.length);
dc.vertex(x1, y1);
dc.vertex(x2, y2);
}
function drawPieces(dc, pieces, length) {
dc.beginShape();
for (var i = 0; i < pieces.length; i++) {
if (length <= 0) {
break;
}
var piece = pieces[i];
if (piece.length < length) {
drawPiece(dc, piece);
} else {
drawPieceLength(dc, piece, length);
}
length -= piece.length;
}
dc.endShape();
}
function Drawing(pieces) {
this.pieces = pieces;
this.bounds = computeBounds(pieces);
}
Drawing.prototype.setTransform = function(dc) {
var bounds = this.bounds;
var sx = dc.width / (bounds.width + 40);
var sy = dc.height / (bounds.height + 40);
var s = min(sx, sy);
var dx = (dc.width - bounds.width * s) / 2;
var dy = (dc.height - bounds.height * s) / 2;
dc.resetMatrix();
dc.translate(dx, dy);
dc.scale(s);
dc.translate(-bounds.minx, -bounds.miny);
}
Drawing.prototype.draw = function(dc, t) {
var length = evaluate(this.pieces, 0, t);
drawPieces(window, this.pieces, length);
}
var drawing;
function setup() {
drawing = new Drawing(makePieces(PIECES));
createCanvas(windowWidth, windowHeight);
restart();
}
function windowResized() {
resizeCanvas(windowWidth, windowHeight);
restart();
}
function setStyle(dc) {
dc.stroke(0);
dc.strokeWeight(1);
dc.strokeJoin(ROUND);
dc.noFill();
}
function restart() {
background(255);
}
function draw() {
var time = millis() / 1000;
var distance = evaluate(drawing.pieces, 0, time);
var velocity = evaluate(drawing.pieces, 1, time);
var acceleration = evaluate(drawing.pieces, 2, time);
var lastPiece = drawing.pieces[drawing.pieces.length - 1];
var maxTime = lastPiece.t1;
var maxDistance = lastPiece.x(maxTime);
var maxVelocity = 250;
var maxAcceleration = 100;
background(255);
setStyle(window);
drawing.setTransform(window);
drawing.draw(window, time);
resetMatrix();
noStroke();
fill(0, 0, 0);
rect(0, 0, time / maxTime * width, 10);
fill(255, 0, 0);
rect(0, height - 10, distance / maxDistance * width, 10);
fill(0, 255, 0);
rect(0, height - 20, velocity / maxVelocity * width, 10);
fill(0, 0, 255);
var w = acceleration / maxAcceleration * width / 2;
rect(width / 2, height - 30, w, 10);
}