chords = {
document.head.innerHTML+=styling;
const width=1200;
const height=1200;
var canvas=DOM.canvas(width, height);
var ctx=canvas.getContext('2d');
const fadeInTime=250.0;
const fadeOutTime=125.0;
var previousnearestecv=-3;
var mystery=true;
var mysteryFade=1.0;
const mysteryFadeTime=500.0;
var animating=true;
let frame;
var ecvxys=[];
for (var i=0;i<ecvs.length;i++) {
const ecv=ecvs[i];
const a=(2.0*Math.PI*(1+i))/(ecvs.length+1);
const dx= 0.4*width*Math.sin(a);
const dy= -0.4*height*Math.cos(a);
const x=width*0.5+dx;
const y=height*0.5+dy;
ecvxys.push(
[x,y]
);
}
var ecvnames=[];
for (var i=0;i<ecvs.length;i++) {
ecvnames.push(ecvs[i].name.replaceAll("\n"," "));
}
var ecvindices={};
for (var i=0;i<ecvs.length;i++) {
ecvindices[ecvnames[i]]=i;
}
var ecvlinks=[];
for (var i=0;i<ecvs.length;i++) {
var row=[];
for (var j=0;j<ecvs[i].links.length;j++) {
if (ecvs[i].links[j] in ecvindices) {
row.push(ecvindices[ecvs[i].links[j]]);
} else {
console.log("Bad ECV link: can't find "+ecvs[i].links[j]+" from "+ecvs[i].name);
}
}
ecvlinks.push(row);
}
var ecvattns=new Array(ecvs.length).fill(0.0);
var ecvlinkattns=new Array(ecvs.length);
for (var i=0;i<ecvs.length;i++) {
ecvlinkattns[i]=new Array(ecvlinks[i].length).fill(0.0);
}
var nearestecv= -2;
var t0=Date.now();
function tick() {
const t1=Date.now();
const dt=t1-t0;
if (animating) {
var stillAnimating=false;
if (mystery) {
if (mysteryFade<1.0) stillAnimating=true;
mysteryFade=Math.min(1.0,mysteryFade+dt/mysteryFadeTime);
} else {
if (mysteryFade>1.0/1024.0) stillAnimating=true;
mysteryFade=Math.max(0.0,mysteryFade-dt/mysteryFadeTime);
}
ctx.fillStyle = "#ffffff";
ctx.fillRect(0, 0, width, height);
ctx.globalAlpha=mysteryFade;
ctx.fillStyle = "#51477d";
ctx.fillRect(0, 0, width, height);
ctx.strokeStyle='#eeeeee';
ctx.globalAlpha=0.2+0.8*mysteryFade;
ctx.lineWidth=1;
ctx.shadowOffsetX=0;
ctx.shadowOffsetY=0;
ctx.shadowBlur=mysteryFade;
ctx.shadowColor = '#eeeeee'
for (var i=0;i<ecvs.length;i++) {
const ecv=ecvs[i];
for (var j=0;j<ecvlinks[i].length;j++) {
const t=ecvlinks[i][j];
const x0=width*0.5+0.8*(ecvxys[i][0]-0.5*width);
const y0=height*0.5+0.8*(ecvxys[i][1]-0.5*height)
const x1=width*0.5+0.8*(ecvxys[t][0]-0.5*width);
const y1=height*0.5+0.8*(ecvxys[t][1]-0.5*height);
ctx.beginPath();
ctx.moveTo(x0,y0);
ctx.quadraticCurveTo(width*0.5,height*0.5,x1,y1);
ctx.stroke();
}
}
ctx.globalAlpha=1.0;
ctx.shadowBlur = 0.0;
var activeecvs=new Array(ecvs.length).fill(0);
for (var i=0;i<ecvs.length;i++) {
if (i==nearestecv || nearestecv== -1) {
activeecvs[i]=2;
for (var j=0;j<ecvlinks[i].length;j++) {
activeecvs[ecvlinks[i][j]]=Math.max(activeecvs[ecvlinks[i][j]],1);
}
}
}
for (var i=0;i<ecvattns.length;i++) {
if (activeecvs[i]>0) {
if (ecvattns[i]<1.0) stillAnimating=true;
ecvattns[i]=Math.min(1.0,ecvattns[i]+dt/fadeInTime);
} else {
if (ecvattns[i]>1.0/1024.0) stillAnimating=true;
ecvattns[i]*=Math.exp(-dt/fadeOutTime);
}
}
for (var i=0;i<ecvlinkattns.length;i++) {
for (var j=0;j<ecvlinkattns[i].length;j++) {
if (activeecvs[i]==2 && activeecvs[ecvlinks[i][j]]>0) {
if (ecvlinkattns[i][j]<1.0) stillAnimating=true;
ecvlinkattns[i][j]=Math.min(1.0,ecvlinkattns[i][j]+dt/fadeInTime);
} else {
if (ecvlinkattns[i][j]>1.0/1024.0) stillAnimating=true;
ecvlinkattns[i][j]*=Math.exp(-dt/fadeOutTime);
}
}
};
for (var i=0;i<ecvs.length;i++) {
const ecv=ecvs[i];
for (var j=0;j<ecvlinks[i].length;j++) {
const t=ecvlinks[i][j];
const x0=width*0.5+0.8*(ecvxys[i][0]-0.5*width);
const y0=height*0.5+0.8*(ecvxys[i][1]-0.5*height)
const x1=width*0.5+0.8*(ecvxys[t][0]-0.5*width);
const y1=height*0.5+0.8*(ecvxys[t][1]-0.5*height);
if (ecvlinkattns[i][j]>1.0/256.0) {
var gradient = ctx.createLinearGradient(x0,y0,x1,y1);
gradient.addColorStop(0,groupcolours[ecvs[i].group]);
gradient.addColorStop(1,groupcolours[ecvs[t].group]);
ctx.globalAlpha=ecvlinkattns[i][j];
ctx.strokeStyle=gradient;
ctx.lineWidth=2;
ctx.beginPath();
ctx.moveTo(x0,y0);
ctx.quadraticCurveTo(width*0.5,height*0.5,x1,y1);
ctx.stroke();
if (nearestecv == i ) {
ctx.globalAlpha=ecvlinkattns[i][j]*ecvlinkattns[i][j];
ctx.fillStyle=groupcolours[ecvs[t].group];
var dx=x1-width*0.5;
var dy=y1-height*0.5;
var d=Math.sqrt(dx*dx+dy*dy);
dx/=d;dy/=d;
var h=4.0;
ctx.beginPath();
ctx.moveTo(x1+1.5*h*dx,y1+1.5*h*dy);
ctx.lineTo(x1-h*dy,y1+h*dx);
ctx.lineTo(x1+h*dy,y1-h*dx);
ctx.fill();
}
}
}
ctx.globalAlpha=1.0;
}
if (!mystery) {
for (var i=0;i<ecvs.length;i++) {
const ecv=ecvs[i];
ctx.font='bold 14px Helvetica Neue,Helvetica,Arial,sans-serif';
ctx.textAlign = 'center';
ctx.textBaseline = 'middle';
ctx.globalAlpha=0.2+0.8*ecvattns[i];
ctx.fillStyle=groupcolours[ecv.group];
const lines=ecv.name.split('\n');
if (lines.length==1) {
ctx.fillText(ecv.name,ecvxys[i][0],ecvxys[i][1]);
} else {
const lineHeight=ctx.measureText('M').width*1.2
for (var line=0;line<lines.length;line++) {
ctx.fillText(lines[line],ecvxys[i][0],ecvxys[i][1]+lineHeight*(-(lines.length-1)/2.0+line));
}
}
ctx.globalAlpha=1.0;
}
}
if (!stillAnimating) {
console.log('Climate Connections animation stopping animation, for now');
}
animating=stillAnimating;
}
frame = requestAnimationFrame(tick);
t0=t1;
}
tick();
function sqr(x) {return x*x;}
canvas.onmouseout = function(e) {
mystery=true;
nearestecv = -2;
animating=true;
var elem = document.getElementById("mystery");
elem.style.visibility='visible';
elem = document.getElementById("about");
elem.textContent="";
}
canvas.onmousemove = function(e) {
animating=true;
const rect = canvas.getBoundingClientRect();
const ex = e.clientX - rect.left;
const ey = e.clientY - rect.top;
nearestecv= -1;
var nearestd=4.0*Math.sqrt(sqr(ex-0.5*width)+sqr(ey-0.5*height));
for (var i=0;i<ecvs.length;i++) {
const x=ecvxys[i][0];
const y=ecvxys[i][1];
const d=Math.sqrt(sqr(x-ex)+sqr(y-ey));
if (d<nearestd) {
nearestecv=i;
nearestd=d;
}
if (nearestecv!=previousnearestecv) {
mystery=false;
var elem = document.getElementById("mystery");
elem.style.visibility='hidden';
elem = document.getElementById("about");
if (nearestecv>=0) {
if ('about' in ecvs[nearestecv])
elem.innerHTML=ecvs[nearestecv].about;
else
elem.innerHTML=ecvs[nearestecv].name;
} else {
elem.textContent="";
elem = document.getElementById("mystery");
elem.style.visibility='visible';
}
previousnearestecv=nearestecv;
}
}
}
invalidation.then(() => cancelAnimationFrame(frame));
return canvas;
}