shlink-web-client/src/visits/helpers/DefaultChart.tsx

178 lines
5.6 KiB
TypeScript
Raw Normal View History

import { useState } from 'react';
import { Doughnut, HorizontalBar } from 'react-chartjs-2';
import { keys, values } from 'ramda';
import classNames from 'classnames';
import Chart, { ChartData, ChartDataSets, ChartOptions } from 'chart.js';
2020-09-20 12:58:40 +03:00
import { fillTheGaps } from '../../utils/helpers/visits';
2020-09-03 21:34:22 +03:00
import { Stats } from '../types';
2020-09-15 23:22:56 +03:00
import { prettify } from '../../utils/helpers/numbers';
import { pointerOnHover, renderDoughnutChartLabel, renderNonDoughnutChartLabel } from '../../utils/helpers/charts';
2020-12-20 14:17:12 +03:00
import { HIGHLIGHTED_COLOR, HIGHLIGHTED_COLOR_ALPHA, MAIN_COLOR, MAIN_COLOR_ALPHA } from '../../utils/theme';
import './DefaultChart.scss';
2020-09-03 21:34:22 +03:00
export interface DefaultChartProps {
title: Function | string;
stats: Stats;
isBarChart?: boolean;
max?: number;
highlightedStats?: Stats;
highlightedLabel?: string;
onClick?: (label: string) => void;
}
const generateGraphData = (
title: Function | string,
isBarChart: boolean,
labels: string[],
data: number[],
highlightedData?: number[],
highlightedLabel?: string,
): ChartData => ({
labels,
datasets: [
{
title,
label: highlightedData ? 'Non-selected' : 'Visits',
data,
2020-12-20 14:17:12 +03:00
backgroundColor: isBarChart ? MAIN_COLOR_ALPHA : [
'#97BBCD',
'#F7464A',
'#46BFBD',
'#FDB45C',
'#949FB1',
'#57A773',
'#414066',
'#08B2E3',
'#B6C454',
'#DCDCDC',
'#463730',
],
2020-12-20 14:17:12 +03:00
borderColor: isBarChart ? MAIN_COLOR : 'white',
borderWidth: 2,
},
highlightedData && {
title,
2020-09-03 21:34:22 +03:00
label: highlightedLabel ?? 'Selected',
data: highlightedData,
2020-12-20 14:17:12 +03:00
backgroundColor: HIGHLIGHTED_COLOR_ALPHA,
borderColor: HIGHLIGHTED_COLOR,
borderWidth: 2,
},
].filter(Boolean) as ChartDataSets[],
});
2020-09-03 21:34:22 +03:00
const dropLabelIfHidden = (label: string) => label.startsWith('hidden') ? '' : label;
2020-09-03 21:34:22 +03:00
const determineHeight = (isBarChart: boolean, labels: string[]): number | undefined => {
if (!isBarChart) {
return 300;
}
2020-09-03 21:34:22 +03:00
return isBarChart && labels.length > 20 ? labels.length * 8 : undefined;
};
2020-09-03 21:34:22 +03:00
const renderPieChartLegend = ({ config }: Chart) => {
const { labels = [], datasets = [] } = config.data ?? {};
const { defaultColor } = config.options ?? {} as any;
const [{ backgroundColor: colors }] = datasets;
return (
<ul className="default-chart__pie-chart-legend">
{labels.map((label, index) => (
2020-09-03 21:34:22 +03:00
<li key={label as string} className="default-chart__pie-chart-legend-item d-flex">
<div
className="default-chart__pie-chart-legend-item-color"
2020-09-03 21:34:22 +03:00
style={{ backgroundColor: (colors as string[])[index] || defaultColor }}
/>
<small className="default-chart__pie-chart-legend-item-text flex-fill">{label}</small>
</li>
))}
</ul>
);
};
2020-09-03 21:34:22 +03:00
const chartElementAtEvent = (onClick?: (label: string) => void) => ([ chart ]: [{ _index: number; _chart: Chart }]) => {
if (!onClick || !chart) {
return;
}
const { _index, _chart: { data } } = chart;
const { labels } = data;
2020-09-03 21:34:22 +03:00
onClick(labels?.[_index] as string);
};
2020-09-03 21:34:22 +03:00
const statsAreDefined = (stats: Stats | undefined): stats is Stats => !!stats && Object.keys(stats).length > 0;
const DefaultChart = (
{ title, isBarChart = false, stats, max, highlightedStats, highlightedLabel, onClick }: DefaultChartProps,
) => {
const Component = isBarChart ? HorizontalBar : Doughnut;
const labels = keys(stats).map(dropLabelIfHidden);
2020-09-03 21:34:22 +03:00
const data = values(
!statsAreDefined(highlightedStats) ? stats : keys(highlightedStats).reduce((acc, highlightedKey) => {
if (acc[highlightedKey]) {
acc[highlightedKey] -= highlightedStats[highlightedKey];
}
return acc;
}, { ...stats }),
);
const highlightedData = statsAreDefined(highlightedStats) ? fillTheGaps(highlightedStats, labels) : undefined;
const [ chartRef, setChartRef ] = useState<HorizontalBar | Doughnut | undefined>()
2020-09-03 21:34:22 +03:00
const options: ChartOptions = {
legend: { display: false },
2020-09-03 21:34:22 +03:00
legendCallback: !isBarChart && renderPieChartLegend as any,
scales: !isBarChart ? undefined : {
xAxes: [
{
2020-09-15 23:22:56 +03:00
ticks: {
beginAtZero: true,
// @ts-expect-error
precision: 0,
callback: prettify,
max,
},
stacked: true,
},
],
yAxes: [{ stacked: true }],
},
tooltips: {
intersect: !isBarChart,
// Do not show tooltip on items with empty label when in a bar chart
filter: ({ yLabel }) => !isBarChart || yLabel !== '',
2020-09-15 23:22:56 +03:00
callbacks: {
label: isBarChart ? renderNonDoughnutChartLabel('xLabel') : renderDoughnutChartLabel,
2020-09-15 23:22:56 +03:00
},
},
onHover: !isBarChart ? undefined : (pointerOnHover) as any, // TODO Types seem to be incorrectly defined in @types/chart.js
};
const graphData = generateGraphData(title, isBarChart, labels, data, highlightedData, highlightedLabel);
const height = determineHeight(isBarChart, labels);
// Provide a key based on the height, so that every time the dataset changes, a new graph is rendered
return (
<div className="row">
<div className={classNames('col-sm-12', { 'col-md-7': !isBarChart })}>
<Component
ref={(element) => setChartRef(element ?? undefined)}
key={height}
data={graphData}
options={options}
height={height}
getElementAtEvent={chartElementAtEvent(onClick)}
/>
</div>
{!isBarChart && (
<div className="col-sm-12 col-md-5">
{chartRef?.chartInstance.generateLegend()}
</div>
)}
</div>
);
};
export default DefaultChart;