2023-02-10 13:10:02 +00:00
|
|
|
import React from "react";
|
|
|
|
|
2023-06-19 07:29:57 +00:00
|
|
|
// ui
|
|
|
|
import { LineGraph } from "components/ui";
|
|
|
|
// helpers
|
2023-06-20 11:02:02 +00:00
|
|
|
import { renderShortNumericDateFormat } from "helpers/date-time.helper";
|
2023-02-10 13:10:02 +00:00
|
|
|
//types
|
2023-06-20 11:02:02 +00:00
|
|
|
import { TCompletionChartDistribution } from "types";
|
2023-02-10 13:10:02 +00:00
|
|
|
|
|
|
|
type Props = {
|
2023-06-20 11:02:02 +00:00
|
|
|
distribution: TCompletionChartDistribution;
|
|
|
|
startDate: string | Date;
|
|
|
|
endDate: string | Date;
|
|
|
|
totalIssues: number;
|
2023-02-10 13:10:02 +00:00
|
|
|
};
|
|
|
|
|
2023-06-19 07:29:57 +00:00
|
|
|
const styleById = {
|
|
|
|
ideal: {
|
|
|
|
strokeDasharray: "6, 3",
|
|
|
|
strokeWidth: 1,
|
|
|
|
},
|
|
|
|
default: {
|
|
|
|
strokeWidth: 1,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const DashedLine = ({ series, lineGenerator, xScale, yScale }: any) =>
|
|
|
|
series.map(({ id, data, color }: any) => (
|
|
|
|
<path
|
|
|
|
key={id}
|
|
|
|
d={lineGenerator(
|
|
|
|
data.map((d: any) => ({
|
|
|
|
x: xScale(d.data.x),
|
|
|
|
y: yScale(d.data.y),
|
|
|
|
}))
|
|
|
|
)}
|
|
|
|
fill="none"
|
|
|
|
stroke={color ?? "#ddd"}
|
|
|
|
style={styleById[id as keyof typeof styleById] || styleById.default}
|
|
|
|
/>
|
|
|
|
));
|
|
|
|
|
2023-06-20 11:02:02 +00:00
|
|
|
const ProgressChart: React.FC<Props> = ({ distribution, startDate, endDate, totalIssues }) => {
|
|
|
|
const chartData = Object.keys(distribution).map((key) => ({
|
|
|
|
currentDate: renderShortNumericDateFormat(key),
|
|
|
|
pending: distribution[key],
|
|
|
|
}));
|
2023-06-19 07:29:57 +00:00
|
|
|
|
2023-02-10 13:10:02 +00:00
|
|
|
return (
|
2023-06-19 07:29:57 +00:00
|
|
|
<div className="w-full flex justify-center items-center">
|
|
|
|
<LineGraph
|
|
|
|
animate
|
|
|
|
curve="monotoneX"
|
|
|
|
height="160px"
|
2023-06-20 11:02:02 +00:00
|
|
|
width="100%"
|
2023-06-19 07:29:57 +00:00
|
|
|
enableGridY={false}
|
|
|
|
lineWidth={1}
|
|
|
|
margin={{ top: 30, right: 30, bottom: 30, left: 30 }}
|
|
|
|
data={[
|
|
|
|
{
|
|
|
|
id: "pending",
|
|
|
|
color: "#3F76FF",
|
|
|
|
data: chartData.map((item, index) => ({
|
|
|
|
index,
|
|
|
|
x: item.currentDate,
|
|
|
|
y: item.pending,
|
|
|
|
color: "#3F76FF",
|
|
|
|
})),
|
|
|
|
enableArea: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "ideal",
|
|
|
|
color: "#a9bbd0",
|
|
|
|
fill: "transparent",
|
|
|
|
data: [
|
|
|
|
{
|
|
|
|
x: chartData[0].currentDate,
|
2023-06-20 11:02:02 +00:00
|
|
|
y: totalIssues,
|
2023-06-19 07:29:57 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
x: chartData[chartData.length - 1].currentDate,
|
|
|
|
y: 0,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
]}
|
|
|
|
layers={["grid", "markers", "areas", DashedLine, "slices", "points", "axes", "legends"]}
|
|
|
|
axisBottom={{
|
|
|
|
tickValues: chartData.map((item, index) => (index % 2 === 0 ? item.currentDate : "")),
|
|
|
|
}}
|
|
|
|
enablePoints={false}
|
|
|
|
enableArea
|
|
|
|
colors={(datum) => datum.color ?? "#3F76FF"}
|
2023-06-20 11:02:02 +00:00
|
|
|
customYAxisTickValues={[0, totalIssues]}
|
2023-06-19 07:29:57 +00:00
|
|
|
gridXValues={chartData.map((item, index) => (index % 2 === 0 ? item.currentDate : ""))}
|
|
|
|
theme={{
|
2023-06-20 11:02:02 +00:00
|
|
|
background: "transparent",
|
2023-06-19 07:29:57 +00:00
|
|
|
axis: {
|
|
|
|
domain: {
|
|
|
|
line: {
|
|
|
|
stroke: "rgb(var(--color-border))",
|
|
|
|
strokeWidth: 1,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2023-03-07 08:13:09 +00:00
|
|
|
}}
|
2023-06-19 07:29:57 +00:00
|
|
|
/>
|
2023-02-10 13:10:02 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default ProgressChart;
|