mirror of
https://github.com/prometheus/prometheus.git
synced 2025-03-05 20:59:13 -08:00
* React UI: Support local timezone on /graph This partially implements https://github.com/prometheus/prometheus/issues/500 in the sense that it only addresses the /graph page, and only allows toggling between UTC and local (browser) time, but no arbitrary timezone selection yet. Signed-off-by: Julius Volz <julius.volz@gmail.com> * Fixup: Also display TZ offset in tooltip Signed-off-by: Julius Volz <julius.volz@gmail.com> * Just show offset, not timezone name abbreviation Signed-off-by: Julius Volz <julius.volz@gmail.com>
146 lines
4 KiB
TypeScript
146 lines
4 KiB
TypeScript
import $ from 'jquery';
|
|
import React, { PureComponent } from 'react';
|
|
import ReactResizeDetector from 'react-resize-detector';
|
|
|
|
import { Legend } from './Legend';
|
|
import { Metric, QueryParams } from '../../types/types';
|
|
import { isPresent } from '../../utils';
|
|
import { normalizeData, getOptions, toHoverColor } from './GraphHelpers';
|
|
|
|
require('../../vendor/flot/jquery.flot');
|
|
require('../../vendor/flot/jquery.flot.stack');
|
|
require('../../vendor/flot/jquery.flot.time');
|
|
require('../../vendor/flot/jquery.flot.crosshair');
|
|
require('jquery.flot.tooltip');
|
|
|
|
export interface GraphProps {
|
|
data: {
|
|
resultType: string;
|
|
result: Array<{ metric: Metric; values: [number, string][] }>;
|
|
};
|
|
stacked: boolean;
|
|
useLocalTime: boolean;
|
|
queryParams: QueryParams | null;
|
|
}
|
|
|
|
export interface GraphSeries {
|
|
labels: { [key: string]: string };
|
|
color: string;
|
|
data: (number | null)[][]; // [x,y][]
|
|
index: number;
|
|
}
|
|
|
|
interface GraphState {
|
|
chartData: GraphSeries[];
|
|
}
|
|
|
|
class Graph extends PureComponent<GraphProps, GraphState> {
|
|
private chartRef = React.createRef<HTMLDivElement>();
|
|
private $chart?: jquery.flot.plot;
|
|
private rafID = 0;
|
|
private selectedSeriesIndexes: number[] = [];
|
|
|
|
state = {
|
|
chartData: normalizeData(this.props),
|
|
};
|
|
|
|
componentDidUpdate(prevProps: GraphProps) {
|
|
const { data, stacked, useLocalTime } = this.props;
|
|
if (prevProps.data !== data) {
|
|
this.selectedSeriesIndexes = [];
|
|
this.setState({ chartData: normalizeData(this.props) }, this.plot);
|
|
} else if (prevProps.stacked !== stacked) {
|
|
this.setState({ chartData: normalizeData(this.props) }, () => {
|
|
if (this.selectedSeriesIndexes.length === 0) {
|
|
this.plot();
|
|
} else {
|
|
this.plot(this.state.chartData.filter((_, i) => this.selectedSeriesIndexes.includes(i)));
|
|
}
|
|
});
|
|
}
|
|
|
|
if (prevProps.useLocalTime !== useLocalTime) {
|
|
this.plot();
|
|
}
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.plot();
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
this.destroyPlot();
|
|
}
|
|
|
|
plot = (data: GraphSeries[] = this.state.chartData) => {
|
|
if (!this.chartRef.current) {
|
|
return;
|
|
}
|
|
this.destroyPlot();
|
|
|
|
this.$chart = $.plot($(this.chartRef.current), data, getOptions(this.props.stacked, this.props.useLocalTime));
|
|
};
|
|
|
|
destroyPlot = () => {
|
|
if (isPresent(this.$chart)) {
|
|
this.$chart.destroy();
|
|
}
|
|
};
|
|
|
|
plotSetAndDraw(data: GraphSeries[] = this.state.chartData) {
|
|
if (isPresent(this.$chart)) {
|
|
this.$chart.setData(data);
|
|
this.$chart.draw();
|
|
}
|
|
}
|
|
|
|
handleSeriesSelect = (selected: number[], selectedIndex: number) => {
|
|
const { chartData } = this.state;
|
|
this.plot(
|
|
this.selectedSeriesIndexes.length === 1 && this.selectedSeriesIndexes.includes(selectedIndex)
|
|
? chartData.map(toHoverColor(selectedIndex, this.props.stacked))
|
|
: chartData.filter((_, i) => selected.includes(i)) // draw only selected
|
|
);
|
|
this.selectedSeriesIndexes = selected;
|
|
};
|
|
|
|
handleSeriesHover = (index: number) => () => {
|
|
if (this.rafID) {
|
|
cancelAnimationFrame(this.rafID);
|
|
}
|
|
this.rafID = requestAnimationFrame(() => {
|
|
this.plotSetAndDraw(this.state.chartData.map(toHoverColor(index, this.props.stacked)));
|
|
});
|
|
};
|
|
|
|
handleLegendMouseOut = () => {
|
|
cancelAnimationFrame(this.rafID);
|
|
this.plotSetAndDraw();
|
|
};
|
|
|
|
handleResize = () => {
|
|
if (isPresent(this.$chart)) {
|
|
this.plot(this.$chart.getData() as GraphSeries[]);
|
|
}
|
|
};
|
|
|
|
render() {
|
|
const { chartData } = this.state;
|
|
return (
|
|
<div className="graph">
|
|
<ReactResizeDetector handleWidth onResize={this.handleResize} skipOnMount />
|
|
<div className="graph-chart" ref={this.chartRef} />
|
|
<Legend
|
|
shouldReset={this.selectedSeriesIndexes.length === 0}
|
|
chartData={chartData}
|
|
onHover={this.handleSeriesHover}
|
|
onLegendMouseOut={this.handleLegendMouseOut}
|
|
onSeriesToggle={this.handleSeriesSelect}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
export default Graph;
|