import React from "react"; import { eachDayOfInterval, isValid } from "date-fns"; // ui import { LineGraph } from "components/ui"; // helpers import { renderFormattedDateWithoutYear } from "helpers/date-time.helper"; //types import { TCompletionChartDistribution } from "@plane/types"; type Props = { distribution: TCompletionChartDistribution; startDate: string | Date; endDate: string | Date; totalIssues: number; }; const styleById = { ideal: { strokeDasharray: "6, 3", strokeWidth: 1, }, default: { strokeWidth: 1, }, }; const DashedLine = ({ series, lineGenerator, xScale, yScale }: any) => series.map(({ id, data, color }: 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} /> )); const ProgressChart: React.FC = ({ distribution, startDate, endDate, totalIssues }) => { const chartData = Object.keys(distribution ?? []).map((key) => ({ currentDate: renderFormattedDateWithoutYear(key), pending: distribution[key], })); const generateXAxisTickValues = () => { const start = new Date(startDate); const end = new Date(endDate); let dates: Date[] = []; if (isValid(start) && isValid(end)) { dates = eachDayOfInterval({ start, end }); } const maxDates = 4; const totalDates = dates.length; if (totalDates <= maxDates) return dates.map((d) => renderFormattedDateWithoutYear(d)); else { const interval = Math.ceil(totalDates / maxDates); const limitedDates = []; for (let i = 0; i < totalDates; i += interval) limitedDates.push(renderFormattedDateWithoutYear(dates[i])); if (!limitedDates.includes(renderFormattedDateWithoutYear(dates[totalDates - 1]))) limitedDates.push(renderFormattedDateWithoutYear(dates[totalDates - 1])); return limitedDates; } }; return (
0 ? chartData.map((item, index) => ({ index, x: item.currentDate, y: item.pending, color: "#3F76FF", })) : [], enableArea: true, }, { id: "ideal", color: "#a9bbd0", fill: "transparent", data: chartData.length > 0 ? [ { x: chartData[0].currentDate, y: totalIssues, }, { x: chartData[chartData.length - 1].currentDate, y: 0, }, ] : [], }, ]} layers={["grid", "markers", "areas", DashedLine, "slices", "points", "axes", "legends"]} axisBottom={{ tickValues: generateXAxisTickValues(), }} enablePoints={false} enableArea colors={(datum) => datum.color ?? "#3F76FF"} customYAxisTickValues={[0, totalIssues]} gridXValues={ chartData.length > 0 ? chartData.map((item, index) => (index % 2 === 0 ? item.currentDate : "")) : undefined } enableSlices="x" sliceTooltip={(datum) => (
{datum.slice.points[0].data.yFormatted} issues pending on {datum.slice.points[0].data.xFormatted}
)} theme={{ background: "transparent", axis: { domain: { line: { stroke: "var(--color-neutral-70)", strokeWidth: 1, }, }, }, }} />
); }; export default ProgressChart;