import { useEffect, useMemo, useState, type ReactElement } from "react";
import { useNavigate } from "react-router-dom";
import { Layout } from "@/components/layout/Layout";
import {
DataTable,
Pagination,
FilterDropdown,
type Column,
} from "@/components/shared";
import { workflowService } from "@/services/workflow-service";
import { moduleService } from "@/services/module-service";
import type { WorkflowTask, WorkflowTaskCounts } from "@/types/workflow";
import { cn } from "@/lib/utils";
import { useAppTheme } from "@/hooks/useAppTheme";
import { Inbox, Clock, Calendar, CheckCircle2, RotateCcw } from "lucide-react";
const formatDate = (value?: string | null): string => {
if (!value) return "-";
return new Date(value).toLocaleDateString("en-US", {
month: "short",
day: "numeric",
year: "numeric",
hour: "2-digit",
minute: "2-digit"
});
};
const StatCard = ({ icon: Icon, label, value, color, style }: { icon: any, label: string, value: number, color?: string, style?: React.CSSProperties }) => (
);
const Tasks = (): ReactElement => {
const { primaryColor } = useAppTheme();
const navigate = useNavigate();
const [tasks, setTasks] = useState([]);
const [counts, setCounts] = useState(null);
const [currentPage, setCurrentPage] = useState(1);
const [limit, setLimit] = useState(10);
const [total, setTotal] = useState(0);
const [isLoading, setIsLoading] = useState(true);
const [error, setError] = useState(null);
// Filters
const [statusFilter, setStatusFilter] = useState("pending");
const [moduleFilter, setModuleFilter] = useState(null);
const [modules, setModules] = useState<{ id: string; name: string }[]>([]);
const offset = (currentPage - 1) * limit;
const totalPages = Math.max(1, Math.ceil(total / limit));
useEffect(() => {
const fetchModules = async () => {
try {
const res = await moduleService.getMyModules();
if (res.success) {
setModules(res.data.map(m => ({ id: m.id, name: m.name })));
}
} catch (err) {
console.error("Failed to load modules", err);
}
};
fetchModules();
}, []);
useEffect(() => {
const loadData = async () => {
try {
setIsLoading(true);
const [tasksRes, countsRes] = await Promise.all([
workflowService.listTasks({
limit,
offset,
status: statusFilter,
module_id: moduleFilter
}),
workflowService.getTaskCounts({ module_id: moduleFilter })
]);
if (tasksRes.success) {
setTasks(tasksRes.data);
setTotal(tasksRes.pagination.total);
}
if (countsRes.success) {
setCounts(countsRes.data);
}
} catch (err: any) {
setError(err?.response?.data?.error?.message || "Failed to load tasks");
} finally {
setIsLoading(false);
}
};
loadData();
}, [limit, offset, statusFilter, moduleFilter]);
const columns: Column[] = useMemo(
() => [
{
key: "entity_name",
label: "Entity",
render: (task) => (
{task.entity.name}
{task.entity.type}
),
},
{
key: "workflow_name",
label: "Workflow",
render: (task) => (
{task.workflow.name}
),
},
{
key: "step_name",
label: "Step",
render: (task) => (
{task.step.name}
),
},
{
key: "assigned_role",
label: "Assigned To",
render: (task) => (
{task.assignment.assigned_role}
),
},
{
key: "status",
label: "Status",
render: (task) => {
const isOverdueActive = task.is_overdue && !["completed", "rejected", "cancelled"].includes(task.status.toLowerCase());
return (
{isOverdueActive ? "Overdue" : task.status.replace(/_/g, " ").replace(/\b\w/g, l => l.toUpperCase())}
);
},
},
{
key: "due_at",
label: "Due Date",
render: (task) => {
const isOverdueActive = task.is_overdue && !["completed", "rejected", "cancelled"].includes(task.status.toLowerCase());
return (
{formatDate(task.due_at)}
);
},
},
{
key: "actions",
label: "",
render: (task) => (
),
},
],
[navigate],
);
return (
{/* Count Stats Area */}
{/* Task Table Area */}
{statusFilter
? `${statusFilter.replace(/_/g, " ").replace(/\b\w/g, l => l.toUpperCase())} Tasks`
: "All Tasks"
}
({ value: m.id, label: m.name }))}
value={moduleFilter}
onChange={(val) => {
setModuleFilter(val as string | null);
setCurrentPage(1);
}}
placeholder="All Modules"
/>
{
setStatusFilter(val as string | null);
setCurrentPage(1);
}}
placeholder="All Status"
/>
{(statusFilter !== "pending" || moduleFilter) && (
)}
task.id}
emptyMessage="No tasks currently assigned to you"
isLoading={isLoading}
error={error}
/>
{total > 0 && (
{
setLimit(value);
setCurrentPage(1);
}}
/>
)}
);
};
export default Tasks;