mirror of
https://github.com/ClusterCockpit/cc-backend
synced 2025-04-03 18:55:55 +02:00
236 lines
6.0 KiB
Svelte
236 lines
6.0 KiB
Svelte
<!--
|
||
@component Histogram Plot based on uPlot Bars
|
||
|
||
Properties:
|
||
- `data [[],[]]`: uPlot data structure array ( [[],[]] == [X, Y] )
|
||
- `usesBins Bool?`: If X-Axis labels are bins ("XX-YY") [Default: false]
|
||
- `width Number?`: Plot width (reactively adaptive) [Default: 500]
|
||
- `height Number?`: Plot height (reactively adaptive) [Default: 300]
|
||
- `title String?`: Plot title [Default: ""]
|
||
- `xlabel String?`: Plot X axis label [Default: ""]
|
||
- `xunit String?`: Plot X axis unit [Default: ""]
|
||
- `ylabel String?`: Plot Y axis label [Default: ""]
|
||
- `yunit String?`: Plot Y axis unit [Default: ""]
|
||
-->
|
||
|
||
<script>
|
||
import uPlot from "uplot";
|
||
import { formatNumber } from "../units.js";
|
||
import { onMount, onDestroy } from "svelte";
|
||
import { Card } from "@sveltestrap/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 = "";
|
||
export let ylabel = "";
|
||
export let yunit = "";
|
||
|
||
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}
|