chart = {
const svg = d3.select(DOM.svg(width, height));
let vis = svg.append('g')
.attr('transform', 'translate(' + [margin, margin] + ")");
svg.append('marker')
.attr('id', 'arrowhead')
.attr('viewBox', '-.1 -5 10 10')
.attr('orient', 'auto')
.attr('markerWidth', 3)
.attr('markerHeight', 3)
.append('path')
.attr('d', 'M-.1,-4L3.9,0L-.1,4');
vis.append('g')
.attr('class', 'y grid')
.selectAll('path')
.data(years).enter()
.append('path')
.attr('class', (t) => (t % 5 !== 0 ? 'minor' : 'major'))
.attr('d', (t) => ('M-12,'+yScale(new Date(t, 0, 1))+'h'+(width-margin*2+24)));
vis.append('g')
.attr('class', 'y axis')
.attr('transform', 'translate(-12,0)')
.call(yAxisLeft);
vis.append('g')
.attr('class', 'y axis')
.attr('transform', 'translate(' + (width - margin * 2 + 12) + ',0)')
.call(yAxisRight);
let teamHaloG = vis.append('g')
.attr('class', 'team-group halo');
let teamPathG = vis.append('g')
.attr('class', 'team-group');
let teamEntryG = vis.append('g')
.attr('class', 'team-group entry');
let teamExitG = vis.append('g')
.attr('class', 'team-group exit');
teamHaloG.selectAll('path')
.data(teamData).enter()
.append('path')
.attr('d', teamPath);
teamPathG.selectAll('path')
.data(teamData).enter()
.append('path')
.attr('id', (d, i) => ('team-path-' + i))
.attr('d', teamPath);
let entry = teamEntryG.selectAll('g')
.data(flattenSort(teamData.map((d,i) => flatten(Object.keys(d.entry).map(k => d.entry[k].map(y => ({'id': k, 'year': y, 'team': d, 'teamIndex': i})))))))
.enter().append('g');
entry.append('path')
.attr('class', d => ('team-entry-'+d.teamIndex))
.attr('d', entryPath)
.style('stroke', (d) => ('url(#team-' + d.teamIndex + '-' + d.id + '-' + d.year +'-entry-gradient)'));
let gradient = entry.append('linearGradient')
.attr('id', (d) => ('team-' + d.teamIndex + '-' + d.id + '-' + d.year +'-entry-gradient'))
.attr('gradientUnits','userSpaceOnUse')
.attr('x1', 0)
.attr('x2', 0)
.attr('y1', d => yScale(new Date(d.year, 0, 1)))
.attr('y2', d => yScale(new Date(d.year, 0, 1))+9);
gradient.selectAll('stop')
.data([0, 100]).enter()
.append('stop')
.attr('offset', s => (s + '%'))
.attr('stop-color', '#007ac3')
.attr('stop-opacity', s => s);
let exit = teamExitG.selectAll('g')
.data(flattenSort(teamData.map((d,i) => flatten(Object.keys(d.exit).map(k => d.exit[k].map(y => ({'id': k, 'year': y, 'team': d, 'teamIndex': i})))))))
.enter().append('g');
exit.append('path')
.attr('d', exitPath)
.attr('class', d => ('team-exit-'+d.teamIndex))
.style('stroke', d => ('url(#team-' + d.teamIndex + '-' + d.id + '-' + d.year + '-exit-gradient)'));
gradient = exit.append('linearGradient')
.attr('id', d => ('team-' + d.teamIndex + '-' + d.id + '-' + d.year + '-exit-gradient'))
.attr('gradientUnits','userSpaceOnUse')
.attr('x1', 0)
.attr('x2', 0)
.attr('y1', d => yScale(new Date(d.year, 0, 1)))
.attr('y2', d => {
if (d.team[d.year+1] && d.team[d.year+1] !== '' && d.team[d.year+1] !== 'independent') {
return yScale(new Date(d.year+1, 0, 1));
} else {
return yScale(new Date(d.year, 0, 1)) - 9;
}
});
gradient.selectAll('stop')
.data([0, 100]).enter()
.append('stop')
.attr('offset', s => (s + '%'))
.attr('stop-color', '#007ac3')
.attr('stop-opacity', s => s);
let confNamesG = vis.append('g')
.attr('class','conference-names');
let confText = confNamesG.selectAll('text')
.data(annotations)
.enter().append('text')
.attr('transform', a => ('translate('+(cScale(conferenceData[a.id].index)+(2.125*conferenceData[a.id][a.year].length))+','+yScale(new Date(a.year, 0, 1))+')'));
confText.selectAll('tspan')
.data(a => (a.label.length > 9 ? a.label.split(/\s+/) : [a.label]))
.enter().append('tspan')
.text(t => t)
.attr('x', 0)
.attr('y', (t, i) => ((i - 2) * 1.1 + 'em'));
let titleG = svg.append('g')
.attr('class', 'title')
.attr('transform', 'translate(30, 30)');
titleG.append('text')
.attr('class', 'main')
.text('Major college football programs since 1965');
titleG.append('text')
.attr('class', 'explained')
.attr('y', 20)
.text('Schools switching conferences are highlighted');
titleG.append('path')
.attr('d', 'M270,30 v-4 c0,-4 12,-8 12,-12 v-4')
.attr('stroke', 'url(#stroke-title-path)');
gradient = titleG.append('linearGradient')
.attr('id', 'stroke-title-path')
.attr('gradientUnits','userSpaceOnUse')
.attr('x1', 0)
.attr('x2', 0)
.attr('y1', 30)
.attr('y2', 10);
gradient.selectAll('stop')
.data(['#d7d7d7','#007ac3']).enter()
.append('stop')
.attr('offset', (s, i) => ((i * 100) + '%'))
.attr('stop-color', s => s);
let voronoiG = vis.append('g')
.attr('class', 'voronoi-hover');
let tooltipG = vis.append('g')
.attr('id', 'tooltip')
tooltipG.append('path');
let textTooltip = tooltipG.append('text')
.attr('dy', '0.3em');
textTooltip.append('tspan')
.attr('id', 'tooltip-name');
textTooltip.append('tspan')
.attr('id', 'tooltip-team');
voronoiG.selectAll('path')
.data(polygons)
.enter().append('path')
.attr('d', (d) => (d ? 'M' + d.join('L') + 'Z' : null))
.on('mouseover', (d) => {
let t = teamData[d.data.teamIndex];
let flip = d.data.x > 700;
textTooltip.style('text-anchor', flip ? 'end' : 'start').attr('x', flip?-10:10);
textTooltip.select('#tooltip-name').text(t.name);
textTooltip.select('#tooltip-team').text(' ' + t.team);
let length = textTooltip.node().getComputedTextLength() + 5;
tooltipG.select('path')
.attr('d', flip ? 'M0,0l-10,-10h'+-length+'v20h'+length+'z'
: 'M0,0l10,10h'+length+'v-20h'+-length+'z');
tooltipG.style('display', 'block')
.attr('transform', 'translate('+[d.data.x, d.data.y]+')rotate(0)')
.interrupt('tooltip')
.transition('tooltip')
.duration(650)
.ease(d3.easeElastic)
.attr('transform', 'translate('+[d.data.x, d.data.y]+')rotate('+(flip?15:-15)+')');
teamPathG.select('#team-path-'+d.data.teamIndex).style('stroke', '#999');
teamEntryG.select('.team-entry-'+d.data.teamIndex).style('stroke-opacity', 1);
teamExitG.select('.team-exit-'+d.data.teamIndex).style('stroke-opacity', 1);
})
.on('mouseout', (d) => {
tooltipG.style('display', 'none');
teamPathG.select('#team-path-'+d.data.teamIndex).style('stroke', '#d7d7d7');
teamEntryG.select('.team-entry-'+d.data.teamIndex).style('stroke-opacity', 0.7);
teamExitG.select('.team-exit-'+d.data.teamIndex).style('stroke-opacity', 0.7);
});
return svg.node();
}