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 const heatmapData = heatmapElement.dataset.heatmap ? JSON.parse(heatmapElement.dataset.heatmap) : [];
15 const displayName = heatmapElement.dataset.displayName;
16 const colorScheme = document.documentElement.getAttribute("data-bs-theme") ?? "auto";
17 const rangeColors = ["#14432a", "#166b34", "#37a446", "#4dd05a"];
18 const startDate = new Date(Date.now() - (365 * 24 * 60 * 60 * 1000));
20 const mediaQuery = window.matchMedia("(prefers-color-scheme: dark)");
22 let cal = new CalHeatmap();
23 let currentTheme = getTheme();
25 function renderHeatmap() {
27 cal = new CalHeatmap();
30 itemSelector: "#cal-heatmap",
36 text: (timestamp) => new Date(timestamp).toLocaleString(OSM.i18n.locale, { timeZone: "UTC", month: "short" }),
40 dynamicDimension: true
62 range: currentTheme === "dark" ? rangeColors : Array.from(rangeColors).reverse(),
63 domain: [10, 20, 30, 40]
68 text: (date, value) => getTooltipText(date, value)
72 cal.on("mouseover", (event, timestamp, value) => {
73 if (!displayName || !value) return;
74 if (event.target.parentElement.nodeName === "a") return;
76 for (const { date, max_id } of heatmapData) {
77 if (!max_id) continue;
78 if (timestamp !== Date.parse(date)) continue;
80 const params = new URLSearchParams({ before: max_id + 1 });
81 const a = document.createElementNS("http://www.w3.org/2000/svg", "a");
82 a.setAttribute("href", `/user/${encodeURIComponent(displayName)}/history?${params}`);
83 $(event.target).wrap(a);
89 function getTooltipText(date, value) {
90 const localizedDate = OSM.i18n.l("date.formats.long", date);
93 return OSM.i18n.t("javascripts.heatmap.tooltip.contributions", { count: value, date: localizedDate });
96 return OSM.i18n.t("javascripts.heatmap.tooltip.no_contributions", { date: localizedDate });
100 if (colorScheme === "auto") {
101 return mediaQuery.matches ? "dark" : "light";
107 if (colorScheme === "auto") {
108 mediaQuery.addEventListener("change", (e) => {
109 currentTheme = e.matches ? "dark" : "light";