wind = {
const div = html`<div id="wind">`;
const context = DOM.context2d(width, height);
const canvas = context.canvas;
const scl = width / mapWidth;
context.scale(scl, scl);
d3.select(div)
.style('position', 'relative')
.style('width', '100%')
.style('height', height + 'px')
.style('pointer-events', 'none')
.append(() => canvas);
addStates(div);
const MAX_AGE = 25;
const N_GENERATIONS = 20;
const N = 120;
const DURATION = N * N_GENERATIONS;
const scale = d3.scaleLinear().domain(extent).range([-2, 2]);
const path = d3.geoPath(projection, context);
context.clearRect(0, 0, mapWidth, mapHeight)
context.strokeStyle = "rgb(200,200,200)";
context.beginPath(), path({type: "Point", coordinates: dc}), context.stroke();
let particles = [];
const n = N;
const eps = 1;
const dx = mapWidth / n;
const dy = mapHeight / n;
for (let i = 0; i < n + 1; i++) {
for (let j = 0; j < n + 1; j++) {
let x0 = (i + eps * (Math.random() - .5)) * dx;
let y0 = (j + eps * (Math.random() - .5)) * dy;
particles.push({ x0: x0, y0: y0, x: x0, y: y0, age: Math.round(MAX_AGE * Math.random()) });
}
}
while (true) {
context.beginPath();
context.strokeStyle = 'rgba(70,130,180,.4)';
particles.forEach(advanceParticle);
context.stroke();
context.fillStyle = 'rgba(255,255,255,.05)';
context.fillRect(0, 0, mapWidth, mapHeight);
yield div;
}
function advanceParticle(d) {
if (d.age++ > MAX_AGE) { d.x = d.x0; d.y = d.y0; d.age = 0; };
const coordinates = projection.invert([d.x, d.y]);
if (isNaN(coordinates[0]) | isNaN(coordinates[1])) return;
const u = bilinear(coordinates).velocity
context.moveTo(d.x, d.y)
d.x += scale(u[0]);
d.y -= scale(u[1]);
context.lineTo(d.x, d.y);
}
}