import { useEffect, useState } from "react"; import Link from "next/link"; import { useRouter } from "next/router"; import { observer } from "mobx-react-lite"; // hooks import { useDashboard } from "hooks/store"; // components import { PieGraph } from "components/ui"; import { DurationFilterDropdown, IssuesByStateGroupEmptyState, WidgetLoader, WidgetProps, } from "components/dashboard/widgets"; // helpers import { getCustomDates } from "helpers/dashboard.helper"; // types import { TIssuesByStateGroupsWidgetFilters, TIssuesByStateGroupsWidgetResponse, TStateGroups } from "@plane/types"; // constants import { STATE_GROUP_GRAPH_COLORS, STATE_GROUP_GRAPH_GRADIENTS } from "constants/dashboard"; import { STATE_GROUPS } from "constants/state"; const WIDGET_KEY = "issues_by_state_groups"; export const IssuesByStateGroupWidget: React.FC = observer((props) => { const { dashboardId, workspaceSlug } = props; // states const [activeStateGroup, setActiveStateGroup] = useState("started"); // router const router = useRouter(); // store hooks const { fetchWidgetStats, widgetDetails: allWidgetDetails, widgetStats: allWidgetStats, updateDashboardWidgetFilters, } = useDashboard(); // derived values const widgetDetails = allWidgetDetails?.[workspaceSlug]?.[dashboardId]?.find((w) => w.key === WIDGET_KEY); const widgetStats = allWidgetStats?.[workspaceSlug]?.[dashboardId]?.[ WIDGET_KEY ] as TIssuesByStateGroupsWidgetResponse[]; const handleUpdateFilters = async (filters: Partial) => { if (!widgetDetails) return; await updateDashboardWidgetFilters(workspaceSlug, dashboardId, widgetDetails.id, { widgetKey: WIDGET_KEY, filters, }); fetchWidgetStats(workspaceSlug, dashboardId, { widget_key: WIDGET_KEY, target_date: getCustomDates(widgetDetails.widget_filters.target_date ?? "this_week"), }); }; useEffect(() => { if (!widgetDetails) return; if (!widgetStats) fetchWidgetStats(workspaceSlug, dashboardId, { widget_key: WIDGET_KEY, target_date: getCustomDates(widgetDetails.widget_filters.target_date ?? "this_week"), }); }, [dashboardId, fetchWidgetStats, widgetDetails, widgetStats, workspaceSlug]); if (!widgetDetails || !widgetStats) return ; const totalCount = widgetStats?.reduce((acc, item) => acc + item?.count, 0); const chartData = widgetStats?.map((item) => ({ color: STATE_GROUP_GRAPH_COLORS[item?.state as keyof typeof STATE_GROUP_GRAPH_COLORS], id: item?.state, label: item?.state, value: (item?.count / totalCount) * 100, })); const CenteredMetric = ({ dataWithArc, centerX, centerY }: any) => { const data = dataWithArc?.find((datum: any) => datum?.id === activeStateGroup); const percentage = chartData?.find((item) => item.id === activeStateGroup)?.value?.toFixed(0); return ( {percentage}% {data?.id} ); }; return (

State of assigned issues

handleUpdateFilters({ target_date: val, }) } />
{totalCount > 0 ? (
datum.data.color} padAngle={1} enableArcLinkLabels={false} enableArcLabels={false} activeOuterRadiusOffset={5} tooltip={() => <>} margin={{ top: 0, right: 5, bottom: 0, left: 5, }} defs={STATE_GROUP_GRAPH_GRADIENTS} fill={Object.values(STATE_GROUPS).map((p) => ({ match: { id: p.key, }, id: `gradient${p.label}`, }))} onClick={(datum, e) => { e.preventDefault(); e.stopPropagation(); router.push(`/${workspaceSlug}/workspace-views/assigned/?state_group=${datum.id}`); }} onMouseEnter={(datum) => setActiveStateGroup(datum.id as TStateGroups)} layers={["arcs", CenteredMetric]} />
{chartData.map((item) => (
{item.label}
{item.value.toFixed(0)}%
))}
) : (
)} ); });