function make_time_series_graphs(variable, opts = {}) {
let {
max_size = 1100,
margin = { left: 70, right: 30, bottom: 25, top: 20 },
opacity = 0.2,
interactive = true,
top5 = true
} = opts;
let data = get_single_value(variable, hbcu_data);
let max = data.max;
if (top5) {
data = d3.sort(data, (a) => a.slice(-1)[0].value).slice(-5);
data.max = max;
}
let graph_width = width < max_size ? width : max_size;
let graph_height = 0.625 * graph_width;
let svg = d3
.create("svg")
.attr("width", graph_width)
.attr("height", graph_height)
.attr("viewBox", [0, 0, graph_width, graph_height]);
let x_scale = d3
.scaleTime()
.domain([new Date("2010"), new Date("2020")])
.range([margin.left, graph_width - margin.right]);
let y_scale = d3
.scaleLinear()
.domain([0, data.max])
.range([graph_height - margin.bottom, margin.top]);
let pts_to_path = d3
.line()
.x((d) => x_scale(d.year))
.y((d) => y_scale(d.value));
let axes = svg.append("g");
let x_axis = d3.axisBottom(x_scale);
axes
.append("g")
.attr("transform", `translate(0,${graph_height - margin.bottom})`)
.call(x_axis.tickSizeOuter(0));
let y_axis = d3.axisLeft(y_scale);
axes
.append("g")
.attr("transform", `translate(${margin.left})`)
.call(y_axis.tickSizeOuter(0));
let time_series_plots = svg.append("g");
let text_group = svg
.append("text")
.attr("x", 1.3 * margin.left)
.attr("y", 1.5 * margin.top)
.attr("font-size", 16)
.attr("text-anchor", "start")
.attr("fill", "steelblue")
.text("Avg highlighted");
data.forEach(function (d) {
time_series_plots
.append("path")
.attr("d", pts_to_path(d))
.attr("class", "time_series")
.attr("stroke", "black")
.attr("id", `uid${d[0].UnitID}`)
.attr("stroke-opacity", opacity)
.attr("stroke-width", 0.7)
.attr("stroke-linejoin", "round")
.attr("fill", "none");
});
let get_mean_values = (date) =>
d3.mean(
data
.map(function (d) {
let dd = d.filter((o) => o.year.getTime() == date.getTime());
return dd;
})
.filter((a) => a.length > 0)
.map((a) => a[0].value)
);
let means = d3
.range(2010, 2021)
.map((d) => new Date(d.toString()))
.map((d) => ({ value: get_mean_values(d), year: d }));
time_series_plots
.append("path")
.attr("d", pts_to_path(means))
.attr("class", "avg_time_series")
.attr("stroke", "steelblue")
.attr("id", "avg")
.attr("stroke-opacity", 1)
.attr("stroke-width", 3)
.attr("stroke-linejoin", "round")
.attr("fill", "none");
svg
.on("touchmove", (e) => e.preventDefault()) // prevent scrolling
.on("pointermove", function (evt) {
let p = d3.pointer(evt);
let x = p[0];
let y = p[1];
let day = x_scale.invert(x);
let value = y_scale.invert(y);
let closest = get_closest(day, value, data);
if (closest) {
time_series_plots
.selectAll("path.time_series")
.attr("stroke-opacity", opacity)
.attr("stroke-width", 0.7)
.attr("stroke", "black");
time_series_plots
.select(`#uid${closest.uid}`)
.attr("stroke-opacity", 1)
.attr("stroke-width", 6)
.attr("stroke", "steelblue")
.raise();
text_group.text(
hbcu_data.filter((o) => o.UnitID == closest.uid)[0][
"Institution Name"
]
);
}
})
.on("pointerleave", function () {
time_series_plots
.selectAll("path.time_series")
.attr("stroke-opacity", opacity)
.attr("stroke-width", 0.7)
.attr("stroke", "black");
time_series_plots
.select("#avg")
.attr("stroke-opacity", 1)
.attr("stroke-width", 3)
.attr("stroke", "steelblue")
.raise();
text_group.text("Avg highlighted");
});
return svg.node();
}