function ForceHulls(
graph,
options = {}
) {
let o = {
groupBy: (d) => d.group,
nodeSize: 25,
height: 500,
width: 800,
hullPadding: 5,
showGroupingNodes: false,
hullColor: d3.scaleOrdinal(d3.schemeCategory10),
fontSize: 6,
clipLabels: true,
minDegree: 0,
useImages: true,
sizeBy: (d) => d.degree,
id: (d) => d.id,
labelBy: (d, i) => d.id,
lineWidth: 1.3,
title: "",
invalidation: null,
this: null,
chargeStrength: null,
maxCharge: -200,
photoBy: (d) => d.photo,
photoMargin: 2,
centerStrength: 0.1,
linkStrength: 0.15,
linkDistance: 5,
disjoint: false,
tipcontent: null,
warmUpTicks: 100,
drawHulls: true,
transitionDuration: 500,
animate: false,
...options
};
let visibleNodes = new Map(graph.nodes.map((d) => [o.id(d), d]));
let clusters = [];
if (o.drawHulls) {
if (graph.clusters) {
clusters = graph.clusters;
} else {
clusters = d3.group(graph.nodes, (d) => ("" + o.groupBy(d)).trim());
}
}
const filteredClusters = filterClustersFromNodes({
clusters,
visibleNodes,
id: o.id
});
const filteredLinks = filterLinksFromNodes({
links: graph.links,
visibleNodes,
id: o.id
});
console.log( "filtered Links", filteredLinks);
const chargeScale = d3
.scalePow()
.exponent(1.5)
.domain([1, 30])
.range([-0.1, o.maxCharge]);
o.chargeStrength = o.chargeStrength || ((d) => chargeScale(d.r));
const svg = o.this ? d3.select(o.this).select("svg") : d3.create("svg");
svg.append("defs").append("style").text(`
circle.highlight { stroke: red; stroke-width: 2px; }
`);
const target = htl.html`<div>
<h2>${o.title}</h2>
${svg.node()}
</div>`;
svg
.attr("width", o.width + "px")
.attr("height", o.height + "px")
.attr("viewBox", [0, 0, o.width, o.height])
.attr(
"style",
"overflow: visible"
);
const r = d3
.scalePow()
.domain([0, d3.max(graph.nodes, o.sizeBy)])
.range([o.nodeSize / 2, o.nodeSize * 2]);
graph.nodes.map((d) => (d.r = o.sizeBy(d) ? r(o.sizeBy(d)) : r.range()[0]));
const collision = d3.forceCollide((d) => d.r + 2).iterations(4);
const simulation = d3
.forceSimulation(graph.nodes)
.force(
"link",
d3
.forceLink(filteredLinks)
.id((n) => o.id(n))
.distance(o.linkDistance)
.strength(o.linkStrength)
)
.force("charge", d3.forceManyBody().strength(o.chargeStrength));
if (!o.this) {
simulation.alpha(3);
} else {
simulation.alpha(1);
}
if (o.disjoint) {
simulation
.force("x", d3.forceX(o.width / 2).strength(o.centerStrength))
.force(
"y",
d3
.forceY(o.height / 2)
.strength((o.centerStrength * 0.7 * width) / o.height)
);
} else {
simulation.force("center", d3.forceCenter(width / 2, o.height / 2));
}
const { hull, getHull } = drawHull({ o, svg, graph, filteredClusters });
const link = drawLinks({ graph, o, svg, filteredLinks });
const node = drawNodes({ svg, graph, simulation, o, filteredClusters });
const label = createLabels({
svg,
_data: graph.nodes.filter(
(d) =>
d.type !== "Connection"
),
className: "nodeLabel",
o
});
const images = drawImages({ svg, graph, simulation, o });
const labelDepts = createLabels({
svg,
_data: graph.nodes.filter((d) => d.type === "Connection"),
className: "connectionLabel",
o
});
const hoverOptions = { raise: false, r: r.range()[1], svg };
if (o.tipcontent) hoverOptions.tipcontent = o.tipcontent;
const hoverObj = hover(node, invalidation, hoverOptions);
function ticked() {
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);
label
.classed("shadow", simulation.alpha() <= 0.1)
.attr("transform", (d) => `translate(${d.x}, ${d.y})`);
labelDepts
.classed("shadow", simulation.alpha() <= 0.1)
.attr("transform", (d) => `translate(${d.x}, ${d.y})`);
node
.classed("shadow", simulation.alpha() <= 0.1)
.attr("transform", (d) => `translate(${d.x}, ${d.y})`);
o.useImages &&
images
.attr("x", (d) => d.x)
.attr("y", (d) => d.y);
hull.attr("d", getHull);
hoverObj.reset();
simulation.force("collide", simulation.alpha() <= 0.5 ? collision : null);
}
simulation.stop();
simulation.on("tick", ticked);
for (let i = 0; i < o.warmUpTicks; i++) {
simulation.tick();
}
simulation.restart();
const _invalidation = o.invalidation || invalidation;
_invalidation.then(() => {
simulation.stop();
});
ticked();
target.clusters = filteredClusters;
return target;
}