Title: | Algorithms for Bundling Edges in Networks and Visualizing Flow and Metro Maps |
---|---|
Description: | Implements several algorithms for bundling edges in networks and flow and metro map layouts. This includes force directed edge bundling <doi:10.1111/j.1467-8659.2009.01450.x>, a flow algorithm based on Steiner trees<doi:10.1080/15230406.2018.1437359> and a multicriteria optimization method for metro map layouts <doi:10.1109/TVCG.2010.24>. |
Authors: | David Schoch [aut, cre] |
Maintainer: | David Schoch <[email protected]> |
License: | MIT + file LICENSE |
Version: | 0.4.2 |
Built: | 2025-01-21 05:11:28 UTC |
Source: | https://github.com/schochastics/edgebundle |
A dataset containing the number of people who migrated from California to other US states
cali2010
cali2010
igraph object
converts edges of an igraph/network/tidygraph object into format useable for edge bundling
convert_edges(object, coords) ## Default S3 method: convert_edges(object, coords) ## S3 method for class 'igraph' convert_edges(object, coords) ## S3 method for class 'network' convert_edges(object, coords) ## S3 method for class 'tbl_graph' convert_edges(object, coords)
convert_edges(object, coords) ## Default S3 method: convert_edges(object, coords) ## S3 method for class 'igraph' convert_edges(object, coords) ## S3 method for class 'network' convert_edges(object, coords) ## S3 method for class 'tbl_graph' convert_edges(object, coords)
object |
graph object |
coords |
coordinates of vertices |
data frame of edges with coordinates
David Schoch
Implements the classic edge bundling by Holten.
edge_bundle_force( object, xy, K = 1, C = 6, P = 1, S = 0.04, P_rate = 2, I = 50, I_rate = 2/3, compatibility_threshold = 0.6, eps = 1e-08 )
edge_bundle_force( object, xy, K = 1, C = 6, P = 1, S = 0.04, P_rate = 2, I = 50, I_rate = 2/3, compatibility_threshold = 0.6, eps = 1e-08 )
object |
a graph object (igraph/network/tbl_graph) |
xy |
coordinates of vertices |
K |
spring constant |
C |
number of iteration cycles |
P |
number of initial edge divisions |
S |
initial step size |
P_rate |
rate of edge divisions |
I |
number of initial iterations |
I_rate |
rate of iteration decrease per cycle |
compatibility_threshold |
threshold for when edges are considered compatible |
eps |
accuracy |
This is a re-implementation of https://github.com/upphiminn/d3.ForceBundle. Force directed edge bundling is slow (O(E^2)).
see online for plotting tips
data.frame containing the bundled edges
David Schoch
Holten, Danny, and Jarke J. Van Wijk. "Force-Directed Edge Bundling for Graph Visualization." Computer Graphics Forum (Blackwell Publishing Ltd) 28, no. 3 (2009): 983-990.
edge_bundle_hammer,edge_bundle_stub,edge_bundle_path
library(igraph) g <- graph_from_edgelist( matrix(c( 1, 12, 2, 11, 3, 10, 4, 9, 5, 8, 6, 7 ), ncol = 2, byrow = TRUE), FALSE ) xy <- cbind(c(rep(0, 6), rep(1, 6)), c(1:6, 1:6)) edge_bundle_force(g, xy)
library(igraph) g <- graph_from_edgelist( matrix(c( 1, 12, 2, 11, 3, 10, 4, 9, 5, 8, 6, 7 ), ncol = 2, byrow = TRUE), FALSE ) xy <- cbind(c(rep(0, 6), rep(1, 6)), c(1:6, 1:6)) edge_bundle_force(g, xy)
Implements the hammer edge bundling by Ian Calvert.
edge_bundle_hammer(object, xy, bw = 0.05, decay = 0.7)
edge_bundle_hammer(object, xy, bw = 0.05, decay = 0.7)
object |
a graph object (igraph/network/tbl_graph) |
xy |
coordinates of vertices |
bw |
bandwidth parameter |
decay |
decay parameter |
This function only wraps existing python code from the datashader library. Original code can be found at https://gitlab.com/ianjcalvert/edgehammer. Datashader is a huge library with a lot of dependencies, so think twice if you want to install it just for edge bundling. Check https://datashader.org/user_guide/Networks.html for help concerning parameters bw and decay. To install all dependencies, use install_bundle_py.
see online for plotting tips
data.frame containing the bundled edges
David Schoch
edge_bundle_force,edge_bundle_stub, edge_bundle_path
Implements edge-path bundling.
edge_bundle_path( g, xy, max_distortion = 2, weight_fac = 2, segments = 20, bundle_strength = 1, mode = "out" )
edge_bundle_path( g, xy, max_distortion = 2, weight_fac = 2, segments = 20, bundle_strength = 1, mode = "out" )
g |
an igraph object |
xy |
coordinates of vertices |
max_distortion |
maximum distortion |
weight_fac |
edge weight factor |
segments |
number of subdivisions of edges |
bundle_strength |
bundle strength factor |
mode |
the parameter fo shortest_paths |
This is a re-implementation of https://github.com/mwallinger-tu/edge-path-bundling
see online for plotting tips
data.frame containing the bundled edges
David Schoch
Wallinger, M., Archambault, D., Auber, D., Nollenburg, M., & Peltonen, J. (2021). Edge-Path Bundling: A Less Ambiguous Edge Bundling Approach. IEEE Transactions on Visualization and Computer Graphics.
edge_bundle_hammer,edge_bundle_stub,edge_bundle_force
library(igraph) g <- graph_from_edgelist(matrix(c( 1, 2, 1, 6, 1, 4, 2, 3, 3, 4, 4, 5, 5, 6 ), ncol = 2, byrow = TRUE), FALSE) xy <- cbind(c(0, 10, 25, 40, 50, 50), c(0, 15, 25, 15, 0, -10)) edge_bundle_path(g, xy)
library(igraph) g <- graph_from_edgelist(matrix(c( 1, 2, 1, 6, 1, 4, 2, 3, 3, 4, 4, 5, 5, 6 ), ncol = 2, byrow = TRUE), FALSE) xy <- cbind(c(0, 10, 25, 40, 50, 50), c(0, 15, 25, 15, 0, -10)) edge_bundle_path(g, xy)
Implements the stub edge bundling by Nocaj and Brandes
edge_bundle_stub( object, xy, alpha = 11, beta = 75, gamma = 40, t = 0.5, tshift = 0.5 )
edge_bundle_stub( object, xy, alpha = 11, beta = 75, gamma = 40, t = 0.5, tshift = 0.5 )
object |
a graph object (igraph/tbl_graph). Does not support network objects |
xy |
coordinates of vertices |
alpha |
maximal angle (in degree) between consecutive edges in a bundle |
beta |
angle (in degree) at which to connect two stubs |
gamma |
maximal overall angle (in degree) of an edge bundle |
t |
numeric between 0 and 1. control point location |
tshift |
numeric between 0 and 1. The closer to one, the longer the bigger bundle |
see online for plotting tips
data.frame containing the bundled edges
David Schoch
Nocaj, Arlind, and Ulrik Brandes. "Stub bundling and confluent spirals for geographic networks." International Symposium on Graph Drawing. Springer, Cham, 2013.
edge_bundle_hammer,edge_bundle_force, edge_bundle_path
library(igraph) g <- graph.star(10, "undirected") xy <- matrix(c( 0, 0, cos(90 * pi / 180), sin(90 * pi / 180), cos(80 * pi / 180), sin(80 * pi / 180), cos(70 * pi / 180), sin(70 * pi / 180), cos(330 * pi / 180), sin(330 * pi / 180), cos(320 * pi / 180), sin(320 * pi / 180), cos(310 * pi / 180), sin(310 * pi / 180), cos(210 * pi / 180), sin(210 * pi / 180), cos(200 * pi / 180), sin(200 * pi / 180), cos(190 * pi / 180), sin(190 * pi / 180) ), ncol = 2, byrow = TRUE) edge_bundle_stub(g, xy) # use ggforce::geom_bezier for plotting
library(igraph) g <- graph.star(10, "undirected") xy <- matrix(c( 0, 0, cos(90 * pi / 180), sin(90 * pi / 180), cos(80 * pi / 180), sin(80 * pi / 180), cos(70 * pi / 180), sin(70 * pi / 180), cos(330 * pi / 180), sin(330 * pi / 180), cos(320 * pi / 180), sin(320 * pi / 180), cos(310 * pi / 180), sin(310 * pi / 180), cos(210 * pi / 180), sin(210 * pi / 180), cos(200 * pi / 180), sin(200 * pi / 180), cos(190 * pi / 180), sin(190 * pi / 180) ), ncol = 2, byrow = TRUE) edge_bundle_stub(g, xy) # use ggforce::geom_bezier for plotting
install datashader and scikit-image
install_bundle_py(method = "auto", conda = "auto")
install_bundle_py(method = "auto", conda = "auto")
method |
Installation method (by default, "auto" automatically finds a method that will work in the local environment, but note that the "virtualenv" method is not available on Windows) |
conda |
Path to conda executable (or "auto" to find conda using the PATH and other conventional install locations) |
A dataset containing the subway network of Berlin
metro_berlin
metro_berlin
igraph object
Kujala, Rainer, et al. "A collection of public transport network data sets for 25 cities." Scientific data 5 (2018): 180089.
Metro map layout based on multicriteria optimization
metro_multicriteria(object, xy, l = 2, gr = 0.0025, w = rep(1, 5), bsize = 5)
metro_multicriteria(object, xy, l = 2, gr = 0.0025, w = rep(1, 5), bsize = 5)
object |
original graph |
xy |
initial layout of the original graph |
l |
desired multiple of grid point spacing. (l*gr determines desired edge length) |
gr |
grid spacing. (l*gr determines desired edge length) |
w |
weight vector for criteria (see details) |
bsize |
number of grid points a station can move away rom its original position |
The function optimizes the following five criteria using a hill climbing algorithm:
Angular Resolution Criterion: The angles of incident edges at each station should be maximized, because if there is only a small angle between any two adjacent edges, then it can become difficult to distinguish between them
Edge Length Criterion: The edge lengths across the whole map should be approximately equal to ensure regular spacing between stations. It is based on the preferred multiple, l, of the grid spacing, g. The purpose of the criterion is to penalize edges that are longer than or shorter than lg.
Balanced Edge Length Criterion: The length of edges incident to a particular station should be similar
Line Straightness Criterion: (not yet implemented) Edges that form part of a line should, where possible, be co-linear either side of each station that the line passes through
Octiinearity Criterion: Each edge should be drawn horizontally, vertically, or diagonally at 45 degree, so we penalize edges that are not at a desired angle see online for more plotting tips
new coordinates for stations
David Schoch
Stott, Jonathan, et al. "Automatic metro map layout using multicriteria optimization." IEEE Transactions on Visualization and Computer Graphics 17.1 (2010): 101-114.
# the algorithm has problems with parallel edges library(igraph) g <- simplify(metro_berlin) xy <- cbind(V(g)$lon, V(g)$lat) * 100 # the algorithm is not very stable. try playing with the parameters xy_new <- metro_multicriteria(g, xy, l = 2, gr = 0.5, w = c(100, 100, 1, 1, 100), bsize = 35)
# the algorithm has problems with parallel edges library(igraph) g <- simplify(metro_berlin) xy <- cbind(V(g)$lon, V(g)$lat) * 100 # the algorithm is not very stable. try playing with the parameters xy_new <- metro_multicriteria(g, xy, l = 2, gr = 0.5, w = c(100, 100, 1, 1, 100), bsize = 35)
uses various sampling strategies to create dummy nodes for the tnss_tree
tnss_dummies( xy, root, circ = TRUE, line = TRUE, diag = TRUE, grid = FALSE, rand = FALSE, ncirc = 9, rcirc = 2, nline = 10, ndiag = 50, ngrid = 50, nrand = 50 )
tnss_dummies( xy, root, circ = TRUE, line = TRUE, diag = TRUE, grid = FALSE, rand = FALSE, ncirc = 9, rcirc = 2, nline = 10, ndiag = 50, ngrid = 50, nrand = 50 )
xy |
coordinates of "real" nodes |
root |
root node id |
circ |
logical. create circular dummy nodes around leafs. |
line |
logical. create dummy nodes on a straight line between root and leafs. |
diag |
logical. create dummy nodes diagonally through space. |
grid |
logical. create dummy nodes on a grid. |
rand |
logical. create random dummy nodes. |
ncirc |
numeric. number of circular dummy nodes per leaf. |
rcirc |
numeric. radius of circles around leaf nodes. |
nline |
numeric. number of straight line nodes per leaf. |
ndiag |
numeric. number of dummy nodes on diagonals. |
ngrid |
numeric. number of dummy nodes per dim on grid. |
nrand |
numeric. number of random nodes to create. |
coordinates of dummy nodes
David Schoch
# dummy nodes for tree rooted in California xy <- cbind(state.center$x, state.center$y) xy_dummy <- tnss_dummies(xy, 4)
# dummy nodes for tree rooted in California xy <- cbind(state.center$x, state.center$y) xy_dummy <- tnss_dummies(xy, 4)
Converts the Steiner tree to smooth paths
tnss_smooth(g, bw = 3, n = 10)
tnss_smooth(g, bw = 3, n = 10)
g |
Steiner tree computed with tnss_tree |
bw |
bandwidth of Gaussian Kernel |
n |
number of extra nodes to include per edge |
see see online for tips on plotting the result
data.frame containing the smoothed paths
David Schoch
xy <- cbind(state.center$x, state.center$y)[!state.name %in% c("Alaska", "Hawaii"), ] xy_dummy <- tnss_dummies(xy, root = 4) gtree <- tnss_tree(cali2010, xy, xy_dummy, root = 4, gamma = 0.9) tree_smooth <- tnss_smooth(gtree, bw = 10, n = 10)
xy <- cbind(state.center$x, state.center$y)[!state.name %in% c("Alaska", "Hawaii"), ] xy_dummy <- tnss_dummies(xy, root = 4) gtree <- tnss_tree(cali2010, xy, xy_dummy, root = 4, gamma = 0.9) tree_smooth <- tnss_smooth(gtree, bw = 10, n = 10)
creates an approximated Steiner tree for a flow map visualization
tnss_tree( g, xy, xydummy, root, gamma = 0.9, epsilon = 0.3, elen = Inf, order = "random" )
tnss_tree( g, xy, xydummy, root, gamma = 0.9, epsilon = 0.3, elen = Inf, order = "random" )
g |
original flow network (must be a one-to-many flow network, i.e star graph). Must have a weight attribute indicating the flow |
xy |
coordinates of "real" nodes |
xydummy |
coordinates of "dummy" nodes |
root |
root node id of the flow |
gamma |
edge length decay parameter |
epsilon |
percentage of points keept on a line after straightening with Visvalingam Algorithm |
elen |
maximal length of edges in triangulation |
order |
in which order shortest paths are calculated ("random","weight","near","far") |
Use tnss_smooth to smooth the edges of the tree
approximated Steiner tree from dummy and real nodes as igraph object
David Schoch
Sun, Shipeng. "An automated spatial flow layout algorithm using triangulation, approximate Steiner tree, and path smoothing." AutoCarto, 2016.
xy <- cbind(state.center$x, state.center$y)[!state.name %in% c("Alaska", "Hawaii"), ] xy_dummy <- tnss_dummies(xy, root = 4) gtree <- tnss_tree(cali2010, xy, xy_dummy, root = 4, gamma = 0.9)
xy <- cbind(state.center$x, state.center$y)[!state.name %in% c("Alaska", "Hawaii"), ] xy_dummy <- tnss_dummies(xy, root = 4) gtree <- tnss_tree(cali2010, xy, xy_dummy, root = 4, gamma = 0.9)
A dataset containing flights between US airports as igraph object
us_flights
us_flights
igraph object
https://gist.githubusercontent.com/mbostock/7608400/raw
A dataset containing the number of people migrating between US states from 2010-2019
us_migration
us_migration
data.frame