2023-08-05 12:14:03 +03:00
|
|
|
import { isDarkThemeEnabled, PRIMARY_DARK_COLOR, PRIMARY_LIGHT_COLOR } from '@shlinkio/shlink-frontend-kit';
|
2023-02-18 12:40:37 +03:00
|
|
|
import type { Chart, ChartData, ChartDataset, ChartOptions } from 'chart.js';
|
2021-09-18 20:05:28 +03:00
|
|
|
import { keys, values } from 'ramda';
|
2023-02-18 13:11:01 +03:00
|
|
|
import type { FC } from 'react';
|
|
|
|
import { memo, useState } from 'react';
|
2021-09-18 20:05:28 +03:00
|
|
|
import { Doughnut } from 'react-chartjs-2';
|
2023-07-29 11:43:15 +03:00
|
|
|
import { renderPieChartLabel } from '../../utils/helpers/charts';
|
2023-02-18 12:40:37 +03:00
|
|
|
import type { Stats } from '../types';
|
2021-09-18 20:05:28 +03:00
|
|
|
import { DoughnutChartLegend } from './DoughnutChartLegend';
|
|
|
|
|
|
|
|
interface DoughnutChartProps {
|
|
|
|
stats: Stats;
|
|
|
|
}
|
|
|
|
|
|
|
|
const generateChartDatasets = (data: number[]): ChartDataset[] => [
|
|
|
|
{
|
|
|
|
data,
|
|
|
|
backgroundColor: [
|
|
|
|
'#97BBCD',
|
|
|
|
'#F7464A',
|
|
|
|
'#46BFBD',
|
|
|
|
'#FDB45C',
|
|
|
|
'#949FB1',
|
|
|
|
'#57A773',
|
|
|
|
'#414066',
|
|
|
|
'#08B2E3',
|
|
|
|
'#B6C454',
|
|
|
|
'#DCDCDC',
|
|
|
|
'#463730',
|
|
|
|
],
|
|
|
|
borderColor: isDarkThemeEnabled() ? PRIMARY_DARK_COLOR : PRIMARY_LIGHT_COLOR,
|
|
|
|
borderWidth: 2,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
const generateChartData = (labels: string[], data: number[]): ChartData => ({
|
|
|
|
labels,
|
|
|
|
datasets: generateChartDatasets(data),
|
|
|
|
});
|
|
|
|
|
|
|
|
export const DoughnutChart: FC<DoughnutChartProps> = memo(({ stats }) => {
|
2023-08-08 14:01:27 +03:00
|
|
|
// Cannot use useRef here, as we need to re-render as soon as the ref is set
|
|
|
|
const [chartRef, setChartRef] = useState<Chart>();
|
2021-09-18 20:05:28 +03:00
|
|
|
const labels = keys(stats);
|
|
|
|
const data = values(stats);
|
|
|
|
|
|
|
|
const options: ChartOptions = {
|
|
|
|
plugins: {
|
|
|
|
legend: { display: false },
|
|
|
|
tooltip: {
|
|
|
|
intersect: true,
|
|
|
|
callbacks: { label: renderPieChartLabel },
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const chartData = generateChartData(labels, data);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="row">
|
|
|
|
<div className="col-sm-12 col-md-7">
|
|
|
|
<Doughnut
|
|
|
|
height={300}
|
2022-03-07 19:39:03 +03:00
|
|
|
data={chartData as any}
|
|
|
|
options={options as any}
|
2021-09-18 20:05:28 +03:00
|
|
|
ref={(element) => {
|
2023-08-08 14:01:27 +03:00
|
|
|
if (element) {
|
|
|
|
setChartRef(element as Chart);
|
|
|
|
}
|
2021-09-18 20:05:28 +03:00
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div className="col-sm-12 col-md-5">
|
|
|
|
{chartRef && <DoughnutChartLegend chart={chartRef} />}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
});
|