groupedGraph = (
graph = {},
{
height = 600,
color = d3
.scaleOrdinal()
.range([
"red",
"green",
"blue",
"#6b486b",
"#a05d56",
"#d0743c",
"#ff8c00"
]),
groupColorNames = [],
margin = 1.2,
curve = "curveCatmullRomClosed",
r = 5,
label = undefined,
labelStyle = {
fontFamily: "sans-serif",
fontSize: 10,
color: true
},
strokeWidth = (d) => Math.sqrt(d.value),
strokeStyle = {
color: "gray",
opacity: 0.6
},
nodeStyle = {
stroke: "white",
strokeWidth: "1.5px"
},
groupStyle = {
fillOpacity: 0.1,
strokeOpacity: 1
},
onHover = undefined,
onClick = undefined,
onLeave = undefined,
groupX = [],
groupY = [],
displayGroupOnHover = false,
chargeStrength = undefined,
linkStrength = undefined,
excludeGroups = [],
fade = true
} = {}
) => {
if (
!Object.keys(graph).includes("links") ||
!Object.keys(graph).includes("nodes")
)
throw new Error(
"The visualization must be provided a graph object containing links and nodes."
);
if (graph.links.length === 0)
throw new Error(
`No links were provided. See documentation for detailed instructions on the required data structure for the graph.`
);
if (graph.nodes.length === 0)
throw new Error(
`No nodes were provided. See documentation for detailed instructions on the required data structure for the graph.`
);
if (groupX.length || groupY.length) {
const _ = [...new Set(graph.nodes.map((d) => d.group))]
.sort((a, b) => a - b)
.forEach((group) => {
if (groupX.length && groupX[group] === undefined)
throw new Error(
`groupX provided but not of correct length: group ${group} has no value.`
);
if (groupY.length && groupY[group] === undefined)
throw new Error(
`groupY provided but not of correct length: group ${group} has no value.`
);
});
}
const svg = d3.create("svg").attr("viewBox", [0, 0, width, height]);
const groups = svg.append("g").attr("class", "groups");
const link = svg
.append("g")
.attr("class", "links")
.selectAll("line")
.data(graph.links)
.join("line")
.attr("stroke-width", strokeWidth)
.attr("stroke", strokeStyle.color)
.attr("stroke-opacity", strokeStyle.opacity);
const node = svg
.append("g")
.attr("class", "nodes")
.selectAll("circle")
.data(graph.nodes)
.join("circle")
.attr("r", r)
.attr("stroke", nodeStyle.stroke)
.attr("stroke-width", nodeStyle.strokeWidth)
.attr("fill", (d) => {
if (!groupColorNames.length) return color(d.group);
return color(groupColorNames[d.group]);
})
.call(
d3
.drag()
.on("start", dragstarted)
.on("drag", dragged)
.on("end", dragended)
);
let _label = svg
.append("g")
.attr("class", "labels")
.selectAll("text")
.data(graph.nodes)
.join("text")
.text(label)
.attr("text-anchor", "middle")
.attr("font-family", labelStyle.fontFamily)
.attr("font-size", labelStyle.fontSize)
.attr("fill", (d) => {
if (!labelStyle.color) return "black";
if (!groupColorNames.length) return color(d.group);
return color(groupColorNames[d.group]);
})
.call(
d3
.drag()
.on("start", dragstarted)
.on("drag", dragged)
.on("end", dragended)
);
if (typeof onHover === "function") {
node.on("mouseover", onHover);
}
if (typeof onLeave === "function") {
node.on("mouseout", onLeave);
}
if (typeof onClick === "function") {
node.on("click", onClick);
}
const simulation = d3.forceSimulation().nodes(graph.nodes);
if (linkStrength && typeof linkStrength === "number") {
simulation.force(
"link",
d3
.forceLink()
.links(graph.links)
.id((d) => d.id)
.strength(linkStrength)
);
} else {
simulation.force(
"link",
d3
.forceLink()
.links(graph.links)
.id((d) => d.id)
);
}
if (chargeStrength && typeof chargeStrength === "number") {
simulation.force("charge", d3.forceManyBody().strength(chargeStrength));
} else {
simulation.force("charge", d3.forceManyBody());
}
if (groupX.length === 0 && groupY.length === 0) {
simulation.force("center", d3.forceCenter(width / 2, height / 2));
}
if (groupX.length) {
simulation
.force(
"x",
d3.forceX().x((d) => groupX[+d.group])
)
.force(
"y",
d3.forceY().y((d) => height / 2)
);
}
if (groupY.length) {
simulation
.force(
"y",
d3.forceY().y((d) => groupY[+d.group])
)
.force(
"x",
d3.forceX().x((d) => width / 2)
);
}
simulation.on("tick", tick);
function tick() {
link
.attr("x1", (d) => d.source.x)
.attr("y1", (d) => d.source.y)
.attr("x2", (d) => d.target.x)
.attr("y2", (d) => d.target.y);
node.attr("cx", (d) => d.x).attr("cy", (d) => d.y);
_label.attr("x", (d) => d.x).attr("y", (d) => d.y);
updateGroups(paths, groupIds, node, curve, margin);
}
const groupIds = [...new Set(graph.nodes.map((n) => +n.group))]
.map((groupId) => ({
groupId: groupId,
count: graph.nodes.filter((n) => +n.group == groupId).length
}))
.filter(
(group) => group.count > 2 && !excludeGroups.includes(group.groupId)
)
.map((group) => group.groupId);
const paths = groups
.selectAll(".path_placeholder")
.data(groupIds, (d) => +d)
.join("g")
.attr("class", "path_placeholder")
.attr("fill-opacity", groupStyle.fillOpacity)
.attr("stroke-opacity", groupStyle.strokeOpacity)
.append("path")
.attr("stroke", (d) => {
if (!groupColorNames.length) return color(d);
return color(groupColorNames[d]);
})
.attr("fill", (d) => {
if (!groupColorNames.length) return color(d);
return color(groupColorNames[d]);
})
.attr("opacity", 0);
if (!displayGroupOnHover) {
fade
? paths.transition().duration(2000).attr("opacity", 1)
: paths.attr("opacity", 1);
} else {
}
groups
.selectAll(".path_placeholder")
.call(
d3
.drag()
.on("start", group_dragstarted)
.on("drag", group_dragged)
.on("end", group_dragended)
);
node.append("title").text(function (d) {
return d.id;
});
function dragstarted(evt, d) {
if (!evt.active) simulation.alphaTarget(0.3).restart();
d.fx = d.x;
d.fy = d.y;
}
function dragged(evt, d) {
d.fx = evt.x;
d.fy = evt.y;
}
function dragended(evt, d) {
if (!evt.active) simulation.alphaTarget(0);
d.fx = null;
d.fy = null;
}
function group_dragstarted(evt, groupId) {
if (!evt.active) simulation.alphaTarget(0.3).restart();
d3.select(this).select("path").style("stroke-width", 3);
}
function group_dragged(evt, groupId) {
node
.filter(function (d) {
return d.group == groupId;
})
.each(function (d) {
d.x += evt.dx;
d.y += evt.dy;
});
}
function group_dragended(evt, groupId) {
if (!evt.active) simulation.alphaTarget(0.3).restart();
d3.select(this).select("path").style("stroke-width", 1);
}
invalidation.then(() => simulation.stop());
return svg;
}