Some random colors (on the left) are clustered according to the Euclidean distance of their corresponding RGB vectors. The image on the right shows the result of the process. Reload the page to create different colors.
In both cases, a Gosper Displacement is used for compactness. Because of this displacement, clusters can be appreciated as contiguous regions of cells having similar colors.
Clustering is provided by the hierarchical clustering feature of the clusterfck library.
(function() {
var clusters, height, items, leaves, original, result, scale, svg, tree, width;
svg = d3.select('svg');
width = svg.node().getBoundingClientRect().width;
height = svg.node().getBoundingClientRect().height;
svg.attr({
viewBox: "" + (-width / 2) + " " + (-height / 2) + " " + width + " " + height
});
items = d3.range(147).map(function(d) {
return [70 + Math.random() * 150, 70 + Math.random() * 150, 70 + Math.random() * 150];
});
console.debug('Computing hierarchical clustering...');
clusters = clusterfck.hcluster(items, clusterfck.EUCLIDEAN_DISTANCE, clusterfck.AVERAGE_LINKAGE);
tree = tree_utils.binary_to_std(clusters);
leaves = tree_utils.get_leaves(tree);
console.debug('Computing the Space-Filling Curve layouts...');
scale = 28;
sfc_layout.displace(items, sfc_layout.GOSPER, scale, scale, -Math.PI / 6);
sfc_layout.displace(leaves, sfc_layout.GOSPER, scale, scale, -Math.PI / 6);
console.debug('Drawing...');
original = svg.append('g').attr('transform', 'translate(-240,0)');
result = svg.append('g').attr('transform', 'translate(240,0)');
original.selectAll('.cell').data(items).enter().append('path').attr('class', 'cell').attr('d', jigsaw.hex_generate_svg_path(scale)).attr('transform', function(d) {
return "translate(" + d.x + "," + d.y + ")";
}).attr('fill', function(d) {
return d3.rgb(d[0], d[1], d[2]);
});
result.selectAll('.cell').data(leaves).enter().append('path').attr('class', 'cell').attr('d', jigsaw.hex_generate_svg_path(scale)).attr('transform', function(d) {
return "translate(" + d.x + "," + d.y + ")";
}).attr('fill', function(d) {
return d3.rgb(d.value[0], d.value[1], d.value[2]);
});
}).call(this);
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta name="description" content="Gosper Clustering" />
<title>Gosper Clustering</title>
<link type="text/css" href="index.css" rel="stylesheet"/>
<script src="//d3js.org/d3.v3.min.js"></script>
<script src="//wafi.iit.cnr.it/webvis/tmp/clusterfck.js"></script>
<script src="//wafi.iit.cnr.it/webvis/libs/jigmaps/zip.js"></script>
<script src="//wafi.iit.cnr.it/webvis/libs/jigmaps/tree_utils.js"></script>
<script src="//wafi.iit.cnr.it/webvis/libs/jigmaps/sfc_layout.js"></script>
<script src="//wafi.iit.cnr.it/webvis/libs/jigmaps/jigsaw.js"></script>
</head>
<body>
<svg height="500" width="960"></svg>
<script src="index.js"></script>
</body>
</html>
svg = d3.select('svg')
width = svg.node().getBoundingClientRect().width
height = svg.node().getBoundingClientRect().height
# translate the viewBox to have (0,0) at the center of the vis
svg
.attr
viewBox: "#{-width/2} #{-height/2} #{width} #{height}"
items = d3.range(147).map (d) -> [
70+Math.random()*150,
70+Math.random()*150,
70+Math.random()*150
]
console.debug 'Computing hierarchical clustering...'
clusters = clusterfck.hcluster(
items,
clusterfck.EUCLIDEAN_DISTANCE,
clusterfck.AVERAGE_LINKAGE
)
tree = tree_utils.binary_to_std(clusters)
leaves = tree_utils.get_leaves(tree)
console.debug 'Computing the Space-Filling Curve layouts...'
scale = 28
sfc_layout.displace(items, sfc_layout.GOSPER, scale, scale, -Math.PI/6)
sfc_layout.displace(leaves, sfc_layout.GOSPER, scale, scale, -Math.PI/6)
console.debug 'Drawing...'
original = svg.append('g')
.attr('transform', 'translate(-240,0)')
result = svg.append('g')
.attr('transform', 'translate(240,0)')
original.selectAll('.cell')
.data(items)
.enter().append('path')
.attr('class', 'cell')
.attr('d', jigsaw.hex_generate_svg_path(scale))
.attr('transform', (d) -> "translate(#{d.x},#{d.y})")
.attr('fill', (d) -> d3.rgb(d[0],d[1],d[2]))
result.selectAll('.cell')
.data(leaves)
.enter().append('path')
.attr('class', 'cell')
.attr('d', jigsaw.hex_generate_svg_path(scale))
.attr('transform', (d) -> "translate(#{d.x},#{d.y})")
.attr('fill', (d) -> d3.rgb(d.value[0],d.value[1],d.value[2]))
svg {
background: white;
}
.cell {
stroke: white;
stroke-width: 3;
}
.label {
fill: #333;
font-family: sans-serif;
text-anchor: middle;
pointer-events: none;
}