BarChart: Add support for data links (#44932)

This commit is contained in:
Nathan Marrs 2022-02-04 18:31:00 -08:00 committed by GitHub
parent b07345e57e
commit 3a2e3267ba
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 185 additions and 28 deletions

View File

@ -1,9 +1,12 @@
import React, { useMemo, useRef } from 'react';
import { TooltipDisplayMode, StackingMode, LegendDisplayMode } from '@grafana/schema';
import React, { useMemo, useRef, useState } from 'react';
import { css } from '@emotion/css';
import { LegendDisplayMode } from '@grafana/schema';
import {
CartesianCoords2D,
compareDataFrameStructures,
DataFrame,
getFieldDisplayName,
GrafanaTheme2,
PanelProps,
TimeRange,
VizOrientation,
@ -13,20 +16,27 @@ import {
GraphNGProps,
measureText,
PlotLegend,
TooltipPlugin,
Portal,
UPlotConfigBuilder,
UPLOT_AXIS_FONT_SIZE,
usePanelContext,
useStyles2,
useTheme2,
VizLayout,
VizLegend,
VizTooltipContainer,
} from '@grafana/ui';
import { PanelDataErrorView } from '@grafana/runtime';
import { PropDiffFn } from '@grafana/ui/src/components/GraphNG/GraphNG';
import { PanelOptions } from './models.gen';
import { prepareBarChartDisplayValues, preparePlotConfigBuilder } from './utils';
import { PanelDataErrorView } from '@grafana/runtime';
import { DataHoverView } from '../geomap/components/DataHoverView';
import { getFieldLegendItem } from '../state-timeline/utils';
import { PropDiffFn } from '@grafana/ui/src/components/GraphNG/GraphNG';
import { CloseButton } from 'app/core/components/CloseButton/CloseButton';
import { HoverEvent, setupConfig } from './config';
const TOOLTIP_OFFSET = 10;
/**
* @alpha
@ -55,8 +65,31 @@ interface Props extends PanelProps<PanelOptions> {}
export const BarChartPanel: React.FunctionComponent<Props> = ({ data, options, width, height, timeZone, id }) => {
const theme = useTheme2();
const styles = useStyles2(getStyles);
const { eventBus } = usePanelContext();
const oldConfig = useRef<UPlotConfigBuilder | undefined>(undefined);
const isToolTipOpen = useRef<boolean>(false);
const [hover, setHover] = useState<HoverEvent | undefined>(undefined);
const [coords, setCoords] = useState<CartesianCoords2D | null>(null);
const [focusedSeriesIdx, setFocusedSeriesIdx] = useState<number | null>(null);
const [focusedPointIdx, setFocusedPointIdx] = useState<number | null>(null);
const [shouldDisplayCloseButton, setShouldDisplayCloseButton] = useState<boolean>(false);
const onCloseToolTip = () => {
isToolTipOpen.current = false;
setCoords(null);
setShouldDisplayCloseButton(false);
};
const onUPlotClick = () => {
isToolTipOpen.current = !isToolTipOpen.current;
// Linking into useState required to re-render tooltip
setShouldDisplayCloseButton(isToolTipOpen.current);
};
const frame0Ref = useRef<DataFrame>();
const info = useMemo(() => prepareBarChartDisplayValues(data?.series, theme, options), [data, theme, options]);
const structureRef = useRef(10000);
@ -86,7 +119,7 @@ export const BarChartPanel: React.FunctionComponent<Props> = ({ data, options, w
// If no max length is set, limit the number of characters to a length where it will use a maximum of half of the height of the viz.
if (!options.xTickLabelMaxLength) {
const rotationAngle = options.xTickLabelRotation;
const textSize = measureText('M', UPLOT_AXIS_FONT_SIZE).width; // M is usually the widest character so let's use that as an aproximation.
const textSize = measureText('M', UPLOT_AXIS_FONT_SIZE).width; // M is usually the widest character so let's use that as an approximation.
const maxHeightForValues = height / 2;
return (
@ -99,14 +132,6 @@ export const BarChartPanel: React.FunctionComponent<Props> = ({ data, options, w
}
}, [height, options.xTickLabelRotation, options.xTickLabelMaxLength]);
// Force 'multi' tooltip setting or stacking mode
const tooltip = useMemo(() => {
if (options.stacking === StackingMode.Normal || options.stacking === StackingMode.Percent) {
return { ...options.tooltip, mode: TooltipDisplayMode.Multi };
}
return options.tooltip;
}, [options.tooltip, options.stacking]);
if (!info.viz?.fields.length) {
return <PanelDataErrorView panelId={id} data={data} message={info.warn} needsNumberField={true} />;
}
@ -119,12 +144,20 @@ export const BarChartPanel: React.FunctionComponent<Props> = ({ data, options, w
}
return (
<DataHoverView
data={info.aligned}
rowIndex={datapointIdx}
columnIndex={seriesIdx}
sortOrder={options.tooltip.sort}
/>
<>
{shouldDisplayCloseButton && (
<>
<CloseButton onClick={onCloseToolTip} />
<div className={styles.closeButtonSpacer} />
</>
)}
<DataHoverView
data={info.aligned}
rowIndex={datapointIdx}
columnIndex={seriesIdx}
sortOrder={options.tooltip.sort}
/>
</>
);
};
@ -220,16 +253,38 @@ export const BarChartPanel: React.FunctionComponent<Props> = ({ data, options, w
height={height}
>
{(config, alignedFrame) => {
if (oldConfig.current !== config) {
oldConfig.current = setupConfig({
config,
onUPlotClick,
setFocusedSeriesIdx,
setFocusedPointIdx,
setCoords,
setHover,
isToolTipOpen,
});
}
return (
<TooltipPlugin
data={alignedFrame}
config={config}
mode={tooltip.mode}
timeZone={timeZone}
renderTooltip={renderTooltip}
/>
<Portal>
{hover && coords && (
<VizTooltipContainer
position={{ x: coords.x, y: coords.y }}
offset={{ x: TOOLTIP_OFFSET, y: TOOLTIP_OFFSET }}
allowPointerEvents
>
{renderTooltip(info.aligned, focusedSeriesIdx, focusedPointIdx)}
</VizTooltipContainer>
)}
</Portal>
);
}}
</GraphNG>
);
};
const getStyles = (theme: GrafanaTheme2) => ({
closeButtonSpacer: css`
margin-bottom: 15px;
`,
});

View File

@ -0,0 +1,102 @@
import { Dispatch, MutableRefObject, SetStateAction } from 'react';
import { UPlotConfigBuilder } from '@grafana/ui';
import { positionTooltip } from '@grafana/ui/src/components/uPlot/plugins/TooltipPlugin';
import { CartesianCoords2D } from '@grafana/data';
export type HoverEvent = {
xIndex: number;
yIndex: number;
pageX: number;
pageY: number;
};
type SetupConfigParams = {
config: UPlotConfigBuilder;
onUPlotClick: () => void;
setFocusedSeriesIdx: Dispatch<SetStateAction<number | null>>;
setFocusedPointIdx: Dispatch<SetStateAction<number | null>>;
setCoords: Dispatch<SetStateAction<CartesianCoords2D | null>>;
setHover: Dispatch<SetStateAction<HoverEvent | undefined>>;
isToolTipOpen: MutableRefObject<boolean>;
};
// This applies config hooks to setup tooltip listener. Ideally this could happen in the same `prepConfig` function
// however the GraphNG structures do not allow access to the `setHover` callback
export const setupConfig = ({
config,
onUPlotClick,
setFocusedSeriesIdx,
setFocusedPointIdx,
setCoords,
setHover,
isToolTipOpen,
}: SetupConfigParams): UPlotConfigBuilder => {
config.addHook('init', (u) => {
u.root.parentElement?.addEventListener('click', onUPlotClick);
});
let rect: DOMRect;
// rect of .u-over (grid area)
config.addHook('syncRect', (u, r) => {
rect = r;
});
const tooltipInterpolator = config.getTooltipInterpolator();
if (tooltipInterpolator) {
config.addHook('setCursor', (u) => {
tooltipInterpolator(
setFocusedSeriesIdx,
setFocusedPointIdx,
(clear) => {
if (clear && !isToolTipOpen.current) {
setCoords(null);
return;
}
if (!rect) {
return;
}
const { x, y } = positionTooltip(u, rect);
if (x !== undefined && y !== undefined && !isToolTipOpen.current) {
setCoords({ x, y });
}
},
u
);
});
}
config.addHook('setLegend', (u) => {
if (!isToolTipOpen.current) {
setFocusedPointIdx(u.legend.idx!);
}
if (u.cursor.idxs != null) {
for (let i = 0; i < u.cursor.idxs.length; i++) {
const sel = u.cursor.idxs[i];
if (sel != null) {
const hover: HoverEvent = {
xIndex: sel,
yIndex: 0,
pageX: rect.left + u.cursor.left!,
pageY: rect.top + u.cursor.top!,
};
if (!isToolTipOpen.current || !hover) {
setHover(hover);
}
return; // only show the first one
}
}
}
});
config.addHook('setSeries', (_, idx) => {
if (!isToolTipOpen.current) {
setFocusedSeriesIdx(idx);
}
});
return config;
};

View File

@ -22,7 +22,7 @@ export const GeomapTooltip = ({ ttip, onClose, isOpen }: Props) => {
{ttip && ttip.layers && (
<VizTooltipContainer position={{ x: ttip.pageX, y: ttip.pageY }} offset={{ x: 10, y: 10 }} allowPointerEvents>
<section ref={ref} {...overlayProps} {...dialogProps}>
<ComplexDataHoverView {...ttip} isOpen={isOpen} onClose={onClose} />
<ComplexDataHoverView layers={ttip.layers} isOpen={isOpen} onClose={onClose} />
</section>
</VizTooltipContainer>
)}