function make_graph(opts = {}) {
let {
rotation = 0,
scale = 1,
svg_width = 0.7 * width,
svg_height = 0.5 * svg_width,
translation = '0 0'
} = opts;
let svg = d3
.create("svg")
.attr('width', svg_width)
.attr('height', svg_height);
let main = svg
.append('g')
.attr(
'transform',
` translate(${translation}) scale(${scale}) rotate(${rotation})`
);
main
.append("svg:defs")
.selectAll("marker")
.data(["end"])
.enter()
.append("svg:marker")
.attr("id", String)
.attr("viewBox", "0 -5 15 10")
.attr("refX", 15)
.attr("refY", -1.5)
.attr("markerWidth", 6)
.attr("markerHeight", 6)
.attr("orient", "auto")
.append("svg:path")
.attr('transform', 'rotate(-10)')
.attr("d", "M0,-5L10,0L0,5");
// Lay down the links
let link = main
.append("g")
.attr("class", "links")
.selectAll("path")
.data(links)
.enter()
.append("path")
.attr('class', 'link')
.style('fill', 'none')
.style('stroke', '#000')
.style('opacity', 0.3)
.style("stroke-width", d => Math.max(0.06 * d.score, 1))
.attr("marker-end", "url(#end)")
.attr(
'title',
d => `${d.team1}: ${d.score}<br />${d.team2}: ${d.other_score}`
)
// The source and target functions need to account for the fact that
// the simulation will rewrite the link.source and link.target attributes.
.attr("source", function(d) {
if (d.source.idx) {
return d.source.idx;
} else {
return d.source;
}
})
.attr("target", function(d) {
if (d.target.idx) {
return d.target.idx;
} else {
return d.target;
}
})
// Highlight this edge's neighborhood on mouseenter.
.on('mouseenter', function(d) {
svg.selectAll("circle").style("opacity", 0.1);
svg.selectAll("path.link").style("opacity", 0.05);
d3.select(this).style('opacity', 0.8);
svg
.select(
"path[source=\'" +
d.target.idx +
"\'][target=\'" +
d.source.idx +
"\']"
)
.style("opacity", 0.8);
svg.select(`circle#c${parseInt(d.source.idx)}`).style('opacity', 1);
svg.select(`circle#c${parseInt(d.target.idx)}`).style('opacity', 1);
});
let node = main
.append("g")
.attr("class", "nodes")
.selectAll("circle")
.data(teams)
.enter()
.append("circle")
.attr('id', d => 'c' + d.idx)
.attr(
'title',
d =>
`${d.name.replace('_', ' ')}<br />
Ranking: ${d.rank}<br />
Rating: ${d3.format('.2f')(d.strength)}<br />
Index: ${d.idx}`
)
.attr("r", d => Math.sqrt(60 * parseFloat(d.strength)))
.attr("fill", "#ff7f0e")
.style('opacity', 1)
// Highlight this node's neighborhood on mouseenter.
.on('mouseenter', function(d) {
svg.selectAll("circle").style("opacity", 0.1);
svg.selectAll("path.link").style("opacity", 0.05);
let nbd = neighbors(d.idx);
nbd.push(d.idx);
nbd.forEach(function(idx) {
d3.select("circle#c" + idx).style("opacity", 1);
});
svg.selectAll("path[source=\'" + d.idx + "\']").style("opacity", 0.8);
svg.selectAll("path[target=\'" + d.idx + "\']").style("opacity", 0.8);
});
let simulation = d3
.forceSimulation()
.nodes(teams)
.force(
"link",
d3
.forceLink()
.id(d => d.idx)
.strength(function(d) {
return d.score / 100;
})
.distance(250)
)
.force("center", d3.forceCenter(svg_width / 2, svg_height / 2))
.force('squish', d3.forceY(svg_width / 2).strength(0.2))
.on('tick', function() {
link
.attr('x1', d => d.source.x)
.attr('y1', d => d.source.y)
.attr('x2', d => d.target.x)
.attr('y2', d => d.target.x)
.attr('d', linkArc);
node.attr('cx', d => d.x).attr('cy', d => d.y);
});
simulation.force("link").links(links);
// Turn off the highlighting when we leave the svg
svg.on('mouseleave', function() {
svg.selectAll('path.link').style('opacity', 0.3);
svg.selectAll('circle').style('opacity', 1);
});
// Add Tippy tooltips on mouseover.
svg
.selectAll('circle')
.nodes()
.forEach(e =>
tippy(e, {
delay: [500, 200],
duration: [100, 50]
})
);
svg
.selectAll('path.link')
.nodes()
.forEach(e =>
tippy(e, {
distance: 5,
placement: 'right',
delay: [500, 200]
})
);
return svg.node();
}