import type { Task, User, Status } from './data';
import { PRIORITY_COLORS, STATUS_COLORS, STATUS_LABELS } from './data';
import { Avatar, PriorityBadge, StatusBadge, ProgressBar } from './Shared';
function TaskCard({ task, onClick }: { task: Task; onClick: () => void }) {
const p = PRIORITY_COLORS[task.priority];
const due = new Date(task.dueDate + 'T00:00:00');
const overdue = due < new Date() && task.status !== 'done';
const commCount = task.comments.length;
return (
{task.tags.slice(0, 2).map(t =>
{t})}
{task.subtasks.length > 0 &&
}
๐
{due.toLocaleDateString('en-US', { month: 'short', day: 'numeric' })}
{commCount > 0 && ๐ฌ {commCount}}
);
}
function KanbanColumn({ status, statusLabel, tasks, color, onTaskClick, onAddTask }: {
status: Status; statusLabel: string; tasks: Task[]; color: string;
onTaskClick: (t: Task) => void; onAddTask: (s: Status) => void;
}) {
return (
{statusLabel}
{tasks.length}
{tasks.length === 0 ? (
No tasks here ยท Click + to add one
) : (
tasks.map(t =>
onTaskClick(t)} />)
)}
);
}
interface KanbanProps {
tasks: Task[]; currentUser: User; onTaskClick: (t: Task) => void;
onAddTask: (s: Status) => void; filterUser: string | null; searchQuery: string;
}
export function KanbanBoard({ tasks, currentUser, onTaskClick, onAddTask, filterUser, searchQuery }: KanbanProps) {
let filtered = tasks;
if (currentUser.role === 'employee') filtered = filtered.filter(t => t.assignee === currentUser.id);
if (filterUser) filtered = filtered.filter(t => t.assignee === filterUser);
if (searchQuery) filtered = filtered.filter(t => t.title.toLowerCase().includes(searchQuery.toLowerCase()));
const statuses: Status[] = ['todo', 'inprogress', 'review', 'done'];
return (
{statuses.map(s => (
t.status === s)} onTaskClick={onTaskClick} onAddTask={onAddTask} />
))}
);
}