You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
287 lines
10 KiB
287 lines
10 KiB
import Sorter from './sorter';
|
|
import Matrices from './matrices';
|
|
|
|
const Diagram = {
|
|
clear: () => d3.select('svg').selectAll("*").remove(),
|
|
|
|
swapGroupArcs: (chords, i, j) => {
|
|
if (i < 0 || j < 0 || i === j) {
|
|
return chords;
|
|
}
|
|
|
|
// Determine which arc is closer to 0, so earlier arc always swaps with later.
|
|
const f = (chords.groups[i].endAngle < chords.groups[j].endAngle ? i : j);
|
|
const s = (chords.groups[i].endAngle < chords.groups[j].endAngle ? j : i);
|
|
|
|
const fst = Object.assign({}, chords.groups[f]);
|
|
const snd = Object.assign({}, chords.groups[s]);
|
|
|
|
const fstAngle = fst.endAngle - fst.startAngle;
|
|
const sndAngle = snd.endAngle - snd.startAngle;
|
|
const offsetAngle = sndAngle - fstAngle;
|
|
|
|
// Bring first forward.
|
|
chords.groups[f].startAngle = snd.endAngle - fstAngle;
|
|
chords.groups[f].endAngle = snd.endAngle;
|
|
chords.groups[f].index = snd.index;
|
|
|
|
// Bring second back.
|
|
chords.groups[s].startAngle = fst.startAngle;
|
|
chords.groups[s].endAngle = fst.startAngle + sndAngle;
|
|
chords.groups[s].index = fst.index;
|
|
|
|
// Bump other groups forward.
|
|
for (let ii = f + 1; ii < s; ii++) {
|
|
chords.groups[ii].startAngle += offsetAngle;
|
|
chords.groups[ii].endAngle += offsetAngle;
|
|
}
|
|
|
|
chords.forEach(v => {
|
|
if (v.source.index === f) {
|
|
v.source.startAngle += (snd.endAngle - fst.endAngle);
|
|
v.source.endAngle += (snd.endAngle - fst.endAngle);
|
|
}
|
|
|
|
if (v.target.index === f) {
|
|
v.target.startAngle += (snd.endAngle - fst.endAngle);
|
|
v.target.endAngle += (snd.endAngle - fst.endAngle);
|
|
}
|
|
|
|
if (v.source.index === s) {
|
|
v.source.startAngle -= (snd.startAngle - fst.startAngle);
|
|
v.source.endAngle -= (snd.startAngle - fst.startAngle);
|
|
}
|
|
|
|
if (v.target.index === s) {
|
|
v.target.startAngle -= (snd.startAngle - fst.startAngle);
|
|
v.target.endAngle -= (snd.startAngle - fst.startAngle);
|
|
}
|
|
|
|
if (v.source.index > f && v.source.index < s) {
|
|
v.source.startAngle += offsetAngle;
|
|
v.source.endAngle += offsetAngle;
|
|
}
|
|
|
|
if (v.target.index > f && v.target.index < s) {
|
|
v.target.startAngle += offsetAngle;
|
|
v.target.endAngle += offsetAngle;
|
|
}
|
|
|
|
if ((v.source.index === f || v.source.index === s) && (v.target.index === f || v.target.index === s)) {
|
|
const tmp = v.source.index;
|
|
v.source.index = v.target.index;
|
|
v.target.index = tmp;
|
|
}
|
|
else if (v.source.index === f) {
|
|
v.source.index = s;
|
|
}
|
|
else if (v.target.index === f) {
|
|
v.target.index = s;
|
|
}
|
|
else if (v.source.index === s) {
|
|
v.source.index = f;
|
|
}
|
|
else if (v.target.index === s) {
|
|
v.target.index = f;
|
|
}
|
|
});
|
|
|
|
// Swap array positions.
|
|
const tmp = chords.groups[f]
|
|
chords.groups[f] = chords.groups[s];
|
|
chords.groups[s] = tmp;
|
|
|
|
return chords;
|
|
},
|
|
|
|
swapGroups: (data, eventKey, chords, i, j) => {
|
|
Diagram.swapGroupArcs(chords, i, j);
|
|
Matrices.swapIndices(data.tourneys[eventKey].teams, i, j);
|
|
return chords;
|
|
},
|
|
|
|
getCountryName: (data, eventKey, n) => {
|
|
const team = data.tourneys[eventKey].teams[n];
|
|
return data.countries[team.cId];
|
|
},
|
|
|
|
getPopulation: (data, eventKey, n) => {
|
|
const team = data.tourneys[eventKey].teams[n];
|
|
return team.p;
|
|
},
|
|
|
|
getGoalsFor: (data, eventKey, n) => {
|
|
const team = data.tourneys[eventKey].teams[n];
|
|
return team.gf;
|
|
},
|
|
|
|
getGoalsAgainst: (data, eventKey, n) => {
|
|
const team = data.tourneys[eventKey].teams[n];
|
|
return team.ga;
|
|
},
|
|
|
|
buildChords: ({ matrix, data, eventKey, sort, SORT_TYPES }) => {
|
|
const chords = d3.chord()
|
|
.padAngle(0.05)
|
|
.call(null, matrix);
|
|
|
|
switch (sort) {
|
|
case SORT_TYPES.COUNTRY:
|
|
return Sorter.sort(chords, 0, chords.groups.length - 1,
|
|
Diagram.getCountryName.bind(null, data, eventKey),
|
|
Diagram.swapGroups.bind(null, data, eventKey));
|
|
case SORT_TYPES.GOALS:
|
|
return Sorter.sort(chords, 0, chords.groups.length - 1,
|
|
Diagram.getGoalsFor.bind(null, data, eventKey),
|
|
Diagram.swapGroups.bind(null, data, eventKey));
|
|
case SORT_TYPES.POPULATION:
|
|
return Sorter.sort(chords, 0, chords.groups.length - 1,
|
|
Diagram.getPopulation.bind(null, data, eventKey),
|
|
Diagram.swapGroups.bind(null, data, eventKey));
|
|
}
|
|
|
|
|
|
|
|
return chords;
|
|
},
|
|
|
|
buildDimensions: () => {
|
|
const svgW = d3.select("svg").attr("width");
|
|
const svgH = (d3.select("svg").attr("height"));
|
|
const arcT = 5;
|
|
const outerR = Math.min(svgW, svgH) * 0.5 - 130;
|
|
const innerR = outerR - arcT;
|
|
|
|
return { svgW, svgH, arcT, outerR, innerR};
|
|
},
|
|
|
|
buildColorScheme: ({ scheme, len }) => {
|
|
if (scheme === 1) {
|
|
return d3.scaleLinear()
|
|
.domain([0, len])
|
|
.range(["#fff", "green"])
|
|
.interpolate(d3.interpolateRgb);
|
|
} else if (scheme === 2) {
|
|
const colors = [
|
|
"#ffffd9",
|
|
"#edf8b1",
|
|
"#c7e9b4",
|
|
"#7fcdbb",
|
|
"#41b6c4",
|
|
"#1d91c0",
|
|
"#225ea8",
|
|
"#253494",
|
|
"#081d58"
|
|
];
|
|
|
|
return (i => colors[i % colors.length]);
|
|
} else if (scheme === 3) {
|
|
return d3.scaleLinear()
|
|
.domain([0, len])
|
|
.range(["red", "blue"])
|
|
.interpolate(d3.interpolateRgb);
|
|
} else if (scheme === 4) {
|
|
return d3.scaleOrdinal(d3.schemeCategory10);
|
|
}
|
|
|
|
return d3.scaleOrdinal(d3.schemeCategory20);
|
|
},
|
|
|
|
buildRibbons: ({ container, color, chords, data, eventKey }) => {
|
|
const dimensions = Diagram.buildDimensions();
|
|
const ribbon = d3.ribbon().radius(dimensions.innerR);
|
|
|
|
container.append("g")
|
|
.attr("class", "ribbons")
|
|
.selectAll("path")
|
|
.data(chords => chords)
|
|
.enter().append("path")
|
|
.attr("d", ribbon)
|
|
.attr('data-round-id', d => d.game.rId)
|
|
.style('fill', d => color(d.target.index))
|
|
.style('stroke', d => d3.rgb(color(d.target.index)).darker())
|
|
.classed("ribbon", true)
|
|
.append("title")
|
|
.text(function(d) {
|
|
const t1 = data.tourneys[eventKey].teams[d.source.index];
|
|
const t2 = data.tourneys[eventKey].teams[d.target.index];
|
|
|
|
const c1 = data.countries[t1.cId];
|
|
const c2 = data.countries[t2.cId];
|
|
|
|
let s1 = d.game.t1 === t1.tId ? d.game.s1 : d.game.s2;
|
|
let s2 = d.game.t2 === t2.tId ? d.game.s2 : d.game.s1;
|
|
|
|
if (d.game.sp1 !== null && d.game.sp2 !== null) {
|
|
s1 = d.game.t1 === t1.tId ? d.game.sp1 : d.game.sp2;
|
|
s2 = d.game.t2 === t2.tId ? d.game.sp2 : d.game.sp1;
|
|
|
|
s1 += " (Penalties)";
|
|
s2 += " (Penalties)";
|
|
}
|
|
else if (d.game.se1 !== null && d.game.se2 !== null) {
|
|
s1 = d.game.t1 === t1.tId ? d.game.se1 : d.game.se2;
|
|
s2 = d.game.t2 === t2.tId ? d.game.se2 : d.game.se1;
|
|
|
|
s1 += " (Extended time)";
|
|
s2 += " (Extended time)";
|
|
}
|
|
|
|
return `${c1}: ${s1}\n${c2}: ${s2}\n${data.rounds[d.game.rId]}`;
|
|
});
|
|
},
|
|
|
|
buildArcs: ({ container, color, eventKey, data }) => {
|
|
const dimensions = Diagram.buildDimensions();
|
|
const arc = d3.arc().innerRadius(dimensions.innerR).outerRadius(dimensions.outerR)
|
|
|
|
const group = container.append("g")
|
|
.attr("class", "groups")
|
|
.selectAll("g")
|
|
.data(chords => chords.groups)
|
|
.enter().append("g");
|
|
|
|
group.append("path")
|
|
.style("fill", d => color(d.index))
|
|
.style("stroke", d => d3.rgb(color(d.index)).darker())
|
|
.attr("d", arc);
|
|
|
|
group.append("text")
|
|
.each(function(d) { d.angle = (d.startAngle + d.endAngle) / 2; })
|
|
.attr("dy", ".35em")
|
|
.attr("transform", function(d) {
|
|
return "rotate(" + (d.angle * 180 / Math.PI - 91) + ")"
|
|
+ "translate(" + (dimensions.innerR + 26) + ")"
|
|
+ (d.angle > Math.PI ? "rotate(180)" : "");
|
|
})
|
|
.style("text-anchor", function(d) { return d.angle > Math.PI ? "end" : null; })
|
|
.text(function(d) {
|
|
const team = data.tourneys[eventKey].teams[d.index];
|
|
let metric = '';
|
|
|
|
// switch (sort) {
|
|
// case SORT_TYPES.GOALS:
|
|
// metric = `(${team.gf})`;
|
|
// break;
|
|
// // case SORT_TYPES.POPULATION:
|
|
// // metric = `(${Number(team.p).toLocaleString()})`;
|
|
// // break;
|
|
// }
|
|
|
|
return `${data.countries[team.cId]} ${metric}`;
|
|
});
|
|
},
|
|
|
|
buildContainer: (chords) => {
|
|
const svg = d3.select("svg");
|
|
const dimensions = Diagram.buildDimensions();
|
|
|
|
return svg.append("g")
|
|
.attr("transform", `translate(${ dimensions.svgW / 2 },${ dimensions.svgH / 2 })`)
|
|
.datum(chords);
|
|
|
|
|
|
},
|
|
};
|
|
|
|
export default Diagram;
|
|
|