Published
Edited
May 8, 2021
2 forks
2 stars
Insert cell
Insert cell
chart = {
const svg = d3.create("svg")
.attr("viewBox", [-width / 2 -100, -height / 2 -100, width+200, height+200])

.attr("font-size", 15)
.attr("font-family", "sans-serif");
//.style("width", "100%")
//.style("height", "auto");

const chords = chord(data);
const group = svg.append("g")
.selectAll("g")
.data(chords.groups)
.enter().append("g");
const groupPath = group.append("path")
.attr("class", "group")
.attr("fill", d => color(d.index))
.attr("stroke", d => d3.rgb(color(d.index)).darker())
.attr("d", arc)
.attr("id", function(d, i) { return "group" + d.index; }) //add id
.on("mouseover", fade(.1))
.on("mouseout", fade(opacityDefault));
// group.append("text")
// .attr("x", 5)
// .attr("dy", 15)
// .append("textPath")
// .attr("xlink:href", function(d) { return "#group" + d.index; })
// .text(function(chords, i){return names[i];})
// .style("fill", "black");
// const g = svg.append('g')
// .attr('transform', `translate(${width / 2}, ${height / 2})`)
// .datum(chord(data))
// group.append('g').append('text')
// .text(d => names[d.index] === 'dummy' ? '' : names[d.index])
// .attr('transform', d => `translate(${arc.centroid(d)})`)
// .attr('text-anchor', d =>
// Math.sin((d.endAngle + d.startAngle) / 2) > 0 ? 'start' : 'end'
// )
// .attr('dominant-baseline', d =>
// Math.cos((d.endAngle + d.startAngle) / 2) > 0 ? 'initial' : 'hanging'
// )
group.append("text")
.each(d => (d.angle = (d.startAngle + d.endAngle) / 2))
.attr("dy", "0.35em")
.attr("transform", d => `
rotate(${(d.angle * 180 / Math.PI - 90)})
translate(${outerRadius + 5})
${d.angle > Math.PI ? "rotate(180)" : ""}
`)
.attr("text-anchor", d => d.angle > Math.PI ? "end" : null)
.text(d => names[d.index]);

group.append("title")
.text(d => `${names[d.index]}
${d3.sum(chords, c => (c.source.index === d.index) * c.source.value)} outgoing →
${d3.sum(chords, c => (c.target.index === d.index) * c.source.value)} incoming ←`);


const ribbons = svg.append("g")
.attr("fill-opacity", 0.67)
.selectAll("path")
.data(chords)
.enter().append("path")
.attr("class", "ribbons")
.attr("d", ribbon)
.attr("fill", d => color(d.target.index))
.attr("stroke", d => d3.rgb(color(d.target.index)).darker())
.on("mouseover", function(d) {
tooltip.transition()
.duration(200)
.style("opacity", opacityDefault);
//tooltip.html(d.source.value + " reaches in " + names[d.source.index] + " are " + names[d.target.index])
tooltip.html(names[d.source.index] + " reached out to " + d.source.value + " " + names[d.target.index] + "s")
.style("left", (d3.event.pageX + 5) + "px")
.style("top", (d3.event.pageY - 28) + "px");
// fade other ribbons
ribbons
.filter(dd => dd !== d)
.transition()
.style('opacity', 0.1);
// fade other groups
groupPath
.filter((dd,i) => dd.index !== d.source.index && dd.index !== d.target.index)
.transition()
.style("opacity", 0.1);
})
.on("mouseout", function(d) {
tooltip.transition()
.duration(500)
.style("opacity", 0);
// unfade ribbons
ribbons
.transition()
.style('opacity', opacityDefault);
// unfade groups
groupPath
.transition()
.style("opacity", opacityDefault);
});

//Returns an event handler for fading a given chord group.
function fade(opacity) {
return function(d,i) {
ribbons
.filter(function(dd) { return dd.source.index != d.index && dd.target.index != d.index; })
.transition()
.style("opacity", opacity);
// fade all other groups
groupPath
.filter(function(dd) { return dd.index != d.index; })
.transition()
.style("opacity", opacity);
};
}
return svg.node();
}
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
arc = d3.arc()
.innerRadius(innerRadius)
.outerRadius(outerRadius)
Insert cell
ribbon = d3.ribbon()
.radius(innerRadius)
Insert cell
tooltip = d3
.select("body")
.append("div")
.attr("class", "tooltip")
.style("width","text.length + 'px'");
Insert cell
// this is the original color setting
// color = d3.scaleOrdinal()
// .domain(d3.range(names.length))
// .range(["#828684", "#ded2c2", "#ebb48c", "#819582", "#d08a55", "#c2c9c9", "#ABA6BF", "#ecb7bf", "#583E2E", "#F1E0D6"])

color = d3.scaleOrdinal()
.domain(d3.range(names.length))
.range(["#016938", "#fddd10", "#f2c439", "#426366","#113d53", "#616264", "#5d9942", "#f09c9d", "#d18729", "#fddd10", "#6ba2a5"])

Insert cell
outerRadius = Math.min(width, height) * 0.5 - 30
Insert cell
innerRadius = outerRadius - 20
Insert cell
height = Math.min(640, width)
Insert cell
// names = ['OCCC', 'OAACC', 'OLCC', 'OVCC', 'EOCDC','Unity', "Salon", "Auto", "Restaurant", "Grocery","Prof. Services"]
Insert cell
names = ['OLCC', 'OVCC', 'OAACC','Unity', 'EOCDC','OCCC', "Restaurant","Professional Service", "Salon", "Consumer Retail","Other Service"]
Insert cell
html`<style>
svg
{
font-family: sanserif;
font-size: 10;
}
.tooltip {
position: absolute;
pointer-events: none;
height: 20px;
padding: 8px;
border-radius: 8px;
background-color: white;
font-size: .8em;
}
</style>`
// Arial, Verdana,
Insert cell
_ = require('lodash')
Insert cell
d3 = require("d3@5")
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