import React, { useEffect, useMemo, useState } from "react"; import { useLocation, useNavigate } from "react-router-dom"; import { Box, Button, Checkbox, Chip, Collapse, FormControl, FormControlLabel, IconButton, InputAdornment, InputLabel, Menu, MenuItem, Paper, Select, Table, TableBody, TableCell, TableHead, TablePagination, TableRow, TableSortLabel, TextField, Tooltip, Typography, } from "@mui/material"; import { alpha, useTheme } from "@mui/material/styles"; import DeleteOutlineIcon from "@mui/icons-material/DeleteOutline"; import EditOutlinedIcon from "@mui/icons-material/EditOutlined"; import LaunchIcon from "@mui/icons-material/Launch"; import KeyboardArrowDownIcon from "@mui/icons-material/KeyboardArrowDown"; import KeyboardArrowUpIcon from "@mui/icons-material/KeyboardArrowUp"; import RestoreFromTrashOutlinedIcon from "@mui/icons-material/RestoreFromTrashOutlined"; import MoreHorizIcon from "@mui/icons-material/MoreHoriz"; import ViewColumnIcon from "@mui/icons-material/ViewColumn"; import SearchIcon from "@mui/icons-material/Search"; import { api } from "../api"; import { useCompanies } from "../hooks/useCompanies"; import { useDebouncedValue } from "../hooks/useDebouncedValue"; import JobDetailsDialog from "./JobDetailsDialog"; import EditJobDialog from "./EditJobDialog"; import { useToast } from "../toast"; import SavedViewsMenu, { SavedViewParams } from "./SavedViewsMenu"; import { useDialogActions } from "../dialogs"; import { useI18n } from "../i18n/I18nProvider"; import { buildJobWorkspacePath, JOB_DETAILS_TABS } from "../jobWorkspaceRoute"; interface JobApplication { id: number; jobTitle: string; status: string; dateApplied: string; daysSince: number; jobUrl?: string | null; notes?: string | null; location?: string | null; salary?: string | null; tags?: string | null; description?: string | null; isDeleted?: boolean; company: { name: string }; companyId?: number; needsFollowUp?: boolean; followUpReason?: string | null; shortSummary?: string | null; fullSummary?: string | null; tailoredCvText?: string | null; } interface PagedResult { items: T[]; total: number; page: number; pageSize: number; } export type JobTableColumns = { status: boolean; dateApplied: boolean; daysSince: boolean; jobUrl: boolean; }; interface Props { refreshToken: number; pageSize: 15 | 20 | 25; onPageSizeChange: (n: 15 | 20 | 25) => void; columns: JobTableColumns; onColumnsChange: (next: JobTableColumns) => void; mode?: "jobs" | "trash"; } function normalizeStatus(status: string): string { return status === "Interviewing" ? "Interview" : status; } function parseTags(raw?: string | null): string[] { if (!raw) return []; try { const parsed = JSON.parse(raw); return Array.isArray(parsed) ? parsed.filter((x) => typeof x === "string") : []; } catch { return raw.split(/[,;\n]/).map((x) => x.trim()).filter(Boolean); } } function statusTone(status: string): string { switch (normalizeStatus(status)) { case "Offer": return "success"; case "Rejected": return "error"; case "Waiting": case "Ghosted": return "warning"; case "Interview": return "info"; default: return "primary"; } } export default function JobTable({ refreshToken, pageSize, onPageSizeChange, columns, onColumnsChange, mode = "jobs" }: Props) { const theme = useTheme(); const { toast } = useToast(); const { t } = useI18n(); const { confirmAction } = useDialogActions(); const location = useLocation(); const navigate = useNavigate(); const [jobs, setJobs] = useState([]); const [total, setTotal] = useState(0); const [page, setPage] = useState(0); const [expanded, setExpanded] = useState([]); const [selectedIds, setSelectedIds] = useState([]); const [search, setSearch] = useState(""); const debouncedSearch = useDebouncedValue(search, 250); const [includeDeleted, setIncludeDeleted] = useState(mode === "trash"); const [columnsAnchor, setColumnsAnchor] = useState(null); const [statusFilter, setStatusFilter] = useState("All"); const [locationFilter, setLocationFilter] = useState(""); const debouncedLocation = useDebouncedValue(locationFilter, 250); const [needsFollowUpOnly, setNeedsFollowUpOnly] = useState(false); const [readinessFilter, setReadinessFilter] = useState<"all" | "needs-work" | "interview">("all"); const { companies } = useCompanies(); const [companyFilterId, setCompanyFilterId] = useState("All"); const [detailsJobId, setDetailsJobId] = useState(null); const [detailsInitialTab, setDetailsInitialTab] = useState(0); const [detailsFollowUpMode, setDetailsFollowUpMode] = useState(undefined); const [editJobId, setEditJobId] = useState(null); const [reloadToken, setReloadToken] = useState(0); const [statusAnchor, setStatusAnchor] = useState(null); const [statusJobId, setStatusJobId] = useState(null); const [sortBy, setSortBy] = useState<"dateApplied" | "company" | "jobTitle" | "status" | "daysSince" | "location">("dateApplied"); const [sortDir, setSortDir] = useState<"asc" | "desc">("desc"); const params = useMemo(() => ({ page: page + 1, pageSize, q: debouncedSearch.trim() || undefined, status: statusFilter !== "All" ? statusFilter : undefined, companyId: companyFilterId === "All" ? undefined : companyFilterId, location: debouncedLocation.trim() || undefined, includeDeleted, deletedOnly: mode === "trash" ? true : undefined, sortBy, sortDir, needsFollowUp: needsFollowUpOnly ? true : undefined, }), [page, pageSize, debouncedSearch, statusFilter, companyFilterId, debouncedLocation, includeDeleted, mode, sortBy, sortDir, needsFollowUpOnly]); useEffect(() => { api.get>("/jobapplications", { params }).then((r) => { setJobs(r.data.items); setTotal(r.data.total); setSelectedIds([]); }); }, [params, refreshToken, reloadToken]); useEffect(() => { const paramsSearch = new URLSearchParams(location.search); const openId = Number(paramsSearch.get("open") || 0); const tabIndex = Number(paramsSearch.get("tab") || 0); const followMode = paramsSearch.get("followMode") || undefined; if (!openId || jobs.length === 0) return; const job = jobs.find((j) => j.id === openId); if (!job) return; setDetailsJobId(openId); setDetailsInitialTab(Number.isFinite(tabIndex) ? Math.max(0, Math.min(9, tabIndex)) : 0); setDetailsFollowUpMode(followMode); paramsSearch.delete("open"); paramsSearch.delete("tab"); navigate({ pathname: location.pathname, search: paramsSearch.toString() ? `?${paramsSearch.toString()}` : "" }, { replace: true }); }, [jobs, location.pathname, location.search, navigate]); const requestSort = (key: typeof sortBy) => { if (sortBy === key) setSortDir((d) => (d === "asc" ? "desc" : "asc")); else { setSortBy(key); setSortDir("asc"); } setPage(0); }; const toggleExpanded = (id: number) => { setExpanded((prev) => (prev.includes(id) ? prev.filter((x) => x !== id) : [...prev, id])); }; const filteredJobs = useMemo(() => { if (readinessFilter === "all") return jobs; if (readinessFilter === "interview") return jobs.filter((job) => job.status === "Interview" || job.status === "Interviewing"); return jobs.filter((job) => !job.tailoredCvText || !job.notes); }, [jobs, readinessFilter]); const selectedAllOnPage = filteredJobs.length > 0 && filteredJobs.every((job) => selectedIds.includes(job.id)); const toggleSelectAll = (checked: boolean) => { setSelectedIds(checked ? filteredJobs.map((job) => job.id) : []); }; const toggleSelected = (id: number, checked: boolean) => { setSelectedIds((prev) => checked ? [...prev, id] : prev.filter((x) => x !== id)); }; const confirmDelete = async (jobsToDelete: JobApplication[]) => { if (jobsToDelete.length === 0) return false; if (jobsToDelete.length === 1) { const job = jobsToDelete[0]; return confirmAction(t("jobTableMoveOneConfirm", { title: job.jobTitle, company: job.company?.name ?? t("jobTableCompany") }), { title: t("jobTableMoveToTrashTitle"), confirmLabel: t("jobTableMove"), destructive: true }); } return confirmAction(t("jobTableMoveManyConfirm", { count: jobsToDelete.length }), { title: t("jobTableMoveJobsToTrashTitle"), confirmLabel: t("jobTableMove"), destructive: true }); }; const softDelete = async (job: JobApplication) => { if (!(await confirmDelete([job]))) return; try { await api.delete(`/jobapplications/${job.id}`); toast(t("jobTableMovedToTrash"), "success", { label: "Undo", onClick: () => { void restore(job.id); } }); setReloadToken((t) => t + 1); } catch { toast(t("jobTableDeleteFailed"), "error"); } }; const restore = async (id: number) => { try { await api.post(`/jobapplications/${id}/restore`); toast(t("jobTableRestored"), "success"); setReloadToken((t) => t + 1); } catch { toast(t("jobTableRestoreFailed"), "error"); } }; const setStatusQuick = async (id: number, status: string) => { try { await api.patch(`/jobapplications/${id}/status`, { status }); toast(t("jobTableStatusSet", { status }), "success"); setReloadToken((t) => t + 1); } catch { toast(t("jobTableStatusUpdateFailed"), "error"); } }; const runBulkAction = async (action: "delete" | "restore" | "status", value?: string) => { if (selectedIds.length === 0) return; const selectedJobs = jobs.filter((job) => selectedIds.includes(job.id)); if (action === "delete" && !(await confirmDelete(selectedJobs))) return; try { await Promise.all(selectedIds.map((id) => { if (action === "delete") return api.delete(`/jobapplications/${id}`); if (action === "restore") return api.post(`/jobapplications/${id}/restore`); return api.patch(`/jobapplications/${id}/status`, { status: value }); })); toast(t("jobTableUpdatedJobs", { count: selectedIds.length }), "success"); setReloadToken((t) => t + 1); setSelectedIds([]); } catch { toast(t("jobTableBulkActionFailed"), "error"); } }; const generateOverview = (job: JobApplication) => { if (job.fullSummary) return job.fullSummary; if (job.shortSummary) return job.shortSummary; const src = (job.description || job.notes || "").replace(/<[^>]+>/g, " ").replace(/\s+/g, " ").trim(); return src.length > 220 ? `${src.slice(0, 220)}...` : src; }; const openFollowUpWorkspace = (jobId: number) => { navigate(buildJobWorkspacePath(jobId, { tab: JOB_DETAILS_TABS.followUp, followMode: "waiting-update" })); }; const openTailoredCvWorkspace = (jobId: number) => { navigate(buildJobWorkspacePath(jobId, { tab: JOB_DETAILS_TABS.tailoredCv })); }; const getPackageActionDetail = (job: JobApplication) => { const missingTailoredCv = !job.tailoredCvText; const missingNotes = !job.notes?.trim(); if (missingTailoredCv && missingNotes) return t("jobTablePackageMissingCvAndNotes"); if (missingTailoredCv) return t("jobTableCvMissing"); if (missingNotes) return t("jobTablePackageMissingNotes"); return null; }; const getActionSignals = (job: JobApplication) => { const signals: Array<{ label: string; detail: string; onClick: () => void; variant: "contained" | "outlined"; color?: "warning" | "primary"; }> = []; if (job.needsFollowUp) { signals.push({ label: t("jobTableFollowUp"), detail: job.followUpReason ?? t("jobTableNeedsFollowUp"), onClick: () => openFollowUpWorkspace(job.id), variant: "contained", color: "warning", }); } const packageDetail = !job.isDeleted ? getPackageActionDetail(job) : null; if (packageDetail) { signals.push({ label: t("jobTablePackageWork"), detail: packageDetail, onClick: () => openTailoredCvWorkspace(job.id), variant: job.needsFollowUp ? "outlined" : "contained", color: job.needsFollowUp ? "primary" : "warning", }); } return signals; }; const getPrimaryAction = (job: JobApplication) => getActionSignals(job)[0] ?? null; return ( { setSearch(e.target.value); setPage(0); }} placeholder={t("jobTableSearchPlaceholder")} size="small" InputProps={{ startAdornment: }} sx={{ minWidth: 320, flex: "1 1 320px" }} /> {t("jobTableStatus")} {t("jobTableCompany")} { setLocationFilter(e.target.value); setPage(0); }} sx={{ minWidth: 200, flex: "1 1 200px" }} /> {mode === "jobs" ? { setNeedsFollowUpOnly(e.target.checked); setPage(0); }} />} label={t("jobTableNeedsFollowUp")} /> : null} {mode === "jobs" ? ( {t("jobTableReadiness")} ) : null} {mode === "jobs" ? { setIncludeDeleted(e.target.checked); setPage(0); }} />} label={t("jobTableShowDeleted")} /> : null} { setSearch(p.q ?? ""); setStatusFilter(p.status ?? "All"); setCompanyFilterId(p.companyId ?? "All"); setLocationFilter(p.location ?? ""); setNeedsFollowUpOnly(Boolean(p.needsFollowUp)); setPage(0); }} /> setColumnsAnchor(e.currentTarget)}> {selectedIds.length > 0 ? ( {t("jobTableSelected", { count: selectedIds.length })} {mode === "trash" ? : } {mode === "jobs" ? ["Waiting", "Interview", "Rejected", "Ghosted", "Offer"].map((s) => ) : null} ) : null} setColumnsAnchor(null)}> {([ ["status", t("settingsColumnStatus")], ["dateApplied", t("settingsColumnDateApplied")], ["daysSince", t("settingsColumnDays")], ["jobUrl", t("settingsColumnJobUrl")] ] as const).map(([key, label]) => ( onColumnsChange({ ...columns, [key]: !columns[key] })}> {label} ))} 0 && !selectedAllOnPage} onChange={(e) => toggleSelectAll(e.target.checked)} /> requestSort("company")}>{t("jobTableCompany")} requestSort("jobTitle")}>{t("jobTableRole")} {columns.status ? requestSort("status")}>{t("jobTableStatus")} : null} {columns.dateApplied ? requestSort("dateApplied")}>{t("jobTableDateApplied")} : null} {columns.daysSince ? requestSort("daysSince")}>{t("jobTableDays")} : null} {columns.jobUrl ? {t("settingsColumnJobUrl")} : null} {t("jobTableActions")} {filteredJobs.map((job) => { const open = expanded.includes(job.id); const toneName = statusTone(job.status); const tone = toneName === "error" ? theme.palette.error.main : toneName === "warning" ? theme.palette.warning.main : toneName === "success" ? theme.palette.success.main : toneName === "info" ? theme.palette.info.main : theme.palette.primary.main; const primaryAction = getPrimaryAction(job); const actionSignals = getActionSignals(job); return ( toggleSelected(job.id, e.target.checked)} /> toggleExpanded(job.id)}>{open ? : } {job.company?.name ?? ""} {job.jobTitle} {actionSignals.map((signal) => ( ))} {columns.status ? : null} {columns.dateApplied ? {new Date(job.dateApplied).toLocaleDateString()} : null} {columns.daysSince ? {job.daysSince} : null} {columns.jobUrl ? {job.jobUrl ? {t("jobTableLink")} : ""} : null} {primaryAction ? ( <> {t("editJobNextAction")} {primaryAction.detail} ) : null} setEditJobId(job.id)}> { setStatusJobId(job.id); setStatusAnchor(e.currentTarget); }}> setDetailsJobId(job.id)}> {(mode === "trash" || (includeDeleted && job.isDeleted)) ? void restore(job.id)}> : void softDelete(job)}>} {t("jobTableLocation")}{job.location ?? "-"} {t("addJobModalSalary")}{job.salary ?? "-"} {t("settingsColumnJobUrl")}{job.jobUrl ? {t("jobTableOpenListing")} : "-"} {t("jobTableSkills")}{parseTags(job.tags).length ? parseTags(job.tags).slice(0, 8).map((tag) => ) : {t("jobTableNoTags")}} {t("jobTableOverview")}{generateOverview(job) || t("jobTableNoSummaryYet")} ); })} {filteredJobs.length === 0 ? {t("jobTableNoJobsFound")} : null}
setPage(next)} rowsPerPage={pageSize} onRowsPerPageChange={(e) => { onPageSizeChange(Number(e.target.value) as 15 | 20 | 25); setPage(0); }} rowsPerPageOptions={[15, 20, 25]} />
{ setDetailsJobId(null); setDetailsInitialTab(0); setDetailsFollowUpMode(undefined); }} /> setEditJobId(null)} onSaved={() => setReloadToken((t) => t + 1)} /> { setStatusAnchor(null); setStatusJobId(null); }}> {(["Waiting", "Interview", "Offer", "Rejected", "Ghosted"] as const).map((s) => { if (statusJobId) void setStatusQuick(statusJobId, s); setStatusAnchor(null); setStatusJobId(null); }}>{t("jobTableSetStatus", { status: s })})}
); }