2011-05-18 10:03:25 -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>Phyllotaxis Raster</title>
|
2011-05-18 10:03:25 -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-05-18 10:03:25 -04:00
|
|
|
<script type="text/paperscript" canvas="canvas">
|
|
|
|
// Example inspired by: 'Phyllotactic Spirals' by Ken Frederick
|
|
|
|
// http://scriptographer.org/scripts/general-scripts/phyllotactic-spirals/
|
|
|
|
|
|
|
|
// Further reading:
|
|
|
|
// http://en.wikipedia.org/wiki/Phyllotaxis
|
2011-05-19 10:59:26 -04:00
|
|
|
var raster;
|
2011-05-18 10:03:25 -04:00
|
|
|
var values = {
|
|
|
|
// The amount of paths to produce:
|
2011-05-19 10:59:26 -04:00
|
|
|
amount: 200
|
|
|
|
};
|
2011-07-07 10:09:02 -04:00
|
|
|
// Create a raster object, using the image
|
2011-06-18 21:43:47 -04:00
|
|
|
var raster = new Raster('sunflower');
|
|
|
|
|
|
|
|
// Transform the raster so it fills the bounding rectangle
|
|
|
|
// of the view:
|
|
|
|
raster.fitBounds(view.bounds, true);
|
|
|
|
|
|
|
|
// Hide the raster:
|
|
|
|
raster.visible = false;
|
2011-05-18 10:03:25 -04:00
|
|
|
|
2011-05-19 10:59:26 -04:00
|
|
|
// Create the group of circle shaped paths and scale it up a bit:
|
2011-05-18 10:03:25 -04:00
|
|
|
var group = createPhyllotaxis(values.amount);
|
2011-05-19 10:59:26 -04:00
|
|
|
group.scale(3);
|
2011-07-07 10:09:02 -04:00
|
|
|
|
2011-05-18 10:03:25 -04:00
|
|
|
function createPhyllotaxis(amount) {
|
|
|
|
var group = new Group();
|
|
|
|
// The Golden Angle (http://en.wikipedia.org/wiki/Golden_angle)
|
|
|
|
var rotation = 137.51;
|
|
|
|
var spacing = 5;
|
|
|
|
for (var i = 1; i <= amount; i++) {
|
|
|
|
var point = new Point({
|
|
|
|
length: spacing * Math.sqrt(i),
|
|
|
|
angle: i * rotation
|
|
|
|
});
|
2011-05-19 10:59:26 -04:00
|
|
|
var radius = 8 - (i / amount * 4);
|
|
|
|
var rectangleSize = new Size(0.95, 0.7) * radius;
|
|
|
|
var rectangle = new Rectangle(point, rectangleSize);
|
|
|
|
var cornerSize = new Size(radius / 4);
|
|
|
|
var path = new Path.RoundRectangle(rectangle, cornerSize);
|
|
|
|
path.rotate(i * rotation + 45);
|
2011-06-19 11:37:43 -04:00
|
|
|
group.addChild(path);
|
2011-05-18 10:03:25 -04:00
|
|
|
}
|
|
|
|
return group;
|
|
|
|
}
|
2011-05-19 10:59:26 -04:00
|
|
|
|
2011-05-18 10:03:25 -04:00
|
|
|
function colorizePaths() {
|
|
|
|
for (var i = 0, l = group.children.length; i < l; i++) {
|
|
|
|
var path = group.children[i];
|
|
|
|
// Set the path's fill color to the average color of the
|
|
|
|
// raster pixels that fall within it:
|
|
|
|
path.fillColor = raster.getAverageColor(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-19 10:59:26 -04:00
|
|
|
var position = view.center;
|
2011-05-18 10:03:25 -04:00
|
|
|
function onMouseMove(event) {
|
2011-05-19 10:59:26 -04:00
|
|
|
position = event.point;
|
2011-05-18 10:03:25 -04:00
|
|
|
}
|
2011-05-19 10:59:26 -04:00
|
|
|
|
2011-06-18 21:43:47 -04:00
|
|
|
// Whenever the window is resized, we need to fit the raster
|
|
|
|
// in the bounding rectangle of the view again:
|
2011-05-18 10:03:25 -04:00
|
|
|
function onResize() {
|
2011-06-18 21:43:47 -04:00
|
|
|
raster.fitBounds(view.bounds, true);
|
2011-05-19 10:59:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function onFrame(event) {
|
|
|
|
// 1/4th of the difference between the center position of
|
|
|
|
// the group and the current position of the mouse:
|
|
|
|
var delta = (position - group.position) / 4;
|
|
|
|
// If the length of the delta vector (the distance) is bigger
|
|
|
|
// than 1, reposition the group:
|
|
|
|
if (delta.length > 1)
|
|
|
|
group.position += delta;
|
|
|
|
// Rotate the group of paths by 1 degree:
|
|
|
|
group.rotate(1);
|
2011-05-19 12:42:03 -04:00
|
|
|
// Rotate each path in the group by 2 degrees:
|
|
|
|
for (var i = 0, l = group.children.length; i < l; i++) {
|
|
|
|
group.children[i].rotate(2);
|
|
|
|
}
|
2011-05-19 10:59:26 -04:00
|
|
|
// Colorize the paths every other frame:
|
|
|
|
if (event.count % 2 == 0) {
|
|
|
|
colorizePaths();
|
|
|
|
}
|
2011-05-18 10:03:25 -04:00
|
|
|
}
|
|
|
|
</script>
|
|
|
|
</head>
|
2011-05-19 10:59:26 -04:00
|
|
|
<body style="background:black">
|
|
|
|
<img width="415" height="492" id="sunflower" style="display: none;" src="
|
2011-06-29 07:44:06 -04:00
|
|
|
<canvas id="canvas" resize></canvas>
|
2011-05-30 18:27:39 -04:00
|
|
|
</body>
|
|
|
|
</html>
|