mirror of
https://github.com/ClusterCockpit/cc-backend
synced 2025-04-03 02:35:55 +02:00
226 lines
6.5 KiB
Svelte
226 lines
6.5 KiB
Svelte
<!--
|
||
@component
|
||
Properties:
|
||
- Todo
|
||
-->
|
||
|
||
<script>
|
||
import uPlot from 'uplot'
|
||
import { formatNumber } from '../units.js'
|
||
import { onMount, onDestroy } from 'svelte'
|
||
import { Card } from 'sveltestrap'
|
||
|
||
export let data
|
||
export let usesBins = false
|
||
export let width = 500
|
||
export let height = 300
|
||
export let title = ''
|
||
export let xlabel = ''
|
||
export let xunit = 'X'
|
||
export let ylabel = ''
|
||
export let yunit = 'Y'
|
||
|
||
const { bars } = uPlot.paths
|
||
|
||
const drawStyles = {
|
||
bars: 1,
|
||
points: 2,
|
||
};
|
||
|
||
function paths(u, seriesIdx, idx0, idx1, extendGap, buildClip) {
|
||
let s = u.series[seriesIdx];
|
||
let style = s.drawStyle;
|
||
|
||
let renderer = ( // If bars to wide, change here
|
||
style == drawStyles.bars ? (
|
||
bars({size: [0.75, 100]})
|
||
) :
|
||
() => null
|
||
)
|
||
|
||
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 uplot = null
|
||
let timeoutId = null
|
||
|
||
function render() {
|
||
let opts = {
|
||
width: width,
|
||
height: height,
|
||
title: title,
|
||
plugins: [
|
||
legendAsTooltipPlugin()
|
||
],
|
||
cursor: {
|
||
points: {
|
||
size: (u, seriesIdx) => u.series[seriesIdx].points.size * 2.5,
|
||
width: (u, seriesIdx, size) => size / 4,
|
||
stroke: (u, seriesIdx) => u.series[seriesIdx].points.stroke(u, seriesIdx) + '90',
|
||
fill: (u, seriesIdx) => "#fff",
|
||
}
|
||
},
|
||
scales: {
|
||
x: {
|
||
time: false
|
||
},
|
||
},
|
||
axes: [
|
||
{
|
||
stroke: "#000000",
|
||
// scale: 'x',
|
||
label: xlabel,
|
||
labelGap: 10,
|
||
size: 25,
|
||
incrs: [1, 2, 5, 6, 10, 12, 50, 100, 500, 1000, 5000, 10000],
|
||
border: {
|
||
show: true,
|
||
stroke: "#000000",
|
||
},
|
||
ticks: {
|
||
width: 1 / devicePixelRatio,
|
||
size: 5 / devicePixelRatio,
|
||
stroke: "#000000",
|
||
},
|
||
values: (_, t) => t.map(v => formatNumber(v)),
|
||
},
|
||
{
|
||
stroke: "#000000",
|
||
// scale: 'y',
|
||
label: ylabel,
|
||
labelGap: 10,
|
||
size: 35,
|
||
border: {
|
||
show: true,
|
||
stroke: "#000000",
|
||
},
|
||
ticks: {
|
||
width: 1 / devicePixelRatio,
|
||
size: 5 / devicePixelRatio,
|
||
stroke: "#000000",
|
||
},
|
||
values: (_, t) => t.map(v => formatNumber(v)),
|
||
},
|
||
],
|
||
series: [
|
||
{
|
||
label: xunit !== '' ? xunit : null,
|
||
value: (u, ts, sidx, didx) => {
|
||
if (usesBins) {
|
||
const min = u.data[sidx][didx - 1] ? u.data[sidx][didx - 1] : 0
|
||
const max = u.data[sidx][didx]
|
||
ts = min + ' - ' + max // narrow spaces
|
||
}
|
||
return ts
|
||
}
|
||
},
|
||
Object.assign({
|
||
label: yunit !== '' ? yunit : null,
|
||
width: 1 / devicePixelRatio,
|
||
drawStyle: drawStyles.points,
|
||
lineInterpolation: null,
|
||
paths,
|
||
}, {
|
||
drawStyle: drawStyles.bars,
|
||
lineInterpolation: null,
|
||
stroke: "#85abce",
|
||
fill: "#85abce", // + "1A", // Transparent Fill
|
||
}),
|
||
]
|
||
};
|
||
|
||
uplot = new uPlot(opts, data, plotWrapper)
|
||
}
|
||
|
||
onMount(() => {
|
||
render()
|
||
})
|
||
|
||
onDestroy(() => {
|
||
if (uplot)
|
||
uplot.destroy()
|
||
|
||
if (timeoutId != null)
|
||
clearTimeout(timeoutId)
|
||
})
|
||
|
||
function sizeChanged() {
|
||
if (timeoutId != null)
|
||
clearTimeout(timeoutId)
|
||
|
||
timeoutId = setTimeout(() => {
|
||
timeoutId = null
|
||
if (uplot)
|
||
uplot.destroy()
|
||
|
||
render()
|
||
}, 200)
|
||
}
|
||
|
||
$: sizeChanged(width, height)
|
||
</script>
|
||
|
||
{#if data.length > 0}
|
||
<div bind:this={plotWrapper}/>
|
||
{:else}
|
||
<Card class="mx-4" body color="warning">Cannot render histogram: No data!</Card>
|
||
{/if}
|
||
|
||
|