Public
Edited
Nov 14
Paused
2 forks
Importers
12 stars
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
{
// run the edgeBundling, resulting paths stored in airlinesGraph.links[i].path
const bundling = edgeBundling(airlinesGraph, {
compatibility_threshold,
bundling_stiffness,
step_size
});
// use bundling.update() to recompute, useful inside the tick function of a forceSimulation

return drawGraph(airlinesGraph);
}
Insert cell
// Toy example
{
const nodes = [
{ id: "a", x: 5, y: 15 },
{ id: "b", x: 17, y: 14 },
{ id: "c", x: 17, y: 15 },
{ id: "d", x: 17, y: 20 }
],
links = [
{ source: "a", target: "b" },
{ source: "a", target: "c" },
{ source: "a", target: "d" },
];
const bundling = edgeBundling(
{ nodes, links },
{
compatibility_threshold,
bundling_stiffness,
step_size
}
);
// then links will contain the bundles in the path attribute

return drawGraph({ nodes, links });

// (Optional) To update the bundling call, useful for force simulations
bundling.update();
}
Insert cell
function drawGraph({ nodes, links }) {
const svg = d3
.create("svg")
.attr("viewBox", [-10, -10, width + 20, height + 20]);

const x = d3
.scaleLinear()
.domain(d3.extent(nodes, (d) => d.x))
.range([0, width])
.nice();
const y = d3
.scaleLinear()
.domain(d3.extent(nodes, (d) => d.y))
.range([0, height])
.nice();
const line = d3
.line()
.x((d) => x(d.x))
.y((d) => y(d.y));


if (useEdgeBundling) {
svg
.selectAll("path")
.data(links)
.join("path")
.attr("d", (d) => line(d.path))
.attr("fill", "none")
.attr("stroke", "#aaa3");
} else {
const nodesMap = new Map(nodes.map(d => [d.id, d]))
svg
.selectAll("line")
.data(links)
.join("line")
.attr("x1", (d) => x(nodesMap.get(d.source).x))
.attr("y1", (d) => y(nodesMap.get(d.source).y))
.attr("x2", (d) => x(nodesMap.get(d.target).x))
.attr("y2", (d) => y(nodesMap.get(d.target).y))
.attr("fill", "none")
.attr("stroke", "#aaa3");
}
svg
.selectAll("circle")
.data(nodes)
.join("circle")
.attr("cx", (d) => x(d.x))
.attr("cy", (d) => y(d.y))
.attr("r", 2);

return svg.node();
}
Insert cell
// Given a network {nodes, links} computes the edge bundles and modifies the links
// array adding a path attribute with the coordinates for each curve
function edgeBundling(
{
nodes, // Array of nodes including x and y coords e.g. [{id: "a", x: 10, y:10}, {id: "b", x: 20, y: 20}, ...]
links // Array of links in D3 forceSimulation format e.g. [{source: "a", target: "b"}, ... ]
},
{
id = (d) => d.id,
pathAttr = "path", // name of the attribute to save the paths
bundling_stiffness = 0.1, // global bundling constant controlling edge stiffness
step_size = 0.1, // init. distance to move points
subdivision_rate = 2, // subdivision rate increase
cycles = 6, // number of cycles to perform
iterations = 90, // init. number of iterations for cycle
iterations_rate = 0.6666667, // rate at which iteration number decreases i.e. 2/3
compatibility_threshold = 0.6 // "which pairs of edges should be considered compatible (default is set to 0.6, 60% compatiblity)"
} = {}
) {
// The library wants the links as the index positions in the nodes array
const dNodes = new Map(nodes.map((d, i) => [id(d), i]));
const linksIs = links.map((l) => ({
source: dNodes.get(typeof l.source === "object" ? id(l.source) : l.source),
target: dNodes.get(typeof l.source === "object" ? id(l.target) : l.target)
}));

const edgeBundling = ForceEdgeBundling()
.nodes(nodes)
.edges(linksIs)
.bundling_stiffness(bundling_stiffness)
.step_size(step_size)
.subdivision_rate(subdivision_rate)
.cycles(cycles)
.iterations(iterations)
.iterations_rate(iterations_rate)
.compatibility_threshold(compatibility_threshold);

edgeBundling.update = () => {
const paths = edgeBundling();
links.map((l, i) => (l[pathAttr] = paths[i]));
};

edgeBundling.update();

return edgeBundling;
}
Insert cell
// Adapted code from https://github.com/upphiminn/d3.ForceBundle
// LICENSE: GPL V2

