function sankeyChord (data, {
title = null,
bind = '.chart',
height = 500,
_width = width * 0.6,
sourceCols = ["#ffd700",
"#fa8775",
"#ea5f94",
"#9d02d7"],
targetCols = ['#ccc'],
margin = {top: 20, right: 100, bottom: 20, left: 100},
} = {}) {
const w = _width - margin.left - margin.right
const h = height - margin.top - margin.bottom
const sourceColors = d3.scaleOrdinal()
.range(sourceCols)
const targetColors = d3.scaleOrdinal()
.range(targetCols)
const _data = nodesAndLinks(data)
console.log('data', data)
console.log('_data formatted', _data)
const sel = d3.create('div').attr('class', 'js-vis')
createDOM(sel)
const svg = sel.select('.js-svg')
.append('svg')
.attr('width', _width)
.attr('height', height)
.append('g')
.attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
const sankey = d3Sankey.sankey()
.nodeWidth(40)
.nodePadding(10)
.extent([[1, 1], [w - 1, h - 6]])
sankey(_data)
const link = svg.append('g')
.attr('class', 'links')
.attr('fill', 'none')
.attr('stroke-opacity', 0.6)
.selectAll('path')
.data(_data.links)
.enter().append('path')
.attr('class', d => {
return `source-${d.source.node} target-${d.target.node}`
})
.attr('d', d3Sankey.sankeyLinkHorizontal())
.attr('stroke', d => sourceColors(d.source.node))
.attr('stroke-width', d => Math.max(1, d.width))
const node = svg.append('g')
.attr('class', 'nodes')
.attr('font-family', 'sans-serif')
.attr('font-size', 10)
.selectAll('g')
.data(_data.nodes)
.enter().append('g')
node.append('rect')
.attr('x', d => d.x0)
.attr('y', d => d.y0)
.attr('class', d => d.type + ' bar')
.attr('height', d => d.y1 - d.y0)
.attr('width', d => d.x1 - d.x0)
.attr('fill', d => (d.type === 'source')
? sourceColors(d.subIndex)
: targetColors(d.subIndex)
)
.on('click', function (e, d) {
if (d.type === 'source') {
highlightSourcePaths(svg, d.index, d.type)
updateTitle(sel, d.label, d.value)
listInfo(sel, d.sourceLinks, {
type: 'source',
colorScale: sourceColors
})
} else {
highlightSourcePaths(svg, d.index, d.type)
updateTitle(sel, d.label, d.value)
listInfo(sel, d.targetLinks, {
type: 'target',
colorScale: sourceColors
})
}
})
.on('dblclick', function (d) {
svg.selectAll('path')
.attr('stroke-opacity', 0.4)
});
node.append('text')
.attr('x', d => d.x0 + 50)
.attr('y', d => (d.y1 + d.y0) / 2)
.attr('dy', '0.35em')
.attr('class', 'label')
.attr('text-anchor', 'start')
.text(d => d.label)
.filter(d => d.x0 < _width / 2)
.attr('x', d => d.x1 - 50)
.attr('text-anchor', 'end');
function highlightSourcePaths (sel, index, type) {
sel.selectAll('path')
.attr('stroke-opacity', 0.05)
.filter(`path.${type}-${index}`)
.attr('stroke-opacity', 0.6);
}
return sel.node();
}