feat: add hover-legend to histograms & metricplots

This commit is contained in:
Christoph Kluge 2023-08-08 13:27:01 +02:00
parent 742c2e399e
commit 423e800d9e
7 changed files with 151 additions and 33 deletions

View File

@ -318,6 +318,7 @@
scopes={item.data.map((x) => x.scope)} scopes={item.data.map((x) => x.scope)}
{width} {width}
isShared={$initq.data.job.exclusive != 1} isShared={$initq.data.job.exclusive != 1}
resources={$initq.data.job.resources}
/> />
{:else} {:else}
<Card body color="warning" <Card body color="warning"

View File

@ -89,6 +89,7 @@
timestep={data.timestep} timestep={data.timestep}
scope={selectedScope} metric={metricName} scope={selectedScope} metric={metricName}
series={series} series={series}
isShared={isShared} /> isShared={isShared}
resources={job.resources} />
{/if} {/if}
{/key} {/key}

View File

@ -203,6 +203,7 @@
subCluster={$nodeMetricsData.data.nodeMetrics[0] subCluster={$nodeMetricsData.data.nodeMetrics[0]
.subCluster} .subCluster}
series={item.metric.series} series={item.metric.series}
resources={[{hostname: hostname}]}
/> />
{:else if item.disabled === true && item.metric} {:else if item.disabled === true && item.metric}
<Card <Card

View File

@ -136,7 +136,8 @@
series={item.data.metric.series} series={item.data.metric.series}
metric={item.data.name} metric={item.data.name}
cluster={clusters.find(c => c.name == cluster)} cluster={clusters.find(c => c.name == cluster)}
subCluster={item.subCluster} /> subCluster={item.subCluster}
resources={[{hostname: item.host}]}/>
{:else if item.disabled === true && item.data} {:else if item.disabled === true && item.data}
<Card style="margin-left: 2rem;margin-right: 2rem;" body color="info">Metric disabled for subcluster <code>{selectedMetric}:{item.subCluster}</code></Card> <Card style="margin-left: 2rem;margin-right: 2rem;" body color="info">Metric disabled for subcluster <code>{selectedMetric}:{item.subCluster}</code></Card>
{:else} {:else}

View File

@ -136,6 +136,7 @@
{cluster} {cluster}
subCluster={job.subCluster} subCluster={job.subCluster}
isShared={(job.exclusive != 1)} isShared={(job.exclusive != 1)}
resources={job.resources}
/> />
{:else if metric.disabled == true && metric.data} {:else if metric.disabled == true && metric.data}
<Card body color="info">Metric disabled for subcluster <code>{metric.data.name}:{job.subCluster}</code></Card> <Card body color="info">Metric disabled for subcluster <code>{metric.data.name}:{job.subCluster}</code></Card>

View File

