2023-07-26 13:44:06 +02:00
|
|
|
<!--
|
|
|
|
@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 width = 500
|
|
|
|
export let height = 300
|
2023-08-09 12:42:25 +02:00
|
|
|
export let title = ''
|
2023-07-26 13:44:06 +02:00
|
|
|
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;
|
|
|
|
|
2023-08-09 12:42:25 +02:00
|
|
|
let renderer = ( // If bars to wide, change here
|
2023-07-26 13:44:06 +02:00
|
|
|
style == drawStyles.bars ? (
|
|
|
|
bars({size: [0.75, 100]})
|
|
|
|
) :
|
|
|
|
() => null
|
|
|
|
)
|
|
|
|
|
|
|
|
return renderer(u, seriesIdx, idx0, idx1, extendGap, buildClip);
|
|
|
|
}
|
|
|
|
|
2023-08-08 13:27:01 +02:00
|
|
|
// 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,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-07-26 13:44:06 +02:00
|
|
|
let plotWrapper = null
|
|
|
|
let uplot = null
|
|
|
|
let timeoutId = null
|
|
|
|
|
|
|
|
function render() {
|
|
|
|
let opts = {
|
|
|
|
width: width,
|
|
|
|
height: height,
|
2023-08-09 12:42:25 +02:00
|
|
|
title: title,
|
2023-08-08 13:27:01 +02:00
|
|
|
plugins: [
|
|
|
|
legendAsTooltipPlugin()
|
|
|
|
],
|
2023-07-26 13:44:06 +02:00
|
|
|
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: [
|
|
|
|
{
|
2023-08-09 12:42:25 +02:00
|
|
|
label: xunit !== '' ? xunit : null,
|
2023-07-26 13:44:06 +02:00
|
|
|
},
|
|
|
|
Object.assign({
|
2023-08-09 12:42:25 +02:00
|
|
|
label: yunit !== '' ? yunit : null,
|
2023-07-26 13:44:06 +02:00
|
|
|
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}
|
2023-08-08 13:27:01 +02:00
|
|
|
<div bind:this={plotWrapper}/>
|
2023-07-26 13:44:06 +02:00
|
|
|
{:else}
|
|
|
|
<Card class="mx-4" body color="warning">Cannot render histogram: No data!</Card>
|
|
|
|
{/if}
|
|
|
|
|
|
|
|
|