2011-04-20 09:41:09 -04:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
|
2011-06-30 09:57:17 -04:00
|
|
|
<title>Meta Balls</title>
|
2011-05-05 11:25:17 -04:00
|
|
|
<link rel="stylesheet" href="../css/style.css">
|
2011-06-12 14:03:18 -04:00
|
|
|
<script type="text/javascript" src="../../dist/paper.js"></script>
|
2011-04-20 09:41:09 -04:00
|
|
|
<script type="text/paperscript" canvas="canvas">
|
|
|
|
// Ported from original Metaball script by SATO Hiroyuki
|
|
|
|
// http://park12.wakwak.com/~shp/lc/et/en_aics_script.html
|
2011-05-16 08:33:15 -04:00
|
|
|
project.currentStyle = {
|
2013-03-03 07:39:15 -05:00
|
|
|
fillColor: 'black'
|
2011-04-20 09:41:09 -04:00
|
|
|
};
|
2011-07-07 10:09:02 -04:00
|
|
|
|
2013-03-03 07:39:15 -05:00
|
|
|
var ballPositions = [[255, 129], [610, 73], [486, 363],
|
|
|
|
[117, 459], [484, 726], [843, 306], [789, 615], [1049, 82],
|
|
|
|
[1292, 428], [1117, 733], [1352, 86], [92, 798]];
|
|
|
|
|
2011-05-15 06:36:10 -04:00
|
|
|
var handle_len_rate = 2.4;
|
|
|
|
var circlePaths = [];
|
|
|
|
var radius = 50;
|
2013-03-03 07:39:15 -05:00
|
|
|
for (var i = 0, l = ballPositions.length; i < l; i++) {
|
|
|
|
var circlePath = new Path.Circle({
|
|
|
|
center: ballPositions[i],
|
|
|
|
radius: 50
|
|
|
|
});
|
|
|
|
circlePaths.push(circlePath);
|
2011-04-20 09:41:09 -04:00
|
|
|
}
|
|
|
|
|
2013-03-03 07:39:15 -05:00
|
|
|
var largeCircle = new Path.Circle({
|
|
|
|
center: [676, 433],
|
|
|
|
radius: 100
|
|
|
|
});
|
|
|
|
circlePaths.push(largeCircle);
|
|
|
|
|
|
|
|
function onMouseMove(event) {
|
|
|
|
largeCircle.position = event.point;
|
|
|
|
generateConnections(circlePaths);
|
2011-04-20 09:41:09 -04:00
|
|
|
}
|
|
|
|
|
2013-03-03 07:39:15 -05:00
|
|
|
var connections = new Group();
|
2011-04-20 09:41:09 -04:00
|
|
|
function generateConnections(paths) {
|
2013-03-03 07:39:15 -05:00
|
|
|
// Remove the last connection paths:
|
|
|
|
connections.children = [];
|
|
|
|
|
|
|
|
for (var i = 0, l = paths.length; i < l; i++) {
|
|
|
|
for (var j = i - 1; j >= 0; j--) {
|
|
|
|
var path = metaball(paths[i], paths[j], 0.5, handle_len_rate, 300);
|
|
|
|
if (path) {
|
|
|
|
connections.appendTop(path);
|
|
|
|
path.removeOnMove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-20 09:41:09 -04:00
|
|
|
}
|
|
|
|
|
2013-03-03 07:39:15 -05:00
|
|
|
generateConnections(circlePaths);
|
|
|
|
|
2011-04-20 09:41:09 -04:00
|
|
|
// ---------------------------------------------
|
2011-05-03 03:57:09 -04:00
|
|
|
function metaball(ball1, ball2, v, handle_len_rate, maxDistance) {
|
2013-03-03 07:39:15 -05:00
|
|
|
var center1 = ball1.position;
|
|
|
|
var center2 = ball2.position;
|
|
|
|
var radius1 = ball1.bounds.width / 2;
|
|
|
|
var radius2 = ball2.bounds.width / 2;
|
|
|
|
var pi2 = Math.PI / 2;
|
|
|
|
var d = center1.getDistance(center2);
|
|
|
|
var u1, u2;
|
|
|
|
|
|
|
|
if (radius1 == 0 || radius2 == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (d > maxDistance || d <= Math.abs(radius1 - radius2)) {
|
|
|
|
return;
|
|
|
|
} else if (d < radius1 + radius2) { // case circles are overlapping
|
|
|
|
u1 = Math.acos((radius1 * radius1 + d * d - radius2 * radius2) /
|
|
|
|
(2 * radius1 * d));
|
|
|
|
u2 = Math.acos((radius2 * radius2 + d * d - radius1 * radius1) /
|
|
|
|
(2 * radius2 * d));
|
|
|
|
} else {
|
|
|
|
u1 = 0;
|
|
|
|
u2 = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
var angle1 = (center2 - center1).getAngleInRadians();
|
|
|
|
var angle2 = Math.acos((radius1 - radius2) / d);
|
|
|
|
var angle1a = angle1 + u1 + (angle2 - u1) * v;
|
|
|
|
var angle1b = angle1 - u1 - (angle2 - u1) * v;
|
|
|
|
var angle2a = angle1 + Math.PI - u2 - (Math.PI - u2 - angle2) * v;
|
|
|
|
var angle2b = angle1 - Math.PI + u2 + (Math.PI - u2 - angle2) * v;
|
|
|
|
var p1a = center1 + getVector(angle1a, radius1);
|
|
|
|
var p1b = center1 + getVector(angle1b, radius1);
|
|
|
|
var p2a = center2 + getVector(angle2a, radius2);
|
|
|
|
var p2b = center2 + getVector(angle2b, radius2);
|
|
|
|
|
|
|
|
// define handle length by the distance between
|
|
|
|
// both ends of the curve to draw
|
|
|
|
var totalRadius = (radius1 + radius2);
|
|
|
|
var d2 = Math.min(v * handle_len_rate, (p1a - p2a).length / totalRadius);
|
|
|
|
|
|
|
|
// case circles are overlapping:
|
|
|
|
d2 *= Math.min(1, d * 2 / (radius1 + radius2));
|
|
|
|
|
|
|
|
radius1 *= d2;
|
|
|
|
radius2 *= d2;
|
|
|
|
|
|
|
|
var path = new Path({
|
|
|
|
segments: [p1a, p2a, p2b, p1b],
|
|
|
|
style: ball1.style,
|
|
|
|
closed: true
|
|
|
|
});
|
|
|
|
var segments = path.segments;
|
|
|
|
segments[0].handleOut = getVector(angle1a - pi2, radius1);
|
|
|
|
segments[1].handleIn = getVector(angle2a + pi2, radius2);
|
|
|
|
segments[2].handleOut = getVector(angle2b - pi2, radius2);
|
|
|
|
segments[3].handleIn = getVector(angle1b + pi2, radius1);
|
|
|
|
return path;
|
2011-04-20 09:41:09 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------------------
|
|
|
|
function getVector(radians, length) {
|
2013-03-03 07:39:15 -05:00
|
|
|
return new Point({
|
|
|
|
// Convert radians to degrees:
|
|
|
|
angle: radians * 180 / Math.PI,
|
|
|
|
length: length
|
|
|
|
});
|
2011-04-20 09:41:09 -04:00
|
|
|
}
|
|
|
|
</script>
|
|
|
|
</head>
|
|
|
|
<body>
|
2011-06-29 07:44:06 -04:00
|
|
|
<canvas id="canvas" resize></canvas>
|
2011-05-30 18:27:39 -04:00
|
|
|
</body>
|
|
|
|
</html>
|