|
|
@ -46,48 +46,115 @@ const zoom_plugin = { |
|
|
|
mode: "x", |
|
|
|
mode: "x", |
|
|
|
}, |
|
|
|
}, |
|
|
|
} |
|
|
|
} |
|
|
|
const trigger_legend_hover = function (e, legendItem, legend) { |
|
|
|
|
|
|
|
if (legendItem.hidden) { |
|
|
|
let hovered = false; |
|
|
|
|
|
|
|
const trigger_legend_hover = function (legendItem, chart) { |
|
|
|
|
|
|
|
if (legendItem.hidden || hovered) { |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
legend.chart.data.datasets.forEach((dataset, i) => { |
|
|
|
hovered = true; |
|
|
|
|
|
|
|
chart.data.datasets.forEach((dataset, i) => { |
|
|
|
dataset.backgroundColor = (legendItem.datasetIndex === i || dataset.backgroundColor.length === 9) ? dataset.backgroundColor : dataset.backgroundColor + '3D'; |
|
|
|
dataset.backgroundColor = (legendItem.datasetIndex === i || dataset.backgroundColor.length === 9) ? dataset.backgroundColor : dataset.backgroundColor + '3D'; |
|
|
|
dataset.borderColor = (legendItem.datasetIndex === i || dataset.borderColor.length === 9) ? dataset.borderColor : dataset.borderColor + '3D'; |
|
|
|
dataset.borderColor = (legendItem.datasetIndex === i || dataset.borderColor.length === 9) ? dataset.borderColor : dataset.borderColor + '3D'; |
|
|
|
dataset.borderWidth = legendItem.datasetIndex === i ? 4 : 3; |
|
|
|
dataset.borderWidth = legendItem.datasetIndex === i ? 4 : 3; |
|
|
|
dataset.pointRadius = dataset.pointRadius===0 ? 0 : 2; |
|
|
|
dataset.pointRadius = dataset.pointRadius===0 ? 0 : 2; |
|
|
|
}); |
|
|
|
}); |
|
|
|
legend.chart.update(); |
|
|
|
chart.update(); |
|
|
|
} |
|
|
|
} |
|
|
|
const trigger_legend_leave = function (e, legendItem, legend) { |
|
|
|
const trigger_legend_leave = function (chart) { |
|
|
|
legend.chart.data.datasets.forEach((dataset, i) => { |
|
|
|
chart.data.datasets.forEach((dataset, i) => { |
|
|
|
dataset.backgroundColor = dataset.backgroundColor.length === 9 ? dataset.backgroundColor.slice(0, -2) : dataset.backgroundColor; |
|
|
|
dataset.backgroundColor = dataset.backgroundColor.length === 9 ? dataset.backgroundColor.slice(0, -2) : dataset.backgroundColor; |
|
|
|
dataset.borderColor = dataset.borderColor.length === 9 ? dataset.borderColor.slice(0, -2) : dataset.borderColor; |
|
|
|
dataset.borderColor = dataset.borderColor.length === 9 ? dataset.borderColor.slice(0, -2) : dataset.borderColor; |
|
|
|
dataset.borderWidth = 3; |
|
|
|
dataset.borderWidth = 3; |
|
|
|
}); |
|
|
|
}); |
|
|
|
legend.chart.update(); |
|
|
|
chart.update(); |
|
|
|
} |
|
|
|
hovered = false; |
|
|
|
const legend_plugin = { |
|
|
|
} |
|
|
|
position: "bottom", |
|
|
|
const trigger_legend_click = function (item) { |
|
|
|
labels: { |
|
|
|
combined_charts.forEach(c_chart => { |
|
|
|
padding: 10, |
|
|
|
if (c_chart.isDatasetVisible(item.datasetIndex)) { |
|
|
|
useBorderRadius: true, |
|
|
|
trigger_legend_leave(c_chart) |
|
|
|
borderRadius: 2, |
|
|
|
c_chart.hide(item.datasetIndex); |
|
|
|
boxWidth: 20, |
|
|
|
item.hidden = true; |
|
|
|
}, |
|
|
|
|
|
|
|
onHover: trigger_legend_hover, |
|
|
|
|
|
|
|
onLeave: trigger_legend_leave, |
|
|
|
|
|
|
|
onClick: function (e, legendItem, legend) { |
|
|
|
|
|
|
|
if (legend.chart.isDatasetVisible(legendItem.datasetIndex)) { |
|
|
|
|
|
|
|
trigger_legend_leave(e, legendItem, legend); |
|
|
|
|
|
|
|
legend.chart.hide(legendItem.datasetIndex); |
|
|
|
|
|
|
|
legendItem.hidden = true; |
|
|
|
|
|
|
|
} else { |
|
|
|
} else { |
|
|
|
legend.chart.show(legendItem.datasetIndex); |
|
|
|
c_chart.show(item.datasetIndex); |
|
|
|
legendItem.hidden = false; |
|
|
|
item.hidden = false; |
|
|
|
trigger_legend_hover(e, legendItem, legend); |
|
|
|
trigger_legend_hover(item, c_chart) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
}) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
// break hover state when mouseout fails to trigger
|
|
|
|
|
|
|
|
window.onmousemove = (e) => { |
|
|
|
|
|
|
|
if (hovered && (e.target.closest("ul") === null || !e.target.closest("ul").classList.contains("legend-list"))) { |
|
|
|
|
|
|
|
console.log("breaking hover") |
|
|
|
|
|
|
|
combined_charts.forEach(chart => { |
|
|
|
|
|
|
|
trigger_legend_leave(chart); |
|
|
|
|
|
|
|
}) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
const get_create_LegendList = (chart, id) => { |
|
|
|
|
|
|
|
let created = false; |
|
|
|
|
|
|
|
const legendContainer = document.getElementById(id); |
|
|
|
|
|
|
|
let listContainer = legendContainer.querySelector("ul"); |
|
|
|
|
|
|
|
if (!listContainer) { |
|
|
|
|
|
|
|
created = true; |
|
|
|
|
|
|
|
listContainer = document.createElement("ul"); |
|
|
|
|
|
|
|
listContainer.classList.add('legend-list'); |
|
|
|
|
|
|
|
legendContainer.appendChild(listContainer); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
return {list: listContainer, created: created}; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
const htmlLegendPlugin = { |
|
|
|
|
|
|
|
id: 'htmlLegend', |
|
|
|
|
|
|
|
afterUpdate(chart, args, options) { |
|
|
|
|
|
|
|
//console.log("update")
|
|
|
|
|
|
|
|
const legendList = get_create_LegendList(chart, options.containerID); |
|
|
|
|
|
|
|
const ul = legendList.list; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Reuse the built-in legendItems generator
|
|
|
|
|
|
|
|
const items = chart.options.plugins.legend.labels.generateLabels(chart); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
items.forEach(item => { |
|
|
|
|
|
|
|
if (legendList.created) { |
|
|
|
|
|
|
|
const li = document.createElement('li'); |
|
|
|
|
|
|
|
li.classList.add("legend-element"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
li.onmouseover = () => {trigger_legend_hover(item, chart)}; |
|
|
|
|
|
|
|
li.onmouseout = () => {trigger_legend_leave(chart)}; |
|
|
|
|
|
|
|
li.onclick = () => {trigger_legend_click(item)}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Color box
|
|
|
|
|
|
|
|
const boxSpan = document.createElement('span'); |
|
|
|
|
|
|
|
boxSpan.style.background = item.fillStyle; |
|
|
|
|
|
|
|
boxSpan.style.borderColor = item.strokeStyle; |
|
|
|
|
|
|
|
boxSpan.style.borderWidth = item.lineWidth + 'px'; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Text
|
|
|
|
|
|
|
|
const textContainer = document.createElement('p'); |
|
|
|
|
|
|
|
textContainer.style.color = item.fontColor; |
|
|
|
|
|
|
|
textContainer.style.textDecoration = item.hidden ? 'line-through' : ''; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const text = document.createTextNode(item.text); |
|
|
|
|
|
|
|
textContainer.appendChild(text); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
li.appendChild(boxSpan); |
|
|
|
|
|
|
|
li.appendChild(textContainer); |
|
|
|
|
|
|
|
ul.appendChild(li); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
const list_element = ul.querySelector(`li.legend-element:nth-of-type(${item.datasetIndex+1})`); |
|
|
|
|
|
|
|
const boxSpan = list_element.querySelector("span"); |
|
|
|
|
|
|
|
const textContainer = list_element.querySelector("p"); |
|
|
|
|
|
|
|
boxSpan.style.background = item.fillStyle; |
|
|
|
|
|
|
|
boxSpan.style.borderColor = item.strokeStyle; |
|
|
|
|
|
|
|
boxSpan.style.borderWidth = item.lineWidth + 'px'; |
|
|
|
|
|
|
|
textContainer.style.color = item.fontColor; |
|
|
|
|
|
|
|
textContainer.style.textDecoration = item.hidden ? 'line-through' : ''; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
window.onload = create_charts; |
|
|
|
window.onload = create_charts; |
|
|
|
|
|
|
|
|
|
|
@ -205,7 +272,12 @@ async function create_charts() { |
|
|
|
plugins: { |
|
|
|
plugins: { |
|
|
|
tooltip: tooltip_plugin, |
|
|
|
tooltip: tooltip_plugin, |
|
|
|
zoom: {...zoom_plugin}, |
|
|
|
zoom: {...zoom_plugin}, |
|
|
|
legend: legend_plugin, |
|
|
|
legend: { |
|
|
|
|
|
|
|
display: false, |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
htmlLegend: { |
|
|
|
|
|
|
|
containerID: 'combined-chart-legend', |
|
|
|
|
|
|
|
} |
|
|
|
}, |
|
|
|
}, |
|
|
|
scales: { |
|
|
|
scales: { |
|
|
|
x: x_scale, |
|
|
|
x: x_scale, |
|
|
@ -221,7 +293,8 @@ async function create_charts() { |
|
|
|
}, |
|
|
|
}, |
|
|
|
responsive: true, |
|
|
|
responsive: true, |
|
|
|
maintainAspectRatio: false, |
|
|
|
maintainAspectRatio: false, |
|
|
|
} |
|
|
|
}, |
|
|
|
|
|
|
|
plugins: [htmlLegendPlugin], |
|
|
|
})); |
|
|
|
})); |
|
|
|
combined_charts.push(new Chart(`progress-chart-combined-progress`, { |
|
|
|
combined_charts.push(new Chart(`progress-chart-combined-progress`, { |
|
|
|
type: "line", |
|
|
|
type: "line", |
|
|
@ -232,7 +305,12 @@ async function create_charts() { |
|
|
|
plugins: { |
|
|
|
plugins: { |
|
|
|
tooltip: tooltip_plugin, |
|
|
|
tooltip: tooltip_plugin, |
|
|
|
zoom: {...zoom_plugin}, |
|
|
|
zoom: {...zoom_plugin}, |
|
|
|
legend: legend_plugin, |
|
|
|
legend: { |
|
|
|
|
|
|
|
display: false, |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
htmlLegend: { |
|
|
|
|
|
|
|
containerID: 'combined-chart-legend', |
|
|
|
|
|
|
|
} |
|
|
|
}, |
|
|
|
}, |
|
|
|
scales: { |
|
|
|
scales: { |
|
|
|
x: x_scale, |
|
|
|
x: x_scale, |
|
|
@ -242,7 +320,8 @@ async function create_charts() { |
|
|
|
}, |
|
|
|
}, |
|
|
|
responsive: true, |
|
|
|
responsive: true, |
|
|
|
maintainAspectRatio: false, |
|
|
|
maintainAspectRatio: false, |
|
|
|
} |
|
|
|
}, |
|
|
|
|
|
|
|
plugins: [htmlLegendPlugin], |
|
|
|
})); |
|
|
|
})); |
|
|
|
minval -= 7200000; |
|
|
|
minval -= 7200000; |
|
|
|
maxval += 7200000; |
|
|
|
maxval += 7200000; |
|
|
@ -384,9 +463,11 @@ function toggle_combined_chart() { |
|
|
|
buttons.forEach(element => element.classList.remove("dropdown-open")); |
|
|
|
buttons.forEach(element => element.classList.remove("dropdown-open")); |
|
|
|
chart.classList.remove("closed"); |
|
|
|
chart.classList.remove("closed"); |
|
|
|
this.classList.add("dropdown-open"); |
|
|
|
this.classList.add("dropdown-open"); |
|
|
|
|
|
|
|
document.querySelector("#combined-chart-legend").classList.remove("hidden"); |
|
|
|
} else { |
|
|
|
} else { |
|
|
|
chart.classList.add("closed"); |
|
|
|
chart.classList.add("closed"); |
|
|
|
this.classList.remove("dropdown-open"); |
|
|
|
this.classList.remove("dropdown-open"); |
|
|
|
|
|
|
|
document.querySelector("#combined-chart-legend").classList.add("hidden"); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -471,7 +552,7 @@ async function update_leaderboard_elements() { |
|
|
|
// replace updated Leaderboard
|
|
|
|
// replace updated Leaderboard
|
|
|
|
document.querySelector(".leaderboard-list").outerHTML = leaderboard_list; |
|
|
|
document.querySelector(".leaderboard-list").outerHTML = leaderboard_list; |
|
|
|
// reapply EventListeners
|
|
|
|
// reapply EventListeners
|
|
|
|
document.querySelectorAll("button.leaderboard-element").forEach(element => element.addEventListener("click", toggle_leaderboard_chart)); |
|
|
|
document.querySelectorAll("button.leaderboard-element").forEach(element => element.addEventListener("mousedown", toggle_leaderboard_chart)); |
|
|
|
// recreate charts
|
|
|
|
// recreate charts
|
|
|
|
create_charts(); |
|
|
|
create_charts(); |
|
|
|
}) |
|
|
|
}) |
|
|
|