ForceEdgeBundling = function () {
let data_nodes = {}, // {'nodeid':{'x':,'y':},..}
data_edges = [], // [{'source':'nodeid1', 'target':'nodeid2'},..]
compatibility_list_for_edge = [],
subdivision_points_for_edge = [],
K = 0.1, // global bundling constant controlling edge stiffness
S_initial = 0.1, // init. distance to move points
P_initial = 1, // init. subdivision number
P_rate = 2, // subdivision rate increase
C = 6, // number of cycles to perform
I_initial = 90, // init. number of iterations for cycle
I_rate = 0.6666667, // rate at which iteration number decreases i.e. 2/3
compatibility_threshold = 0.6,
eps = 1e-6,
P = null;

/*** Geometry Helper Methods ***/
function vector_dot_product(p, q) {
return p.x * q.x + p.y * q.y;
}

function edge_as_vector(P) {
return {
x: data_nodes[P.target].x - data_nodes[P.source].x,
y: data_nodes[P.target].y - data_nodes[P.source].y
};
}

function edge_length(e) {
// handling nodes that are on the same location, so that K/edge_length != Inf
if (
Math.abs(data_nodes[e.source].x - data_nodes[e.target].x) < eps &&
Math.abs(data_nodes[e.source].y - data_nodes[e.target].y) < eps
) {
return eps;
}

return Math.sqrt(
Math.pow(data_nodes[e.source].x - data_nodes[e.target].x, 2) +
Math.pow(data_nodes[e.source].y - data_nodes[e.target].y, 2)
);
}

function custom_edge_length(e) {
return Math.sqrt(
Math.pow(e.source.x - e.target.x, 2) +
Math.pow(e.source.y - e.target.y, 2)
);
}

function edge_midpoint(e) {
let middle_x = (data_nodes[e.source].x + data_nodes[e.target].x) / 2.0;
let middle_y = (data_nodes[e.source].y + data_nodes[e.target].y) / 2.0;

return {
x: middle_x,
y: middle_y
};
}

function compute_divided_edge_length(e_idx) {
let length = 0;

for (let i = 1; i < subdivision_points_for_edge[e_idx].length; i++) {
let segment_length = euclidean_distance(
subdivision_points_for_edge[e_idx][i],
subdivision_points_for_edge[e_idx][i - 1]
);
length += segment_length;
}

return length;
}

function euclidean_distance(p, q) {
return Math.sqrt(Math.pow(p.x - q.x, 2) + Math.pow(p.y - q.y, 2));
}

function project_point_on_line(p, Q) {
let L = Math.sqrt(
(Q.target.x - Q.source.x) * (Q.target.x - Q.source.x) +
(Q.target.y - Q.source.y) * (Q.target.y - Q.source.y)
);
let r =
((Q.source.y - p.y) * (Q.source.y - Q.target.y) -
(Q.source.x - p.x) * (Q.target.x - Q.source.x)) /
(L * L);

return {
x: Q.source.x + r * (Q.target.x - Q.source.x),
y: Q.source.y + r * (Q.target.y - Q.source.y)
};
}

/*** ********************** ***/

/*** Initialization Methods ***/
function initialize_edge_subdivisions() {
for (let i = 0; i < data_edges.length; i++) {
if (P_initial === 1) {
subdivision_points_for_edge[i] = []; //0 subdivisions
} else {
subdivision_points_for_edge[i] = [];
subdivision_points_for_edge[i].push(data_nodes[data_edges[i].source]);
subdivision_points_for_edge[i].push(data_nodes[data_edges[i].target]);
}
}
}

function initialize_compatibility_lists() {
for (let i = 0; i < data_edges.length; i++) {
compatibility_list_for_edge[i] = []; //0 compatible edges.
}
}

function filter_self_loops(edgelist) {
let filtered_edge_list = [];

for (let e = 0; e < edgelist.length; e++) {
if (
data_nodes[edgelist[e].source].x != data_nodes[edgelist[e].target].x ||
data_nodes[edgelist[e].source].y != data_nodes[edgelist[e].target].y
) {
//or smaller than eps
filtered_edge_list.push(edgelist[e]);
}
}

return filtered_edge_list;
}

/*** ********************** ***/

/*** Force Calculation Methods ***/
function apply_spring_force(e_idx, i, kP) {
let prev = subdivision_points_for_edge[e_idx][i - 1];
let next = subdivision_points_for_edge[e_idx][i + 1];
let crnt = subdivision_points_for_edge[e_idx][i];
let x = prev.x - crnt.x + next.x - crnt.x;
let y = prev.y - crnt.y + next.y - crnt.y;

x *= kP;
y *= kP;

return {
x: x,
y: y
};
}

function apply_electrostatic_force(e_idx, i) {
let sum_of_forces = {
x: 0,
y: 0
};
let compatible_edges_list = compatibility_list_for_edge[e_idx];

for (let oe = 0; oe < compatible_edges_list.length; oe++) {
let force = {
x:
subdivision_points_for_edge[compatible_edges_list[oe]][i].x -
subdivision_points_for_edge[e_idx][i].x,
y:
subdivision_points_for_edge[compatible_edges_list[oe]][i].y -
subdivision_points_for_edge[e_idx][i].y
};

if (Math.abs(force.x) > eps || Math.abs(force.y) > eps) {
let diff =
1 /
Math.pow(
custom_edge_length({
source: subdivision_points_for_edge[compatible_edges_list[oe]][i],
target: subdivision_points_for_edge[e_idx][i]
}),
1
);

sum_of_forces.x += force.x * diff;
sum_of_forces.y += force.y * diff;
}
}

return sum_of_forces;
}

function apply_resulting_forces_on_subdivision_points(e_idx, P, S) {
let kP = K / (edge_length(data_edges[e_idx]) * (P + 1)); // kP=K/|P|(number of segments), where |P| is the initial length of edge P.
// (length * (num of sub division pts - 1))
let resulting_forces_for_subdivision_points = [
{
x: 0,
y: 0
}
];

for (let i = 1; i < P + 1; i++) {
// exclude initial end points of the edge 0 and P+1
let resulting_force = {
x: 0,
y: 0
};

let spring_force = apply_spring_force(e_idx, i, kP);
let electrostatic_force = apply_electrostatic_force(e_idx, i, S);

resulting_force.x = S * (spring_force.x + electrostatic_force.x);
resulting_force.y = S * (spring_force.y + electrostatic_force.y);

resulting_forces_for_subdivision_points.push(resulting_force);
}

resulting_forces_for_subdivision_points.push({
x: 0,
y: 0
});

return resulting_forces_for_subdivision_points;
}

/*** ********************** ***/

/*** Edge Division Calculation Methods ***/
function update_edge_divisions(P) {
for (let e_idx = 0; e_idx < data_edges.length; e_idx++) {
if (P === 1) {
subdivision_points_for_edge[e_idx].push(
data_nodes[data_edges[e_idx].source]
); // source
subdivision_points_for_edge[e_idx].push(
edge_midpoint(data_edges[e_idx])
); // mid point
subdivision_points_for_edge[e_idx].push(
data_nodes[data_edges[e_idx].target]
); // target
} else {
let divided_edge_length = compute_divided_edge_length(e_idx);
let segment_length = divided_edge_length / (P + 1);
let current_segment_length = segment_length;
let new_subdivision_points = [];
new_subdivision_points.push(data_nodes[data_edges[e_idx].source]); //source

for (let i = 1; i < subdivision_points_for_edge[e_idx].length; i++) {
let old_segment_length = euclidean_distance(
subdivision_points_for_edge[e_idx][i],
subdivision_points_for_edge[e_idx][i - 1]
);

while (old_segment_length > current_segment_length) {
let percent_position = current_segment_length / old_segment_length;
let new_subdivision_point_x =
subdivision_points_for_edge[e_idx][i - 1].x;
let new_subdivision_point_y =
subdivision_points_for_edge[e_idx][i - 1].y;

new_subdivision_point_x +=
percent_position *
(subdivision_points_for_edge[e_idx][i].x -
subdivision_points_for_edge[e_idx][i - 1].x);
new_subdivision_point_y +=
percent_position *
(subdivision_points_for_edge[e_idx][i].y -
subdivision_points_for_edge[e_idx][i - 1].y);
new_subdivision_points.push({
x: new_subdivision_point_x,
y: new_subdivision_point_y
});

old_segment_length -= current_segment_length;
current_segment_length = segment_length;
}
current_segment_length -= old_segment_length;
}
new_subdivision_points.push(data_nodes[data_edges[e_idx].target]); //target
subdivision_points_for_edge[e_idx] = new_subdivision_points;
}
}
}

/*** ********************** ***/

/*** Edge compatibility measures ***/
function angle_compatibility(P, Q) {
return Math.abs(
vector_dot_product(edge_as_vector(P), edge_as_vector(Q)) /
(edge_length(P) * edge_length(Q))
);
}

function scale_compatibility(P, Q) {
let lavg = (edge_length(P) + edge_length(Q)) / 2.0;
return (
2.0 /
(lavg / Math.min(edge_length(P), edge_length(Q)) +
Math.max(edge_length(P), edge_length(Q)) / lavg)
);
}

function position_compatibility(P, Q) {
let lavg = (edge_length(P) + edge_length(Q)) / 2.0;
let midP = {
x: (data_nodes[P.source].x + data_nodes[P.target].x) / 2.0,
y: (data_nodes[P.source].y + data_nodes[P.target].y) / 2.0
};
let midQ = {
x: (data_nodes[Q.source].x + data_nodes[Q.target].x) / 2.0,
y: (data_nodes[Q.source].y + data_nodes[Q.target].y) / 2.0
};

return lavg / (lavg + euclidean_distance(midP, midQ));
}

function edge_visibility(P, Q) {
let I0 = project_point_on_line(data_nodes[Q.source], {
source: data_nodes[P.source],
target: data_nodes[P.target]
});
let I1 = project_point_on_line(data_nodes[Q.target], {
source: data_nodes[P.source],
target: data_nodes[P.target]
}); //send actual edge points positions
let midI = {
x: (I0.x + I1.x) / 2.0,
y: (I0.y + I1.y) / 2.0
};
let midP = {
x: (data_nodes[P.source].x + data_nodes[P.target].x) / 2.0,
y: (data_nodes[P.source].y + data_nodes[P.target].y) / 2.0
};

return Math.max(
0,
1 - (2 * euclidean_distance(midP, midI)) / euclidean_distance(I0, I1)
);
}

function visibility_compatibility(P, Q) {
return Math.min(edge_visibility(P, Q), edge_visibility(Q, P));
}

function compatibility_score(P, Q) {
return (
angle_compatibility(P, Q) *
scale_compatibility(P, Q) *
position_compatibility(P, Q) *
visibility_compatibility(P, Q)
);
}

function are_compatible(P, Q) {
return compatibility_score(P, Q) >= compatibility_threshold;
}

function compute_compatibility_lists() {
for (let e = 0; e < data_edges.length - 1; e++) {
for (let oe = e + 1; oe < data_edges.length; oe++) {
// don't want any duplicates
if (are_compatible(data_edges[e], data_edges[oe])) {
compatibility_list_for_edge[e].push(oe);
compatibility_list_for_edge[oe].push(e);
}
}
}
}

/*** ************************ ***/

/*** Main Bundling Loop Methods ***/
let forcebundle = function () {
let S = S_initial;
let I = I_initial;
let P = P_initial;

initialize_edge_subdivisions();
initialize_compatibility_lists();
update_edge_divisions(P);
compute_compatibility_lists();

for (let cycle = 0; cycle < C; cycle++) {
for (let iteration = 0; iteration < I; iteration++) {
let forces = [];
for (let edge = 0; edge < data_edges.length; edge++) {
forces[edge] = apply_resulting_forces_on_subdivision_points(
edge,
P,
S
);
}
for (let e = 0; e < data_edges.length; e++) {
for (let i = 0; i < P + 1; i++) {
subdivision_points_for_edge[e][i].x += forces[e][i].x;
subdivision_points_for_edge[e][i].y += forces[e][i].y;
}
}
}
// prepare for next cycle
S = S / 2;
P = P * P_rate;
I = I_rate * I;

update_edge_divisions(P);
//console.log('C' + cycle);
//console.log('P' + P);
//console.log('S' + S);
}
return subdivision_points_for_edge;
};
/*** ************************ ***/

/*** Getters/Setters Methods ***/
forcebundle.nodes = function (nl) {
if (arguments.length === 0) {
return data_nodes;
} else {
data_nodes = nl;
}

return forcebundle;
};

forcebundle.edges = function (ll) {
if (arguments.length === 0) {
return data_edges;
} else {
data_edges = filter_self_loops(ll); //remove edges to from to the same point
}

return forcebundle;
};

forcebundle.bundling_stiffness = function (k) {
if (arguments.length === 0) {
return K;
} else {
K = k;
}

return forcebundle;
};

forcebundle.step_size = function (step) {
if (arguments.length === 0) {
return S_initial;
} else {
S_initial = step;
}

return forcebundle;
};

forcebundle.cycles = function (c) {
if (arguments.length === 0) {
return C;
} else {
C = c;
}

return forcebundle;
};

forcebundle.iterations = function (i) {
if (arguments.length === 0) {
return I_initial;
} else {
I_initial = i;
}

return forcebundle;
};

forcebundle.iterations_rate = function (i) {
if (arguments.length === 0) {
return I_rate;
} else {
I_rate = i;
}

return forcebundle;
};

forcebundle.subdivision_points_seed = function (p) {
if (arguments.length == 0) {
return P;
} else {
P = p;
}

return forcebundle;
};

forcebundle.subdivision_rate = function (r) {
if (arguments.length === 0) {
return P_rate;
} else {
P_rate = r;
}

return forcebundle;
};

forcebundle.compatibility_threshold = function (t) {
if (arguments.length === 0) {
return compatibility_threshold;
} else {
compatibility_threshold = t;
}

return forcebundle;
};

/*** ************************ ***/

return forcebundle;
}
Insert cell
height = 500;
Insert cell
airlinesGraph = FileAttachment("airlines@1.json").json()
Insert cell
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