{
const svg = d3.select(DOM.svg(500, 500));
svg
.append("path")
.attr("fill", "none")
.attr("stroke", "black")
.attr("stroke-width", 10)
.attr("d", line(data));
svg
.append("polyline")
.attr("fill", "none")
.attr("stroke", "blue")
.attr("stroke-width", 3)
.attr(
"points",
interpPts.map((d) => `${scaleX(d[0])},${scaleY(d[1])} `).join("")
);
svg
.append("circle")
.attr("cx", scaleX(t))
.attr("cy", scaleY(interpolate(t)))
.attr("fill", "blue")
.attr("r", 15);
const circles = svg
.append("g")
.selectAll("circle")
.data(data)
.join("circle")
.attr("cx", (d) => scaleX(d[0]))
.attr("cy", (d) => scaleY(d[1]))
.attr("r", 10)
.attr("fill", "red")
.attr("cursor", "grab")
.call(
d3.drag().subject(dragSubject).on("drag", dragged).on("end", dragEnd)
);
function dragSubject() {
const { x, y } = d3.event;
return {
dx: x - d3.select(this).attr("cx"),
dy: y - d3.select(this).attr("cy")
};
}
function dragged() {
const {
x,
y,
subject: { dx, dy }
} = d3.event;
const cx = Math.min(Math.max(x - dx, 0), 500);
const cy = Math.min(Math.max(y - dy, 0), 500);
d3.select(this).attr("cx", cx).attr("cy", cy).attr("cursor", "grabbing");
}
function dragEnd(d, i) {
const cx = d3.select(this).attr("cx");
const cy = d3.select(this).attr("cy");
const newdata = data.slice();
newdata.splice(i, 1, [scaleX.invert(cx), scaleY.invert(cy)]);
newdata.sort((a, b) => a[0] - b[0]);
mutable data = newdata;
circles.attr("cursor", "grab");
}
return svg.node();
}