function create_animationFrames(data) {
let geojson = data.geojson,
size = data.size,
waterlines_total = data.waterlines_total,
colour = data.colour,
waterlines_spacingExponent = data.waterlines_spacingExponent,
radial_distance = data.radial_distance,
frames_total = data.frames_total;
geojson = turf.rewind(geojson, { reverse: true });
let projection_canvas = d3
.geoMercator()
.fitSize([size - radial_distance, size - radial_distance], geojson);
let projection_translate = projection_canvas.translate();
projection_canvas = projection_canvas.translate([
projection_translate[0] + radial_distance / 2,
projection_translate[1] + radial_distance / 2
]);
function curveContext(curve) {
return {
moveTo(x, y) {
curve.lineStart();
curve.point(x, y);
},
lineTo(x, y) {
curve.point(x, y);
},
closePath() {
curve.lineEnd();
}
};
}
function geoCurvePath(curve, projection, context) {
return object => {
const pathContext = context === undefined ? d3.path() : context;
d3.geoPath(projection_canvas, curveContext(curve(pathContext)))(object);
return context === undefined ? pathContext + "" : undefined;
};
}
let curve_this = d3.curveCatmullRomClosed,
path_svg = geoCurvePath(curve_this)(geojson),
path = new Path2D(path_svg);
let waterlines_spacing = d3
.scalePow()
.exponent(waterlines_spacingExponent)
.domain([0, waterlines_total - 1])
.range([radial_distance, 2]);
let waterline_index_array = Array.from(Array(waterlines_total).keys()),
waterline_spacing_array = waterline_index_array.map(d =>
waterlines_spacing(d)
);
let frame_paths = [];
for (let j = 0; j < frames_total; j++) {
let new_waterline_spacing_array = waterline_spacing_array
.slice(0, waterline_spacing_array.length - 1)
.map((d, i) =>
d3.interpolateNumber(waterline_spacing_array[i + 1], d)(
j / frames_total
)
);
new_waterline_spacing_array.push(
d3.interpolateNumber(0, 2)(j / frames_total)
);
frame_paths.push(new_waterline_spacing_array);
}
let animation_frames = [];
for (let j = 0; j < frames_total; j++) {
const canvas = DOM.canvas(size, size);
const context = canvas.getContext('2d');
const path_orig = d3.geoPath(projection_canvas, context);
for (let i = 0; i < waterlines_total; i++) {
context.beginPath();
context.strokeStyle = colour;
if (i == 0) {
context.globalAlpha = 1 - j / frames_total;
} else if (i == waterlines_total - 1) {
context.globalAlpha = j / frames_total;
} else {
context.globalAlpha = 1;
}
context.lineWidth = frame_paths[j][i] + 2;
context.stroke(path);
context.beginPath();
context.globalAlpha = 1;
context.lineWidth = frame_paths[j][i];
context.globalCompositeOperation = 'destination-out';
context.stroke(path);
context.globalCompositeOperation = 'source-over';
}
context.beginPath();
path_orig(geojson);
context.lineWidth = 2;
context.fillStyle = colour;
context.fill();
animation_frames.push(context.getImageData(0, 0, size, size));
}
return animation_frames;
}