Published
Edited
Mar 7, 2020
2 forks
Importers
8 stars
Insert cell
md`# Hive Plot

Martin Krzwinski’s Hive Plot is one of my favorite methods for handeling the hairball. Check out his 2010 presentation here for an indepth explination <a href="http://www.hiveplot.net/talks/hive-plot.pdf">hive-plot</a> Simplistically,hive plots define a linear layout for nodes, grouping nodes by type and arranging them along radial axes based on some property of data. The explicit position encoding has the potential to better reveal the network structure while communicating additional information. Hive plots can also be extended to show aggregate relationships.

Each node above represents a class in a software library. Nodes are divided into three categories. The 12 o’clock axis (the top) shows source nodes—classes with only outgoing dependencies. The bottom-left axis shows target nodes with only incoming dependencies. The remaining nodes in the bottom-right have both incoming and outgoing dependencies; these are duplicated to reveal dependencies within this category.

In a standard hive plot, nodes are sorted by link degree. This example instead arranges node by package, grouping related classes. This better reveals the “macro” structure of the dependencies, while demonstrating the great flexibility of hive plots—you can use any number of methods to group and position nodes along each axis, customizing the layout to suit your needs.

`
Insert cell
chart = {
var width = 960,
height = 850,
innerRadius = 40,
outerRadius = 640,
majorAngle = (2 * Math.PI) / 3,
minorAngle = (1 * Math.PI) / 12;

var angle = d3
.scaleOrdinal()
.domain(["source", "source-target", "target-source", "target"])
.range([
0,
majorAngle - minorAngle,
majorAngle + minorAngle,
2 * majorAngle
]);

var radius = d3.scaleLinear().range([innerRadius, outerRadius]);

var color = d3.scaleOrdinal(d3.schemeSet1).domain(d3.range(10));

var svgDOM = d3
.select(DOM.svg(width, height))
.attr("class", "hiveplot")
.attr("width", width)
.attr("height", height);

var svg = svgDOM
.append("g")
.attr(
"transform",
"translate(" + outerRadius * .20 + "," + outerRadius * .57 + ")"
);

// Load the data and display the plot!
var nodes = data;

var nodesByName = {},
links = [],
formatNumber = d3.format(",d"),
defaultInfo;

// Construct an index by node name.
nodes.forEach(function(d) {
d.connectors = [];
d.packageName = d.name.split(".")[1];
nodesByName[d.name] = d;
});

// Convert the import lists into links with sources and targets.
nodes.forEach(function(source) {
source.imports.forEach(function(targetName) {
var target = nodesByName[targetName];
if (!source.source)
source.connectors.push((source.source = { node: source, degree: 0 }));
if (!target.target)
target.connectors.push((target.target = { node: target, degree: 0 }));
links.push({ source: source.source, target: target.target });
});
});

// Determine the type of each node, based on incoming and outgoing links.
nodes.forEach(function(node) {
if (node.source && node.target) {
node.type = node.source.type = "target-source";
node.target.type = "source-target";
} else if (node.source) {
node.type = node.source.type = "source";
} else if (node.target) {
node.type = node.target.type = "target";
} else {
node.connectors = [{ node: node }];
node.type = "source";
}
});

// Initialize the info display.
var info = d3
.select("#info")
.text(
(defaultInfo =
"Showing " +
formatNumber(links.length) +
" dependencies among " +
formatNumber(nodes.length) +
" classes.")
);

// Normally, Hive Plots sort nodes by degree along each axis. However, since
// this example visualizes a package hierarchy, we get more interesting
// results if we group nodes by package. We don't need to sort explicitly
// because the data file is already sorted by class name.

// Nest nodes by type, for computing the rank.
var nodesByType = d3
.nest()
.key(function(d) {
return d.type;
})
.sortKeys(d3.ascending)
.entries(nodes);

// Duplicate the target-source axis as source-target.
nodesByType.push({ key: "source-target", values: nodesByType[2].values });

// Compute the rank for each type, with padding between packages.
nodesByType.forEach(function(type) {
var lastName = type.values[0].packageName,
count = 0;
type.values.forEach(function(d, i) {
if (d.packageName != lastName) (lastName = d.packageName), (count += 2);
d.index = count++;
});
type.count = count - 1;
});

// Set the radius domain.
radius.domain(
d3.extent(nodes, function(d) {
return d.index;
})
);

// Draw the axes.
svg
.selectAll(".axis")
.data(nodesByType)
.enter()
.append("line")
.attr("class", "axis")
.attr("transform", function(d) {
return "rotate(" + degrees(angle(d.key)) + ")";
})
.attr("x1", radius(-2))
.attr("x2", function(d) {
return radius(d.count + 2);
});

// Draw the links.
svg
.append("g")
.attr("class", "links")
.selectAll(".link")
.data(links)
.enter()
.append("path")
.attr("class", "link")
.attr(
"d",
link()
.angle(function(d) {
return angle(d.type);
})
.radius(function(d) {
return radius(d.node.index);
})
)
.on("mouseover", linkMouseover)
.on("mouseout", mouseout);

// Draw the nodes. Note that each node can have up to two connectors,
// representing the source (outgoing) and target (incoming) links.
svg
.append("g")
.attr("class", "nodes")
.selectAll(".node")
.data(nodes)
.enter()
.append("g")
.attr("class", "node")
.style("fill", function(d) {
return color(d.packageName);
})
.selectAll("circle")
.data(function(d) {
return d.connectors;
})
.enter()
.append("circle")
.attr("transform", function(d) {
return "rotate(" + degrees(angle(d.type)) + ")";
})
.attr("cx", function(d) {
return radius(d.node.index);
})
.attr("r", 4)
.on("mouseover", nodeMouseover)
.on("mouseout", mouseout);

// Highlight the link and connected nodes on mouseover.
function linkMouseover(d) {
svg.selectAll(".link").classed("active", function(p) {
return p === d;
});
svg.selectAll(".node circle").classed("active", function(p) {
return p === d.source || p === d.target;
});
// info.text(d.source.node.name + " → " + d.target.node.name);
}

// Highlight the node and connected links on mouseover.
function nodeMouseover(d) {
svg.selectAll(".link").classed("active", function(p) {
return p.source === d || p.target === d;
});
d3.select(this).classed("active", true);
// info.text(d.node.name);
}

// Clear any highlighted nodes or links.
function mouseout() {
svg.selectAll(".active").classed("active", false);
// info.text(defaultInfo);
}

// A shape generator for Hive links, based on a source and a target.
// The source and target are defined in polar coordinates (angle and radius).
// Ratio links can also be drawn by using a startRadius and endRadius.
// This class is modeled after d3.svg.chord.
function link() {
var source = function(d) {
return d.source;
},
target = function(d) {
return d.target;
},
angle = function(d) {
return d.angle;
},
startRadius = function(d) {
return d.radius;
},
endRadius = startRadius,
arcOffset = -Math.PI / 2;

function link(d, i) {
var s = node(source, this, d, i),
t = node(target, this, d, i),
x;
if (t.a < s.a) (x = t), (t = s), (s = x);
if (t.a - s.a > Math.PI) s.a += 2 * Math.PI;
var a1 = s.a + (t.a - s.a) / 3,
a2 = t.a - (t.a - s.a) / 3;
return s.r0 - s.r1 || t.r0 - t.r1
? "M" +
Math.cos(s.a) * s.r0 +
"," +
Math.sin(s.a) * s.r0 +
"L" +
Math.cos(s.a) * s.r1 +
"," +
Math.sin(s.a) * s.r1 +
"C" +
Math.cos(a1) * s.r1 +
"," +
Math.sin(a1) * s.r1 +
" " +
Math.cos(a2) * t.r1 +
"," +
Math.sin(a2) * t.r1 +
" " +
Math.cos(t.a) * t.r1 +
"," +
Math.sin(t.a) * t.r1 +
"L" +
Math.cos(t.a) * t.r0 +
"," +
Math.sin(t.a) * t.r0 +
"C" +
Math.cos(a2) * t.r0 +
"," +
Math.sin(a2) * t.r0 +
" " +
Math.cos(a1) * s.r0 +
"," +
Math.sin(a1) * s.r0 +
" " +
Math.cos(s.a) * s.r0 +
"," +
Math.sin(s.a) * s.r0
: "M" +
Math.cos(s.a) * s.r0 +
"," +
Math.sin(s.a) * s.r0 +
"C" +
Math.cos(a1) * s.r1 +
"," +
Math.sin(a1) * s.r1 +
" " +
Math.cos(a2) * t.r1 +
"," +
Math.sin(a2) * t.r1 +
" " +
Math.cos(t.a) * t.r1 +
"," +
Math.sin(t.a) * t.r1;
}

function node(method, thiz, d, i) {
var node = method.call(thiz, d, i),
a =
+(typeof angle === "function" ? angle.call(thiz, node, i) : angle) +
arcOffset,
r0 = +(typeof startRadius === "function"
? startRadius.call(thiz, node, i)
: startRadius),
r1 =
startRadius === endRadius
? r0
: +(typeof endRadius === "function"
? endRadius.call(thiz, node, i)
: endRadius);
return { r0: r0, r1: r1, a: a };
}

link.source = function(_) {
if (!arguments.length) return source;
source = _;
return link;
};

link.target = function(_) {
if (!arguments.length) return target;
target = _;
return link;
};

link.angle = function(_) {
if (!arguments.length) return angle;
angle = _;
return link;
};

link.radius = function(_) {
if (!arguments.length) return startRadius;
startRadius = endRadius = _;
return link;
};

link.startRadius = function(_) {
if (!arguments.length) return startRadius;
startRadius = _;
return link;
};

link.endRadius = function(_) {
if (!arguments.length) return endRadius;
endRadius = _;
return link;
};

return link;
}

function degrees(radians) {
return (radians / Math.PI) * 180 - 90;
}

// var div = document.createElement("div");
// div.appendChild(info);
// div.appendChild(svgDOM.node());
return svgDOM.node(); //div;
}
Insert cell
data = d3.json(
"https://gist.githubusercontent.com/cbuie/63cc187b890c83e578a1425a939207b2/raw/2283d7de13e540216196c3f78d862c097fa4de63/flare"
)
Insert cell
html`
<style>
svg.hiveplot .axis {
stroke: #000;
stroke-width: 1.5px;
}

svg.hiveplot .node circle {
stroke: #000;
}

svg.hiveplot .link {
fill: none;
stroke: #999;
stroke-width: 1.5px;
stroke-opacity: .3;
}

svg.hiveplot .link.active {
stroke: red;
stroke-width: 2px;
stroke-opacity: 1;
}

svg.hiveplot .node circle.active {
stroke: red;
stroke-width: 3px;
}
</style>
`
Insert cell
d3 = require("https://d3js.org/d3.v5.min.js")
Insert cell

One platform to build and deploy the best data apps

Experiment and prototype by building visualizations in live JavaScript notebooks. Collaborate with your team and decide which concepts to build out.
Use Observable Framework to build data apps locally. Use data loaders to build in any language or library, including Python, SQL, and R.
Seamlessly deploy to Observable. Test before you ship, use automatic deploy-on-commit, and ensure your projects are always up-to-date.
Learn more