mirror of
https://github.com/prometheus/prometheus.git
synced 2025-03-05 20:59:13 -08:00
Implement stacked graphs (dirty) and improve React wrapping
Some checks failed
CI / Go tests (push) Has been cancelled
CI / More Go tests (push) Has been cancelled
CI / Go tests with previous Go version (push) Has been cancelled
CI / UI tests (push) Has been cancelled
CI / Go tests on Windows (push) Has been cancelled
CI / Mixins tests (push) Has been cancelled
CI / Build Prometheus for common architectures (0) (push) Has been cancelled
CI / Build Prometheus for common architectures (1) (push) Has been cancelled
CI / Build Prometheus for common architectures (2) (push) Has been cancelled
CI / Build Prometheus for all architectures (0) (push) Has been cancelled
CI / Build Prometheus for all architectures (1) (push) Has been cancelled
CI / Build Prometheus for all architectures (10) (push) Has been cancelled
CI / Build Prometheus for all architectures (11) (push) Has been cancelled
CI / Build Prometheus for all architectures (2) (push) Has been cancelled
CI / Build Prometheus for all architectures (3) (push) Has been cancelled
CI / Build Prometheus for all architectures (4) (push) Has been cancelled
CI / Build Prometheus for all architectures (5) (push) Has been cancelled
CI / Build Prometheus for all architectures (6) (push) Has been cancelled
CI / Build Prometheus for all architectures (7) (push) Has been cancelled
CI / Build Prometheus for all architectures (8) (push) Has been cancelled
CI / Build Prometheus for all architectures (9) (push) Has been cancelled
CI / Check generated parser (push) Has been cancelled
CI / golangci-lint (push) Has been cancelled
CI / fuzzing (push) Has been cancelled
CI / codeql (push) Has been cancelled
CI / Report status of build Prometheus for all architectures (push) Has been cancelled
CI / Publish main branch artifacts (push) Has been cancelled
CI / Publish release artefacts (push) Has been cancelled
CI / Publish UI on npm Registry (push) Has been cancelled
Some checks failed
CI / Go tests (push) Has been cancelled
CI / More Go tests (push) Has been cancelled
CI / Go tests with previous Go version (push) Has been cancelled
CI / UI tests (push) Has been cancelled
CI / Go tests on Windows (push) Has been cancelled
CI / Mixins tests (push) Has been cancelled
CI / Build Prometheus for common architectures (0) (push) Has been cancelled
CI / Build Prometheus for common architectures (1) (push) Has been cancelled
CI / Build Prometheus for common architectures (2) (push) Has been cancelled
CI / Build Prometheus for all architectures (0) (push) Has been cancelled
CI / Build Prometheus for all architectures (1) (push) Has been cancelled
CI / Build Prometheus for all architectures (10) (push) Has been cancelled
CI / Build Prometheus for all architectures (11) (push) Has been cancelled
CI / Build Prometheus for all architectures (2) (push) Has been cancelled
CI / Build Prometheus for all architectures (3) (push) Has been cancelled
CI / Build Prometheus for all architectures (4) (push) Has been cancelled
CI / Build Prometheus for all architectures (5) (push) Has been cancelled
CI / Build Prometheus for all architectures (6) (push) Has been cancelled
CI / Build Prometheus for all architectures (7) (push) Has been cancelled
CI / Build Prometheus for all architectures (8) (push) Has been cancelled
CI / Build Prometheus for all architectures (9) (push) Has been cancelled
CI / Check generated parser (push) Has been cancelled
CI / golangci-lint (push) Has been cancelled
CI / fuzzing (push) Has been cancelled
CI / codeql (push) Has been cancelled
CI / Report status of build Prometheus for all architectures (push) Has been cancelled
CI / Publish main branch artifacts (push) Has been cancelled
CI / Publish release artefacts (push) Has been cancelled
CI / Publish UI on npm Registry (push) Has been cancelled
Signed-off-by: Julius Volz <julius.volz@gmail.com>
This commit is contained in:
parent
648751568d
commit
2c972dba26
|
@ -2,7 +2,7 @@ import { FC, useEffect, useId, useState } from "react";
|
||||||
import { Alert, Skeleton, Box, LoadingOverlay } from "@mantine/core";
|
import { Alert, Skeleton, Box, LoadingOverlay } from "@mantine/core";
|
||||||
import { IconAlertTriangle, IconInfoCircle } from "@tabler/icons-react";
|
import { IconAlertTriangle, IconInfoCircle } from "@tabler/icons-react";
|
||||||
import { InstantQueryResult } from "../../api/responseTypes/query";
|
import { InstantQueryResult } from "../../api/responseTypes/query";
|
||||||
import { useAPIQuery } from "../../api/api";
|
import { SuccessAPIResponse, useAPIQuery } from "../../api/api";
|
||||||
import classes from "./Graph.module.css";
|
import classes from "./Graph.module.css";
|
||||||
import {
|
import {
|
||||||
GraphDisplayMode,
|
GraphDisplayMode,
|
||||||
|
@ -55,37 +55,45 @@ const Graph: FC<GraphProps> = ({
|
||||||
enabled: expr !== "",
|
enabled: expr !== "",
|
||||||
});
|
});
|
||||||
|
|
||||||
// Keep the displayed chart range separate from the actual query range, so that
|
// Bundle the chart data and the displayed range together. This has two purposes:
|
||||||
// the chart will keep displaying the old range while a query for a new range
|
// 1. If we update them separately, we cause unnecessary rerenders of the uPlot chart itself.
|
||||||
// is still in progress.
|
// 2. We want to keep displaying the old range in the chart while a query for a new range
|
||||||
const [displayedChartRange, setDisplayedChartRange] =
|
// is still in progress.
|
||||||
useState<UPlotChartRange>({
|
const [dataAndRange, setDataAndRange] = useState<{
|
||||||
startTime: startTime,
|
data: SuccessAPIResponse<InstantQueryResult>;
|
||||||
endTime: effectiveEndTime,
|
range: UPlotChartRange;
|
||||||
resolution: effectiveResolution,
|
} | null>(null);
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setDisplayedChartRange({
|
if (data !== undefined) {
|
||||||
startTime: startTime,
|
setDataAndRange({
|
||||||
endTime: effectiveEndTime,
|
data: data,
|
||||||
resolution: effectiveResolution,
|
range: {
|
||||||
});
|
startTime: startTime,
|
||||||
// We actually want to update the displayed range only once the new data is there.
|
endTime: effectiveEndTime,
|
||||||
|
resolution: effectiveResolution,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// We actually want to update the displayed range only once the new data is there,
|
||||||
|
// so we don't want to include any of the range-related parameters in the dependencies.
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
}, [data]);
|
}, [data]);
|
||||||
|
|
||||||
|
// Re-execute the query when the user presses Enter (or hits the Execute button).
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
expr !== "" && refetch();
|
expr !== "" && refetch();
|
||||||
}, [retriggerIdx, refetch, expr, endTime, range, resolution]);
|
}, [retriggerIdx, refetch, expr, endTime, range, resolution]);
|
||||||
|
|
||||||
|
// The useElementSize hook above only gets a valid size on the second render, so this
|
||||||
|
// is a workaround to make the component render twice after mount.
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (data !== undefined && rerender) {
|
if (dataAndRange !== null && rerender) {
|
||||||
setRerender(false);
|
setRerender(false);
|
||||||
}
|
}
|
||||||
}, [data, rerender, setRerender]);
|
}, [dataAndRange, rerender, setRerender]);
|
||||||
|
|
||||||
// TODO: Share all the loading/error/empty data notices with the DataTable.
|
// TODO: Share all the loading/error/empty data notices with the DataTable?
|
||||||
|
|
||||||
// Show a skeleton only on the first load, not on subsequent ones.
|
// Show a skeleton only on the first load, not on subsequent ones.
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
|
@ -110,11 +118,11 @@ const Graph: FC<GraphProps> = ({
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data === undefined) {
|
if (dataAndRange === null) {
|
||||||
return <Alert variant="transparent">No data queried yet</Alert>;
|
return <Alert variant="transparent">No data queried yet</Alert>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { result, resultType } = data.data;
|
const { result, resultType } = dataAndRange.data.data;
|
||||||
|
|
||||||
if (resultType !== "matrix") {
|
if (resultType !== "matrix") {
|
||||||
return (
|
return (
|
||||||
|
@ -150,8 +158,8 @@ const Graph: FC<GraphProps> = ({
|
||||||
// styles={{ loader: { width: "100%", height: "100%" } }}
|
// styles={{ loader: { width: "100%", height: "100%" } }}
|
||||||
/>
|
/>
|
||||||
<UPlotChart
|
<UPlotChart
|
||||||
data={data.data.result}
|
data={dataAndRange.data.data.result}
|
||||||
range={displayedChartRange}
|
range={dataAndRange.range}
|
||||||
width={width}
|
width={width}
|
||||||
showExemplars={showExemplars}
|
showExemplars={showExemplars}
|
||||||
displayMode={displayMode}
|
displayMode={displayMode}
|
||||||
|
|
|
@ -14,7 +14,7 @@ import {
|
||||||
IconGraph,
|
IconGraph,
|
||||||
IconTable,
|
IconTable,
|
||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { FC, useState } from "react";
|
import { FC, useCallback, useState } from "react";
|
||||||
import { useAppDispatch, useAppSelector } from "../../state/hooks";
|
import { useAppDispatch, useAppSelector } from "../../state/hooks";
|
||||||
import {
|
import {
|
||||||
GraphDisplayMode,
|
GraphDisplayMode,
|
||||||
|
@ -47,6 +47,24 @@ const QueryPanel: FC<PanelProps> = ({ idx, metricNames }) => {
|
||||||
const panel = useAppSelector((state) => state.queryPage.panels[idx]);
|
const panel = useAppSelector((state) => state.queryPage.panels[idx]);
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
|
|
||||||
|
const onSelectRange = useCallback(
|
||||||
|
(start: number, end: number) =>
|
||||||
|
dispatch(
|
||||||
|
setVisualizer({
|
||||||
|
idx,
|
||||||
|
visualizer: {
|
||||||
|
...panel.visualizer,
|
||||||
|
range: (end - start) * 1000,
|
||||||
|
endTime: end * 1000,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
),
|
||||||
|
// TODO: How to have panel.visualizer in the dependencies, but not re-create
|
||||||
|
// the callback every time it changes by the callback's own update? This leads
|
||||||
|
// to extra renders of the plot further down.
|
||||||
|
[dispatch, idx, panel.visualizer]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack gap="lg">
|
<Stack gap="lg">
|
||||||
<ExpressionInput
|
<ExpressionInput
|
||||||
|
@ -220,18 +238,7 @@ const QueryPanel: FC<PanelProps> = ({ idx, metricNames }) => {
|
||||||
showExemplars={panel.visualizer.showExemplars}
|
showExemplars={panel.visualizer.showExemplars}
|
||||||
displayMode={panel.visualizer.displayMode}
|
displayMode={panel.visualizer.displayMode}
|
||||||
retriggerIdx={retriggerIdx}
|
retriggerIdx={retriggerIdx}
|
||||||
onSelectRange={(start: number, end: number) =>
|
onSelectRange={onSelectRange}
|
||||||
dispatch(
|
|
||||||
setVisualizer({
|
|
||||||
idx,
|
|
||||||
visualizer: {
|
|
||||||
...panel.visualizer,
|
|
||||||
range: (end - start) * 1000,
|
|
||||||
endTime: end * 1000,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</Tabs.Panel>
|
</Tabs.Panel>
|
||||||
</Tabs>
|
</Tabs>
|
||||||
|
|
|
@ -10,6 +10,7 @@ import { useComputedColorScheme } from "@mantine/core";
|
||||||
import "uplot/dist/uPlot.min.css";
|
import "uplot/dist/uPlot.min.css";
|
||||||
import "./uplot.css";
|
import "./uplot.css";
|
||||||
import { getUPlotData, getUPlotOptions } from "./uPlotChartHelpers";
|
import { getUPlotData, getUPlotOptions } from "./uPlotChartHelpers";
|
||||||
|
import { setStackedOpts } from "./uPlotStackHelpers";
|
||||||
|
|
||||||
export interface UPlotChartRange {
|
export interface UPlotChartRange {
|
||||||
startTime: number;
|
startTime: number;
|
||||||
|
@ -26,13 +27,19 @@ export interface UPlotChartProps {
|
||||||
onSelectRange: (start: number, end: number) => void;
|
onSelectRange: (start: number, end: number) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This wrapper component translates the incoming Prometheus RangeSamples[] data to the
|
||||||
|
// uPlot format and sets up the uPlot options object depending on the UI settings.
|
||||||
const UPlotChart: FC<UPlotChartProps> = ({
|
const UPlotChart: FC<UPlotChartProps> = ({
|
||||||
data,
|
data,
|
||||||
range: { startTime, endTime, resolution },
|
range: { startTime, endTime, resolution },
|
||||||
width,
|
width,
|
||||||
|
displayMode,
|
||||||
onSelectRange,
|
onSelectRange,
|
||||||
}) => {
|
}) => {
|
||||||
const [options, setOptions] = useState<uPlot.Options | null>(null);
|
const [options, setOptions] = useState<uPlot.Options | null>(null);
|
||||||
|
const [processedData, setProcessedData] = useState<uPlot.AlignedData | null>(
|
||||||
|
null
|
||||||
|
);
|
||||||
const { useLocalTime } = useSettings();
|
const { useLocalTime } = useSettings();
|
||||||
const theme = useComputedColorScheme();
|
const theme = useComputedColorScheme();
|
||||||
|
|
||||||
|
@ -41,32 +48,49 @@ const UPlotChart: FC<UPlotChartProps> = ({
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setOptions(
|
const seriesData: uPlot.AlignedData = getUPlotData(
|
||||||
getUPlotOptions(
|
data,
|
||||||
width,
|
startTime,
|
||||||
data,
|
endTime,
|
||||||
useLocalTime,
|
resolution
|
||||||
theme === "light",
|
|
||||||
onSelectRange
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
}, [width, data, useLocalTime, theme, onSelectRange]);
|
|
||||||
|
|
||||||
const seriesData: uPlot.AlignedData = getUPlotData(
|
const opts = getUPlotOptions(
|
||||||
|
seriesData,
|
||||||
|
width,
|
||||||
|
data,
|
||||||
|
useLocalTime,
|
||||||
|
theme === "light",
|
||||||
|
onSelectRange
|
||||||
|
);
|
||||||
|
|
||||||
|
if (displayMode === GraphDisplayMode.Stacked) {
|
||||||
|
setProcessedData(setStackedOpts(opts, seriesData).data);
|
||||||
|
} else {
|
||||||
|
setProcessedData(seriesData);
|
||||||
|
}
|
||||||
|
|
||||||
|
setOptions(opts);
|
||||||
|
}, [
|
||||||
|
width,
|
||||||
data,
|
data,
|
||||||
|
displayMode,
|
||||||
startTime,
|
startTime,
|
||||||
endTime,
|
endTime,
|
||||||
resolution
|
resolution,
|
||||||
);
|
useLocalTime,
|
||||||
|
theme,
|
||||||
|
onSelectRange,
|
||||||
|
]);
|
||||||
|
|
||||||
if (options === null) {
|
if (options === null || processedData === null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<UplotReact
|
<UplotReact
|
||||||
options={options}
|
options={options}
|
||||||
data={seriesData}
|
data={processedData}
|
||||||
className={classes.uplotChart}
|
className={classes.uplotChart}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { formatSeries } from "../../lib/formatSeries";
|
||||||
import { formatTimestamp } from "../../lib/formatTime";
|
import { formatTimestamp } from "../../lib/formatTime";
|
||||||
import { getSeriesColor } from "./colorPool";
|
import { getSeriesColor } from "./colorPool";
|
||||||
import { computePosition, shift, flip, offset } from "@floating-ui/dom";
|
import { computePosition, shift, flip, offset } from "@floating-ui/dom";
|
||||||
import uPlot, { Series } from "uplot";
|
import uPlot, { AlignedData, Series } from "uplot";
|
||||||
|
|
||||||
const formatYAxisTickValue = (y: number | null): string => {
|
const formatYAxisTickValue = (y: number | null): string => {
|
||||||
if (y === null) {
|
if (y === null) {
|
||||||
|
@ -81,7 +81,7 @@ const formatLabels = (labels: { [key: string]: string }): string => `
|
||||||
.join("")}
|
.join("")}
|
||||||
</div>`;
|
</div>`;
|
||||||
|
|
||||||
const tooltipPlugin = (useLocalTime: boolean) => {
|
const tooltipPlugin = (useLocalTime: boolean, data: AlignedData) => {
|
||||||
let over: HTMLDivElement;
|
let over: HTMLDivElement;
|
||||||
let boundingLeft: number;
|
let boundingLeft: number;
|
||||||
let boundingTop: number;
|
let boundingTop: number;
|
||||||
|
@ -141,7 +141,7 @@ const tooltipPlugin = (useLocalTime: boolean) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
const ts = u.data[0][idx];
|
const ts = u.data[0][idx];
|
||||||
const value = u.data[selectedSeriesIdx][idx];
|
const value = data[selectedSeriesIdx][idx];
|
||||||
const series = u.series[selectedSeriesIdx];
|
const series = u.series[selectedSeriesIdx];
|
||||||
// @ts-expect-error - uPlot doesn't have a field for labels, but we just attach some anyway.
|
// @ts-expect-error - uPlot doesn't have a field for labels, but we just attach some anyway.
|
||||||
const labels = series.labels;
|
const labels = series.labels;
|
||||||
|
@ -286,6 +286,7 @@ const onlyDrawPointsForDisconnectedSamplesFilter = (
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getUPlotOptions = (
|
export const getUPlotOptions = (
|
||||||
|
data: AlignedData,
|
||||||
width: number,
|
width: number,
|
||||||
result: RangeSamples[],
|
result: RangeSamples[],
|
||||||
useLocalTime: boolean,
|
useLocalTime: boolean,
|
||||||
|
@ -309,7 +310,7 @@ export const getUPlotOptions = (
|
||||||
tzDate: useLocalTime
|
tzDate: useLocalTime
|
||||||
? undefined
|
? undefined
|
||||||
: (ts) => uPlot.tzDate(new Date(ts * 1e3), "Etc/UTC"),
|
: (ts) => uPlot.tzDate(new Date(ts * 1e3), "Etc/UTC"),
|
||||||
plugins: [tooltipPlugin(useLocalTime)],
|
plugins: [tooltipPlugin(useLocalTime, data)],
|
||||||
legend: {
|
legend: {
|
||||||
show: true,
|
show: true,
|
||||||
live: false,
|
live: false,
|
||||||
|
@ -408,15 +409,15 @@ export const getUPlotData = (
|
||||||
}
|
}
|
||||||
|
|
||||||
const values = inputData.map(({ values, histograms }) => {
|
const values = inputData.map(({ values, histograms }) => {
|
||||||
// Insert nulls for all missing steps.
|
|
||||||
const data: (number | null)[] = [];
|
const data: (number | null)[] = [];
|
||||||
let valuePos = 0;
|
let valuePos = 0;
|
||||||
let histogramPos = 0;
|
let histogramPos = 0;
|
||||||
|
|
||||||
for (let t = startTime; t <= endTime; t += resolution) {
|
for (let t = startTime; t <= endTime; t += resolution) {
|
||||||
// Allow for floating point inaccuracy.
|
|
||||||
const currentValue = values && values[valuePos];
|
const currentValue = values && values[valuePos];
|
||||||
const currentHistogram = histograms && histograms[histogramPos];
|
const currentHistogram = histograms && histograms[histogramPos];
|
||||||
|
|
||||||
|
// Allow for floating point inaccuracy.
|
||||||
if (
|
if (
|
||||||
currentValue &&
|
currentValue &&
|
||||||
values.length > valuePos &&
|
values.length > valuePos &&
|
||||||
|
@ -432,6 +433,7 @@ export const getUPlotData = (
|
||||||
data.push(parseValue(currentHistogram[1].sum));
|
data.push(parseValue(currentHistogram[1].sum));
|
||||||
histogramPos++;
|
histogramPos++;
|
||||||
} else {
|
} else {
|
||||||
|
// Insert nulls for all missing steps.
|
||||||
data.push(null);
|
data.push(null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
96
web/ui/mantine-ui/src/pages/query/uPlotStackHelpers.ts
Normal file
96
web/ui/mantine-ui/src/pages/query/uPlotStackHelpers.ts
Normal file
|
@ -0,0 +1,96 @@
|
||||||
|
import { lighten } from "@mantine/core";
|
||||||
|
import uPlot, { AlignedData, TypedArray } from "uplot";
|
||||||
|
|
||||||
|
// Stacking code adapted from https://leeoniya.github.io/uPlot/demos/stack.js
|
||||||
|
function stack(
|
||||||
|
data: uPlot.AlignedData,
|
||||||
|
omit: (i: number) => boolean
|
||||||
|
): { data: uPlot.AlignedData; bands: uPlot.Band[] } {
|
||||||
|
const data2: uPlot.AlignedData = [];
|
||||||
|
let bands: uPlot.Band[] = [];
|
||||||
|
const d0Len = data[0].length;
|
||||||
|
const accum = Array(d0Len);
|
||||||
|
|
||||||
|
for (let i = 0; i < d0Len; i++) {
|
||||||
|
accum[i] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 1; i < data.length; i++) {
|
||||||
|
data2.push(
|
||||||
|
(omit(i)
|
||||||
|
? data[i]
|
||||||
|
: data[i].map((v, i) => (accum[i] += +(v || 0)))) as TypedArray
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 1; i < data.length; i++) {
|
||||||
|
!omit(i) &&
|
||||||
|
bands.push({
|
||||||
|
series: [data.findIndex((_s, j) => j > i && !omit(j)), i],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
bands = bands.filter((b) => b.series[1] > -1);
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: [data[0]].concat(data2) as AlignedData,
|
||||||
|
bands,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function setStackedOpts(opts: uPlot.Options, data: uPlot.AlignedData) {
|
||||||
|
const stacked = stack(data, (_i) => false);
|
||||||
|
opts.bands = stacked.bands;
|
||||||
|
|
||||||
|
opts.cursor = opts.cursor || {};
|
||||||
|
opts.cursor.dataIdx = (_u, seriesIdx, closestIdx, _xValue) =>
|
||||||
|
data[seriesIdx][closestIdx] == null ? null : closestIdx;
|
||||||
|
|
||||||
|
opts.series.forEach((s) => {
|
||||||
|
// s.value = (u, v, si, i) => data[si][i];
|
||||||
|
|
||||||
|
s.points = s.points || {};
|
||||||
|
|
||||||
|
if (s.stroke) {
|
||||||
|
s.fill = lighten(s.stroke as string, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
// scan raw unstacked data to return only real points
|
||||||
|
s.points.filter = (
|
||||||
|
_self: uPlot,
|
||||||
|
seriesIdx: number,
|
||||||
|
show: boolean,
|
||||||
|
_gaps?: null | number[][]
|
||||||
|
): number[] | null => {
|
||||||
|
if (show) {
|
||||||
|
const pts: number[] = [];
|
||||||
|
data[seriesIdx].forEach((v, i) => {
|
||||||
|
v != null && pts.push(i);
|
||||||
|
});
|
||||||
|
return pts;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
// force 0 to be the sum minimum this instead of the bottom series
|
||||||
|
opts.scales = opts.scales || {};
|
||||||
|
opts.scales.y = {
|
||||||
|
range: (_u, _min, max) => {
|
||||||
|
const minMax = uPlot.rangeNum(0, max, 0.1, true);
|
||||||
|
return [0, minMax[1]];
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// restack on toggle
|
||||||
|
opts.hooks = opts.hooks || {};
|
||||||
|
opts.hooks.setSeries = opts.hooks.setSeries || [];
|
||||||
|
opts.hooks.setSeries.push((u, _i) => {
|
||||||
|
const stacked = stack(data, (i) => !u.series[i].show);
|
||||||
|
u.delBand(null);
|
||||||
|
stacked.bands.forEach((b) => u.addBand(b));
|
||||||
|
u.setData(stacked.data);
|
||||||
|
});
|
||||||
|
|
||||||
|
return { opts, data: stacked.data };
|
||||||
|
}
|
Loading…
Reference in a new issue