canvasMap = ()=>{
const projection = d3.geoEquirectangular()
const points = {type: "MultiPoint", coordinates: mData.map(d => [d.longitude, d.latitude])};
const margin = 10;
const [[x0, y0], [x1, y1]] = d3.geoPath(projection.fitWidth(width - margin * 2, points)).bounds(points);
const [tx, ty] = projection.translate();
let height = Math.ceil(y1 - y0);
projection.translate([tx + margin, ty + margin]);
height = height + margin * 2;
const context = DOM.context2d(width, height);
const path = d3.geoPath(projection, context);
context.canvas.style.maxWidth = "100%";
context.fillRect(0, 0, width, height);
context.strokeStyle = "#eee";
context.lineWidth = 1.5;
context.lineJoin = "round";
context.beginPath(), path(mLand), context.stroke();
for (const {longitude, latitude, speed, dir} of mData) {
context.save();
context.translate(...projection([longitude, latitude]));
context.scale(areaScale(speed), areaScale(speed));
context.rotate(dir * Math.PI / 180);
const arrowBaseY = 3;
context.beginPath();
context.moveTo(-2, arrowBaseY);
context.lineTo(2, arrowBaseY);
context.lineTo(0, 8);
context.closePath();
context.fillStyle = colorScale(dir);
context.fill();
context.beginPath();
context.moveTo(0, arrowBaseY);
context.lineTo(0, -2);
context.strokeStyle = colorScale(dir)
context.stroke();
context.restore();
}
return context.canvas;
}