2011-06-30 07:55:35 -04:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
2013-06-02 16:41:10 -04:00
|
|
|
<meta charset="UTF-8">
|
2011-06-30 09:57:17 -04:00
|
|
|
<title>Future Splash</title>
|
2011-06-30 07:55:35 -04:00
|
|
|
<link rel="stylesheet" href="../css/style.css">
|
|
|
|
<script type="text/javascript" src="../../dist/paper.js"></script>
|
|
|
|
<script type="text/paperscript" canvas="canvas">
|
|
|
|
// Code ported to Paper.js from http://the389.com/9/1/
|
|
|
|
// with permission.
|
|
|
|
|
|
|
|
var values = {
|
|
|
|
friction: 0.8,
|
|
|
|
timeStep: 0.01,
|
|
|
|
amount: 15,
|
|
|
|
mass: 2,
|
|
|
|
count: 0
|
|
|
|
};
|
|
|
|
values.invMass = 1 / values.mass;
|
|
|
|
|
2013-03-09 17:37:56 -05:00
|
|
|
var path, springs;
|
2011-06-30 07:55:35 -04:00
|
|
|
var size = view.size * [1.2, 1];
|
|
|
|
|
2013-03-09 17:37:56 -05:00
|
|
|
var Spring = function(a, b, strength, restLength) {
|
|
|
|
this.a = a;
|
|
|
|
this.b = b;
|
|
|
|
this.restLength = restLength || 80;
|
|
|
|
this.strength = strength ? strength : 0.55;
|
|
|
|
this.mamb = values.invMass * values.invMass;
|
|
|
|
};
|
|
|
|
|
|
|
|
Spring.prototype.update = function() {
|
|
|
|
var delta = this.b - this.a;
|
|
|
|
var dist = delta.length;
|
|
|
|
var normDistStrength = (dist - this.restLength) /
|
|
|
|
(dist * this.mamb) * this.strength;
|
|
|
|
delta.y *= normDistStrength * values.invMass * 0.2;
|
|
|
|
if (!this.a.fixed)
|
|
|
|
this.a.y += delta.y;
|
|
|
|
if (!this.b.fixed)
|
|
|
|
this.b.y -= delta.y;
|
|
|
|
};
|
2011-07-07 10:09:02 -04:00
|
|
|
|
2011-06-30 07:55:35 -04:00
|
|
|
|
|
|
|
function createPath(strength) {
|
2013-03-02 14:54:52 -05:00
|
|
|
var path = new Path({
|
|
|
|
fillColor: 'black'
|
|
|
|
});
|
2013-03-09 17:37:56 -05:00
|
|
|
springs = [];
|
2011-06-30 07:55:35 -04:00
|
|
|
for (var i = 0; i <= values.amount; i++) {
|
|
|
|
var segment = path.add(new Point(i / values.amount, 0.5) * size);
|
|
|
|
var point = segment.point;
|
2013-03-09 17:37:56 -05:00
|
|
|
if (i == 0 || i == values.amount)
|
|
|
|
point.y += size.height;
|
2011-06-30 07:55:35 -04:00
|
|
|
point.px = point.x;
|
|
|
|
point.py = point.y;
|
2013-03-09 17:37:56 -05:00
|
|
|
// The first two and last two points are fixed:
|
|
|
|
point.fixed = i < 2 || i > values.amount - 2;
|
|
|
|
if (i > 0) {
|
|
|
|
var spring = new Spring(segment.previous.point, point, strength);
|
|
|
|
springs.push(spring);
|
2011-06-30 07:55:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
path.position.x -= size.width / 4;
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2011-06-30 08:01:18 -04:00
|
|
|
function onResize() {
|
2013-03-09 17:37:56 -05:00
|
|
|
if (path)
|
2011-06-30 08:01:18 -04:00
|
|
|
path.remove();
|
|
|
|
size = view.bounds.size * [2, 1];
|
|
|
|
path = createPath(0.1);
|
|
|
|
}
|
|
|
|
|
2011-06-30 07:55:35 -04:00
|
|
|
function onMouseMove(event) {
|
2013-03-09 17:37:56 -05:00
|
|
|
var location = path.getNearestLocation(event.point);
|
|
|
|
var segment = location.segment;
|
2011-06-30 07:55:35 -04:00
|
|
|
var point = segment.point;
|
|
|
|
|
2013-03-09 17:37:56 -05:00
|
|
|
if (!point.fixed && location.distance < size.height / 4) {
|
|
|
|
var y = event.point.y;
|
2011-06-30 07:55:35 -04:00
|
|
|
point.y += (y - point.y) / 6;
|
2013-03-09 17:37:56 -05:00
|
|
|
if (segment.previous && !segment.previous.fixed) {
|
|
|
|
var previous = segment.previous.point;
|
|
|
|
previous.y += (y - previous.y) / 24;
|
|
|
|
}
|
|
|
|
if (segment.next && !segment.next.fixed) {
|
|
|
|
var next = segment.next.point;
|
|
|
|
next.y += (y - next.y) / 24;
|
|
|
|
}
|
2011-06-30 07:55:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function onFrame(event) {
|
|
|
|
updateWave(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
function updateWave(path) {
|
|
|
|
var force = 1 - values.friction * values.timeStep * values.timeStep;
|
2013-03-09 17:37:56 -05:00
|
|
|
for (var i = 0, l = path.segments.length; i < l; i++) {
|
|
|
|
var point = path.segments[i].point;
|
2011-06-30 07:55:35 -04:00
|
|
|
var dy = (point.y - point.py) * force;
|
2013-03-09 17:37:56 -05:00
|
|
|
point.py = point.y;
|
|
|
|
point.y = Math.max(point.y + dy, 0);
|
2011-06-30 07:55:35 -04:00
|
|
|
}
|
|
|
|
|
2013-03-09 17:37:56 -05:00
|
|
|
for (var j = 0, l = springs.length; j < l; j++) {
|
|
|
|
springs[j].update();
|
2011-06-30 07:55:35 -04:00
|
|
|
}
|
|
|
|
path.smooth();
|
|
|
|
}
|
2011-06-30 08:01:18 -04:00
|
|
|
|
2011-06-30 07:55:35 -04:00
|
|
|
function onKeyDown(event) {
|
2013-03-09 17:37:56 -05:00
|
|
|
if (event.key == 'space') {
|
2011-06-30 07:55:35 -04:00
|
|
|
path.fullySelected = !path.fullySelected;
|
2013-03-09 17:37:56 -05:00
|
|
|
path.fillColor = path.fullySelected ? null : 'black';
|
|
|
|
}
|
2011-06-30 07:55:35 -04:00
|
|
|
}
|
|
|
|
</script>
|
|
|
|
</head>
|
|
|
|
<body>
|
2013-11-06 06:53:50 -05:00
|
|
|
<canvas id="canvas" resize hidpi></canvas>
|
2011-06-30 07:55:35 -04:00
|
|
|
</body>
|
|
|
|
</html>
|