@ -39,8 +39,62 @@
return renderer(u, seriesIdx, idx0, idx1, extendGap, buildClip); return renderer(u, seriesIdx, idx0, idx1, extendGap, buildClip);
} }
// converts the legend into a simple tooltip
function legendAsTooltipPlugin({ className, style = { backgroundColor:"rgba(255, 249, 196, 0.92)", color: "black" } } = {}) {
let legendEl;
function init(u, opts) {
legendEl = u.root.querySelector(".u-legend");
legendEl.classList.remove("u-inline");
className && legendEl.classList.add(className);
uPlot.assign(legendEl.style, {
textAlign: "left",
pointerEvents: "none",
display: "none",
position: "absolute",
left: 0,
top: 0,
zIndex: 100,
boxShadow: "2px 2px 10px rgba(0,0,0,0.5)",
...style
});
// hide series color markers
const idents = legendEl.querySelectorAll(".u-marker");
for (let i = 0; i < idents.length; i++)
idents[i].style.display = "none";
const overEl = u.over;
overEl.style.overflow = "visible";
// move legend into plot bounds
overEl.appendChild(legendEl);
// show/hide tooltip on enter/exit
overEl.addEventListener("mouseenter", () => {legendEl.style.display = null;});
overEl.addEventListener("mouseleave", () => {legendEl.style.display = "none";});
// let tooltip exit plot
// overEl.style.overflow = "visible";
}
function update(u) {
const { left, top } = u.cursor;
legendEl.style.transform = "translate(" + (left + 15) + "px, " + (top + 15) + "px)";
}
return {
hooks: {
init: init,
setCursor: update,
}
};
}
let plotWrapper = null let plotWrapper = null
let legendWrapper = null
let uplot = null let uplot = null
let timeoutId = null let timeoutId = null
@ -48,6 +102,9 @@
let opts = { let opts = {
width: width, width: width,
height: height, height: height,
plugins: [
legendAsTooltipPlugin()
],
cursor: { cursor: {
points: { points: {
size: (u, seriesIdx) => u.series[seriesIdx].points.size * 2.5, size: (u, seriesIdx) => u.series[seriesIdx].points.size * 2.5,
@ -98,15 +155,6 @@
values: (_, t) => t.map(v => formatNumber(v)), values: (_, t) => t.map(v => formatNumber(v)),
}, },
], ],
legend : {
mount: (self, legend) => {
legendWrapper.appendChild(legend)
},
markers: {
show: false,
stroke: "#000000"
}
},
series: [ series: [
{ {
label: xunit, label: xunit,
@ -158,9 +206,7 @@
</script> </script>
{#if data.length > 0} {#if data.length > 0}
<div bind:this={plotWrapper}> <div bind:this={plotWrapper}/>
<div bind:this={legendWrapper}/>
</div>
{:else} {:else}
<Card class="mx-4" body color="warning">Cannot render histogram: No data!</Card> <Card class="mx-4" body color="warning">Cannot render histogram: No data!</Card>
{/if} {/if}

View File

@ -26,16 +26,17 @@
import { getContext, onMount, onDestroy } from 'svelte' import { getContext, onMount, onDestroy } from 'svelte'
import { Card } from 'sveltestrap' import { Card } from 'sveltestrap'
export let metric
export let scope = 'node'
export let resources = []
export let width export let width
export let height export let height
export let timestep export let timestep
export let series export let series
export let useStatsSeries = null
export let statisticsSeries = null export let statisticsSeries = null
export let cluster export let cluster
export let subCluster export let subCluster
export let metric
export let useStatsSeries = null
export let scope = 'node'
export let isShared = false export let isShared = false
if (useStatsSeries == null) if (useStatsSeries == null)
@ -53,6 +54,62 @@
const backgroundColors = { normal: 'rgba(255, 255, 255, 1.0)', caution: 'rgba(255, 128, 0, 0.3)', alert: 'rgba(255, 0, 0, 0.3)' } const backgroundColors = { normal: 'rgba(255, 255, 255, 1.0)', caution: 'rgba(255, 128, 0, 0.3)', alert: 'rgba(255, 0, 0, 0.3)' }
const thresholds = findThresholds(metricConfig, scope, typeof subCluster == 'string' ? cluster.subClusters.find(sc => sc.name == subCluster) : subCluster) const thresholds = findThresholds(metricConfig, scope, typeof subCluster == 'string' ? cluster.subClusters.find(sc => sc.name == subCluster) : subCluster)
// converts the legend into a simple tooltip
function legendAsTooltipPlugin({ className, style = { backgroundColor:"rgba(255, 249, 196, 0.92)", color: "black" } } = {}) {
let legendEl;
function init(u, opts) {
legendEl = u.root.querySelector(".u-legend");
legendEl.classList.remove("u-inline");
className && legendEl.classList.add(className);
uPlot.assign(legendEl.style, {
textAlign: "left",
pointerEvents: "none",
display: "none",
position: "absolute",
left: 0,
top: 0,
zIndex: 100,
boxShadow: "2px 2px 10px rgba(0,0,0,0.5)",
...style
});
// hide series color markers
const idents = legendEl.querySelectorAll(".u-marker");
for (let i = 0; i < idents.length; i++)
idents[i].style.display = "none";
const overEl = u.over;
overEl.style.overflow = "visible";
// move legend into plot bounds
overEl.appendChild(legendEl);
// show/hide tooltip on enter/exit
overEl.addEventListener("mouseenter", () => {legendEl.style.display = null;});
overEl.addEventListener("mouseleave", () => {legendEl.style.display = "none";});
// let tooltip exit plot
// overEl.style.overflow = "visible";
}
function update(u) {
const { left, top } = u.cursor;
const width = u.over.querySelector(".u-legend").offsetWidth;
legendEl.style.transform = "translate(" + (left - width - 15) + "px, " + (top + 15) + "px)";
}
return {
hooks: {
init: init,
setCursor: update,
}
};
}
function backgroundColor() { function backgroundColor() {
if (clusterCockpitConfig.plot_general_colorBackground == false if (clusterCockpitConfig.plot_general_colorBackground == false
|| !thresholds || !thresholds
@ -93,7 +150,7 @@
? (statisticsSeries.max.reduce((max, x) => Math.max(max, x), thresholds.normal) || thresholds.normal) ? (statisticsSeries.max.reduce((max, x) => Math.max(max, x), thresholds.normal) || thresholds.normal)
: (series.reduce((max, series) => Math.max(max, series.statistics?.max), thresholds.normal) || thresholds.normal) : (series.reduce((max, series) => Math.max(max, series.statistics?.max), thresholds.normal) || thresholds.normal)
: null : null
const plotSeries = [{}] const plotSeries = [{label: 'Runtime', value: (u, ts, sidx, didx) => didx == null ? null : formatTime(ts)}]
const plotData = [new Array(longestSeries)] const plotData = [new Array(longestSeries)]
for (let i = 0; i < longestSeries; i++) // TODO: Cache/Reuse this array? for (let i = 0; i < longestSeries; i++) // TODO: Cache/Reuse this array?
plotData[0][i] = i * timestep plotData[0][i] = i * timestep
@ -103,9 +160,9 @@
plotData.push(statisticsSeries.min) plotData.push(statisticsSeries.min)
plotData.push(statisticsSeries.max) plotData.push(statisticsSeries.max)
plotData.push(statisticsSeries.mean) plotData.push(statisticsSeries.mean)
plotSeries.push({ scale: 'y', width: lineWidth, stroke: 'red' }) plotSeries.push({ label: 'min', scale: 'y', width: lineWidth, stroke: 'red' })
plotSeries.push({ scale: 'y', width: lineWidth, stroke: 'green' }) plotSeries.push({ label: 'max', scale: 'y', width: lineWidth, stroke: 'green' })
plotSeries.push({ scale: 'y', width: lineWidth, stroke: 'black' }) plotSeries.push({ label: 'mean', scale: 'y', width: lineWidth, stroke: 'black' })
plotBands = [ plotBands = [
{ series: [2,3], fill: 'rgba(0,255,0,0.1)' }, { series: [2,3], fill: 'rgba(0,255,0,0.1)' },
{ series: [3,1], fill: 'rgba(255,0,0,0.1)' } { series: [3,1], fill: 'rgba(255,0,0,0.1)' }
@ -114,6 +171,9 @@
for (let i = 0; i < series.length; i++) { for (let i = 0; i < series.length; i++) {
plotData.push(series[i].data) plotData.push(series[i].data)
plotSeries.push({ plotSeries.push({
label: scope === 'node' ? resources[i].hostname :
// scope === 'accelerator' ? resources[0].accelerators[i] :
scope + ' #' + (i+1),
scale: 'y', scale: 'y',
width: lineWidth, width: lineWidth,
stroke: lineColor(i, series.length) stroke: lineColor(i, series.length)
@ -124,6 +184,9 @@
const opts = { const opts = {
width, width,
height, height,
plugins: [
legendAsTooltipPlugin()
],
series: plotSeries, series: plotSeries,
axes: [ axes: [
{ {
@ -177,8 +240,7 @@
x: { time: false }, x: { time: false },
y: maxY ? { range: [0., maxY * 1.1] } : {} y: maxY ? { range: [0., maxY * 1.1] } : {}
}, },
cursor: { drag: { x: true, y: true } }, cursor: { drag: { x: true, y: true } }
legend: { show: false, live: false }
} }
// console.log(opts) // console.log(opts)
@ -249,16 +311,21 @@
} }
</script> </script>
<script context="module"> <script context="module">
export function formatTime(t) { export function formatTime(t) {
let h = Math.floor(t / 3600) if (t !== null) {
let m = Math.floor((t % 3600) / 60) if (isNaN(t)) {
if (h == 0) return t
return `${m}m` } else {
else if (m == 0) let h = Math.floor(t / 3600)
return `${h}h` let m = Math.floor((t % 3600) / 60)
else if (h == 0)
return `${h}:${m}h` return `${m}m`
else if (m == 0)
return `${h}h`
else
return `${h}:${m}h`
}
}
} }
export function timeIncrs(timestep, maxX) { export function timeIncrs(timestep, maxX) {