2 //= require cal-heatmap/dist/cal-heatmap
4 //= require cal-heatmap/dist/plugins/Tooltip
6 /* global CalHeatmap, Tooltip */
7 document.addEventListener("DOMContentLoaded", () => {
8 const heatmapElement = document.querySelector("#cal-heatmap");
10 if (!heatmapElement) {
14 /** @type {{date: string; max_id: number; total_changes: number}[]} */
15 const heatmapData = heatmapElement.dataset.heatmap ? JSON.parse(heatmapElement.dataset.heatmap) : [];
16 const displayName = heatmapElement.dataset.displayName;
17 const colorScheme = document.documentElement.getAttribute("data-bs-theme") ?? "auto";
18 const rangeColorsDark = ["#14432a", "#4dd05a"];
19 const rangeColorsLight = ["#4dd05a", "#14432a"];
20 const startDate = new Date(Date.now() - (365 * 24 * 60 * 60 * 1000));
21 const monthNames = OSM.i18n.t("date.abbr_month_names");
23 const mediaQuery = window.matchMedia("(prefers-color-scheme: dark)");
25 let cal = new CalHeatmap();
26 let currentTheme = getTheme();
28 function renderHeatmap() {
30 cal = new CalHeatmap();
33 itemSelector: "#cal-heatmap",
39 text: (timestamp) => monthNames[new Date(timestamp).getUTCMonth() + 1],
43 dynamicDimension: true
65 range: currentTheme === "dark" ? rangeColorsDark : rangeColorsLight,
66 domain: [0, Math.max(0, ...heatmapData.map(d => d.total_changes))]
71 text: (date, value) => getTooltipText(date, value)
75 cal.on("mouseover", (event, timestamp, value) => {
76 if (!displayName || !value) return;
77 if (event.target.parentElement.nodeName === "a") return;
79 for (const { date, max_id } of heatmapData) {
80 if (!max_id) continue;
81 if (timestamp !== Date.parse(date)) continue;
83 const params = new URLSearchParams({ before: max_id + 1 });
84 const a = document.createElementNS("http://www.w3.org/2000/svg", "a");
85 a.setAttribute("href", `/user/${encodeURIComponent(displayName)}/history?${params}`);
86 $(event.target).wrap(a);
92 function getTooltipText(date, value) {
93 const localizedDate = OSM.i18n.l("date.formats.long", date);
96 return OSM.i18n.t("javascripts.heatmap.tooltip.contributions", { count: value, date: localizedDate });
99 return OSM.i18n.t("javascripts.heatmap.tooltip.no_contributions", { date: localizedDate });
102 function getTheme() {
103 if (colorScheme === "auto") {
104 return mediaQuery.matches ? "dark" : "light";
110 if (colorScheme === "auto") {
111 mediaQuery.addEventListener("change", (e) => {
112 currentTheme = e.matches ? "dark" : "light";