2026-03-27 05:35:18 +01:00
|
|
|
import { For, Show, createMemo, createSignal, onMount } from 'solid-js';
|
2026-03-19 14:21:49 +01:00
|
|
|
import AdminShell from '~/components/AdminShell';
|
2026-03-27 05:35:18 +01:00
|
|
|
import type { CrudRecord } from '~/lib/admin/types';
|
|
|
|
|
|
|
|
|
|
const API = '/api/gateway';
|
|
|
|
|
|
|
|
|
|
type VerificationRecord = CrudRecord & {
|
|
|
|
|
applicantName?: string;
|
|
|
|
|
userType: 'CUSTOMER' | 'PROFESSIONAL' | 'COMPANY' | 'JOBSEEKER';
|
|
|
|
|
verificationType: 'IDENTITY' | 'BUSINESS' | 'PROFILE' | 'DOCUMENT' | 'MIXED';
|
|
|
|
|
submittedDate?: string;
|
|
|
|
|
documentsCount?: number;
|
2026-03-30 21:43:49 +02:00
|
|
|
documents?: Array<{
|
|
|
|
|
name: string;
|
|
|
|
|
fileName: string;
|
|
|
|
|
state: 'APPROVED' | 'RECEIVED' | 'MISSING' | 'REQUESTED';
|
|
|
|
|
}>;
|
|
|
|
|
requestedDocuments?: string[];
|
|
|
|
|
roleTags?: string[];
|
|
|
|
|
primaryService?: string;
|
|
|
|
|
area?: string;
|
|
|
|
|
userId?: string;
|
|
|
|
|
roleKey?: string;
|
2026-03-27 05:35:18 +01:00
|
|
|
assignedVerifier?: string;
|
|
|
|
|
priority: 'LOW' | 'MEDIUM' | 'HIGH';
|
|
|
|
|
status: 'PENDING' | 'IN_REVIEW' | 'PARTIALLY_VERIFIED' | 'VERIFIED' | 'FLAGGED' | 'RE_UPLOAD_REQUESTED' | 'REJECTED';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function StatusBadge(props: { status: string }) {
|
|
|
|
|
const getColors = () => {
|
|
|
|
|
switch (props.status) {
|
|
|
|
|
case 'VERIFIED': return { border: '#B7E4C7', bg: '#DEF7E8', text: '#0B8A4A', dot: '#0B8A4A' };
|
|
|
|
|
case 'IN_REVIEW': return { border: '#F6D78F', bg: '#FFF3D6', text: '#B7791F', dot: '#B7791F' };
|
|
|
|
|
case 'PENDING': return { border: '#D1D5DB', bg: '#F3F4F6', text: '#4B5563', dot: '#9CA3AF' };
|
|
|
|
|
case 'RE_UPLOAD_REQUESTED': return { border: '#FDE68A', bg: '#FEF3C7', text: '#D97706', dot: '#D97706' };
|
|
|
|
|
case 'FLAGGED': return { border: '#FECACA', bg: '#FEF2F2', text: '#DC2626', dot: '#DC2626' };
|
|
|
|
|
case 'REJECTED': return { border: '#FECACA', bg: '#FEF2F2', text: '#DC2626', dot: '#DC2626' };
|
|
|
|
|
default: return { border: '#D1D5DB', bg: '#F3F4F6', text: '#4B5563', dot: '#9CA3AF' };
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
const colors = getColors();
|
|
|
|
|
const label = props.status.split('_').map(w => w.charAt(0) + w.slice(1).toLowerCase()).join(' ');
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<span style={`display:inline-flex;align-items:center;border-radius:9999px;border:1px solid ${colors.border};background:${colors.bg};color:${colors.text};padding:2px 10px;font-size:12px;font-weight:500`}>
|
|
|
|
|
<span style={`display:inline-block;width:6px;height:6px;border-radius:50%;background:${colors.dot};margin-right:5px;flex-shrink:0`} />
|
|
|
|
|
{label}
|
|
|
|
|
</span>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function PriorityBadge(props: { priority: string }) {
|
|
|
|
|
const color = props.priority === 'HIGH' ? '#DC2626' : props.priority === 'MEDIUM' ? '#F59E0B' : '#16A34A';
|
|
|
|
|
return (
|
|
|
|
|
<span style={`display:inline-flex;align-items:center;gap:4px;font-size:12px;font-weight:600;color:${color}`}>
|
|
|
|
|
<span style={`width:6px;height:6px;border-radius:50%;background:${color}`} />
|
|
|
|
|
{props.priority}
|
|
|
|
|
</span>
|
|
|
|
|
);
|
|
|
|
|
}
|
2026-03-19 14:21:49 +01:00
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const DEFAULT_DOCS_BY_USER_TYPE: Record<VerificationRecord['userType'], string[]> = {
|
|
|
|
|
COMPANY: ['GST Certificate', 'PAN Card', 'Incorporation Certificate'],
|
|
|
|
|
CUSTOMER: ['Identity Proof', 'Address Proof'],
|
|
|
|
|
PROFESSIONAL: ['Identity Proof', 'Address Proof', 'Portfolio Ownership'],
|
|
|
|
|
JOBSEEKER: ['Identity Proof', 'Address Proof', 'Resume'],
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const toTitle = (value: string) => String(value || '')
|
|
|
|
|
.replace(/_/g, ' ')
|
|
|
|
|
.replace(/\b\w/g, (c) => c.toUpperCase());
|
|
|
|
|
|
|
|
|
|
type SubmissionData = {
|
|
|
|
|
role_key?: string;
|
|
|
|
|
onboarding?: {
|
|
|
|
|
status?: string;
|
|
|
|
|
progress_json?: Record<string, unknown>;
|
|
|
|
|
} | null;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function extractRoleTags(source: any): string[] {
|
|
|
|
|
const values: string[] = [];
|
|
|
|
|
const pushValue = (value: unknown) => {
|
|
|
|
|
if (!value) return;
|
|
|
|
|
if (Array.isArray(value)) {
|
|
|
|
|
value.forEach((v) => pushValue(v));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const text = String(value || '').trim();
|
|
|
|
|
if (!text) return;
|
|
|
|
|
values.push(text);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
pushValue(source?.role_key);
|
|
|
|
|
pushValue(source?.roleKey);
|
|
|
|
|
pushValue(source?.role_keys);
|
|
|
|
|
pushValue(source?.roleKeys);
|
|
|
|
|
pushValue(source?.roles);
|
|
|
|
|
pushValue(source?.categories);
|
|
|
|
|
pushValue(source?.category);
|
|
|
|
|
pushValue(source?.service_category);
|
|
|
|
|
pushValue(source?.serviceCategory);
|
|
|
|
|
pushValue(source?.profession);
|
|
|
|
|
pushValue(source?.service_type);
|
|
|
|
|
|
|
|
|
|
const unique = Array.from(new Set(values.map((v) => toTitle(v))));
|
|
|
|
|
return unique.slice(0, 4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function flattenFields(obj: Record<string, unknown>, prefix = ''): Array<{ key: string; value: string }> {
|
|
|
|
|
const result: Array<{ key: string; value: string }> = [];
|
|
|
|
|
for (const [k, v] of Object.entries(obj || {})) {
|
|
|
|
|
const label = prefix ? `${prefix}.${k}` : k;
|
|
|
|
|
if (v !== null && typeof v === 'object' && !Array.isArray(v)) {
|
|
|
|
|
result.push(...flattenFields(v as Record<string, unknown>, label));
|
|
|
|
|
} else if (Array.isArray(v)) {
|
|
|
|
|
result.push({ key: label, value: v.map((item) => String(item ?? '')).join(', ') });
|
|
|
|
|
} else {
|
|
|
|
|
result.push({ key: label, value: String(v ?? '') });
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadSubmissionDetails(userId: string, roleKey: string): Promise<SubmissionData | null> {
|
|
|
|
|
if (!userId) return null;
|
|
|
|
|
const qs = roleKey ? `?roleKey=${encodeURIComponent(roleKey)}` : '';
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(`${API}/api/admin/approvals/submission/${userId}${qs}`);
|
|
|
|
|
if (!res.ok) return null;
|
|
|
|
|
return res.json();
|
|
|
|
|
} catch {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-19 14:21:49 +01:00
|
|
|
export default function VerificationManagementPage() {
|
2026-03-30 21:43:49 +02:00
|
|
|
const [listTab, setListTab] = createSignal<'all' | 'view' | 'flagged'>('all');
|
2026-03-27 05:35:18 +01:00
|
|
|
const [detailTab, setDetailTab] = createSignal<'overview' | 'documents' | 'checklist' | 'logs'>('overview');
|
|
|
|
|
|
|
|
|
|
const [search, setSearch] = createSignal('');
|
|
|
|
|
const [rows, setRows] = createSignal<VerificationRecord[]>([]);
|
|
|
|
|
const [viewingCase, setViewingCase] = createSignal<VerificationRecord | null>(null);
|
|
|
|
|
const [openMenuId, setOpenMenuId] = createSignal<string | null>(null);
|
|
|
|
|
|
|
|
|
|
const [statusFilter, setStatusFilter] = createSignal<'all' | 'pending' | 'flagged'>('all');
|
|
|
|
|
const [sortBy, setSortBy] = createSignal<'submitted_desc' | 'submitted_asc' | 'priority_desc' | 'priority_asc'>('submitted_desc');
|
|
|
|
|
const [sortMenuOpen, setSortMenuOpen] = createSignal(false);
|
|
|
|
|
const [filterMenuOpen, setFilterMenuOpen] = createSignal(false);
|
2026-03-30 21:43:49 +02:00
|
|
|
const [requestNote, setRequestNote] = createSignal('');
|
|
|
|
|
const [requestingDocs, setRequestingDocs] = createSignal<string[]>([]);
|
2026-03-30 04:48:09 +02:00
|
|
|
const [error, setError] = createSignal('');
|
2026-03-30 21:43:49 +02:00
|
|
|
const [loadingLinkedDetails, setLoadingLinkedDetails] = createSignal(false);
|
|
|
|
|
const [actionNote, setActionNote] = createSignal('');
|
2026-03-27 05:35:18 +01:00
|
|
|
|
|
|
|
|
const load = async () => {
|
2026-03-30 04:48:09 +02:00
|
|
|
setError('');
|
|
|
|
|
try {
|
|
|
|
|
const accessToken = typeof sessionStorage !== 'undefined'
|
|
|
|
|
? sessionStorage.getItem('nxtgauge_admin_access_token') || ''
|
|
|
|
|
: '';
|
|
|
|
|
const res = await fetch(`${API}/api/admin/approvals?page=1&limit=100`, {
|
|
|
|
|
headers: {
|
|
|
|
|
Accept: 'application/json',
|
|
|
|
|
...(accessToken ? { Authorization: `Bearer ${accessToken}` } : {}),
|
|
|
|
|
},
|
|
|
|
|
credentials: 'include',
|
|
|
|
|
});
|
|
|
|
|
if (!res.ok) throw new Error(`Request failed (${res.status})`);
|
|
|
|
|
const payload = await res.json().catch(() => ({} as any));
|
|
|
|
|
const jobs = Array.isArray(payload?.jobs) ? payload.jobs : [];
|
|
|
|
|
const requirements = Array.isArray(payload?.requirements) ? payload.requirements : [];
|
|
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const buildDocuments = (userType: VerificationRecord['userType'], source: any) => {
|
|
|
|
|
const names = DEFAULT_DOCS_BY_USER_TYPE[userType];
|
|
|
|
|
const provided = Array.isArray(source?.documents) ? source.documents : [];
|
|
|
|
|
const mapped = names.map((name, index) => {
|
|
|
|
|
const fromPayload = provided[index];
|
|
|
|
|
const fileName = String(
|
|
|
|
|
fromPayload?.file_name
|
|
|
|
|
|| fromPayload?.fileName
|
|
|
|
|
|| fromPayload?.name
|
|
|
|
|
|| `${name.toLowerCase().replace(/\s+/g, '_')}.pdf`,
|
|
|
|
|
);
|
|
|
|
|
const stateRaw = String(fromPayload?.state || fromPayload?.status || '').toUpperCase();
|
|
|
|
|
const state: 'APPROVED' | 'RECEIVED' | 'MISSING' | 'REQUESTED' =
|
|
|
|
|
stateRaw === 'APPROVED' || stateRaw === 'VERIFIED'
|
|
|
|
|
? 'APPROVED'
|
|
|
|
|
: stateRaw === 'REQUESTED' || stateRaw === 'RE_UPLOAD_REQUESTED'
|
|
|
|
|
? 'REQUESTED'
|
|
|
|
|
: stateRaw === 'RECEIVED'
|
|
|
|
|
? 'RECEIVED'
|
|
|
|
|
: index === names.length - 1
|
|
|
|
|
? 'MISSING'
|
|
|
|
|
: 'APPROVED';
|
|
|
|
|
return { name, fileName, state };
|
|
|
|
|
});
|
|
|
|
|
const requested = mapped.filter((d) => d.state === 'REQUESTED' || d.state === 'MISSING').map((d) => d.name);
|
|
|
|
|
return { mapped, requested };
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const jobCases: VerificationRecord[] = jobs.map((job: any) => {
|
|
|
|
|
const docs = buildDocuments('COMPANY', job);
|
|
|
|
|
return {
|
|
|
|
|
id: `job-${String(job.id)}`,
|
|
|
|
|
name: `Job Verification - ${String(job.title || 'Untitled Job')}`,
|
|
|
|
|
applicantName: String(job.title || 'Untitled Job'),
|
|
|
|
|
userType: 'COMPANY',
|
|
|
|
|
verificationType: 'BUSINESS',
|
|
|
|
|
submittedDate: String(job.created_at || ''),
|
|
|
|
|
documentsCount: docs.mapped.length,
|
|
|
|
|
documents: docs.mapped,
|
|
|
|
|
requestedDocuments: docs.requested,
|
|
|
|
|
roleTags: extractRoleTags(job),
|
|
|
|
|
primaryService: String(job.category || job.department || job.role || 'Job Posting'),
|
|
|
|
|
area: String(job.location || job.city || job.work_mode || '—'),
|
|
|
|
|
userId: String(job.user_id || job.created_by || ''),
|
|
|
|
|
roleKey: String(job.role_key || job.roleKey || ''),
|
|
|
|
|
assignedVerifier: 'Unassigned',
|
|
|
|
|
priority: 'HIGH',
|
|
|
|
|
status: docs.requested.length ? 'RE_UPLOAD_REQUESTED' : 'IN_REVIEW',
|
|
|
|
|
updatedAt: String(job.updated_at || job.created_at || ''),
|
|
|
|
|
};
|
|
|
|
|
});
|
2026-03-30 04:48:09 +02:00
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const requirementCases: VerificationRecord[] = requirements.map((req: any) => {
|
|
|
|
|
const docs = buildDocuments('CUSTOMER', req);
|
|
|
|
|
return {
|
|
|
|
|
id: `requirement-${String(req.id)}`,
|
|
|
|
|
name: `Requirement Verification - ${String(req.title || 'Untitled Requirement')}`,
|
|
|
|
|
applicantName: String(req.title || 'Untitled Requirement'),
|
|
|
|
|
userType: 'CUSTOMER',
|
|
|
|
|
verificationType: 'PROFILE',
|
|
|
|
|
submittedDate: String(req.created_at || ''),
|
|
|
|
|
documentsCount: docs.mapped.length,
|
|
|
|
|
documents: docs.mapped,
|
|
|
|
|
requestedDocuments: docs.requested,
|
|
|
|
|
roleTags: extractRoleTags(req),
|
|
|
|
|
primaryService: String(req.category || req.profession || req.service_type || 'Requirement'),
|
|
|
|
|
area: String(req.location || req.city || req.area || '—'),
|
|
|
|
|
userId: String(req.user_id || req.created_by || ''),
|
|
|
|
|
roleKey: String(req.role_key || req.roleKey || ''),
|
|
|
|
|
assignedVerifier: 'Unassigned',
|
|
|
|
|
priority: 'MEDIUM',
|
|
|
|
|
status: docs.requested.length ? 'RE_UPLOAD_REQUESTED' : 'PENDING',
|
|
|
|
|
updatedAt: String(req.updated_at || req.created_at || ''),
|
|
|
|
|
};
|
|
|
|
|
});
|
2026-03-30 04:48:09 +02:00
|
|
|
|
|
|
|
|
setRows([...jobCases, ...requirementCases]);
|
|
|
|
|
} catch (e: any) {
|
|
|
|
|
setRows([]);
|
|
|
|
|
setError(e?.message || 'Could not reach verification API.');
|
|
|
|
|
}
|
2026-03-27 05:35:18 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
onMount(() => void load());
|
|
|
|
|
|
|
|
|
|
const formatDate = (v?: string) => {
|
|
|
|
|
const s = v || '';
|
|
|
|
|
if (/^\d{4}-\d{2}-\d{2}$/.test(s)) return s;
|
|
|
|
|
return s.slice(0, 10) || '—';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const filteredRows = createMemo(() => {
|
|
|
|
|
let list = rows();
|
|
|
|
|
const f = statusFilter();
|
|
|
|
|
if (f === 'pending') list = list.filter((r) => r.status === 'PENDING' || r.status === 'IN_REVIEW');
|
|
|
|
|
if (f === 'flagged') list = list.filter((r) => r.status === 'FLAGGED');
|
|
|
|
|
|
|
|
|
|
const q = search().trim().toLowerCase();
|
|
|
|
|
if (q) {
|
|
|
|
|
list = list.filter((r) =>
|
|
|
|
|
String(r.applicantName || '').toLowerCase().includes(q)
|
|
|
|
|
|| String(r.id || '').toLowerCase().includes(q)
|
|
|
|
|
|| String(r.verificationType || '').toLowerCase().includes(q)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const sorted = [...list];
|
|
|
|
|
const mode = sortBy();
|
|
|
|
|
const priorityRank = (p: VerificationRecord['priority']) => (p === 'HIGH' ? 3 : p === 'MEDIUM' ? 2 : 1);
|
|
|
|
|
sorted.sort((a, b) => {
|
|
|
|
|
const ad = Date.parse(String(a.submittedDate || a.updatedAt || '')) || 0;
|
|
|
|
|
const bd = Date.parse(String(b.submittedDate || b.updatedAt || '')) || 0;
|
|
|
|
|
if (mode === 'submitted_asc') return ad - bd;
|
|
|
|
|
if (mode === 'priority_desc') return priorityRank(b.priority) - priorityRank(a.priority);
|
|
|
|
|
if (mode === 'priority_asc') return priorityRank(a.priority) - priorityRank(b.priority);
|
|
|
|
|
return bd - ad;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return sorted;
|
|
|
|
|
});
|
|
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const flaggedCount = createMemo(() => rows().filter((r) => r.status === 'FLAGGED').length);
|
|
|
|
|
const reviewCount = createMemo(() => rows().filter((r) => r.status === 'PENDING' || r.status === 'IN_REVIEW').length);
|
|
|
|
|
|
|
|
|
|
const setCaseStatus = (id: string, status: VerificationRecord['status']) => {
|
|
|
|
|
const now = new Date().toISOString();
|
|
|
|
|
setRows((prev) => prev.map((r) => (r.id === id ? { ...r, status, updatedAt: now } : r)));
|
|
|
|
|
setViewingCase((prev) => (prev && prev.id === id ? { ...prev, status, updatedAt: now } : prev));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const markVerifiedCase = (row: VerificationRecord) => {
|
|
|
|
|
const now = new Date().toISOString();
|
|
|
|
|
setRows((prev) => prev.map((r) => {
|
|
|
|
|
if (r.id !== row.id) return r;
|
|
|
|
|
return {
|
|
|
|
|
...r,
|
|
|
|
|
status: 'VERIFIED',
|
|
|
|
|
requestedDocuments: [],
|
|
|
|
|
documents: (r.documents || []).map((d) => (d.state === 'MISSING' || d.state === 'REQUESTED' ? { ...d, state: 'RECEIVED' } : d)),
|
|
|
|
|
updatedAt: now,
|
|
|
|
|
};
|
|
|
|
|
}));
|
|
|
|
|
setViewingCase((prev) => {
|
|
|
|
|
if (!prev || prev.id !== row.id) return prev;
|
|
|
|
|
return {
|
|
|
|
|
...prev,
|
|
|
|
|
status: 'VERIFIED',
|
|
|
|
|
requestedDocuments: [],
|
|
|
|
|
documents: (prev.documents || []).map((d) => (d.state === 'MISSING' || d.state === 'REQUESTED' ? { ...d, state: 'RECEIVED' } : d)),
|
|
|
|
|
updatedAt: now,
|
|
|
|
|
};
|
|
|
|
|
});
|
2026-03-27 05:35:18 +01:00
|
|
|
setOpenMenuId(null);
|
|
|
|
|
};
|
|
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const requestReuploadCase = (row: VerificationRecord, requested: string[]) => {
|
|
|
|
|
const fallback = (row.documents || []).filter((d) => d.state === 'MISSING' || d.state === 'REQUESTED').map((d) => d.name);
|
|
|
|
|
const nextRequested = requested.length ? requested : fallback;
|
|
|
|
|
if (!nextRequested.length) return;
|
|
|
|
|
const now = new Date().toISOString();
|
|
|
|
|
const requestedSet = new Set(nextRequested);
|
|
|
|
|
setRows((prev) => prev.map((r) => {
|
|
|
|
|
if (r.id !== row.id) return r;
|
|
|
|
|
return {
|
|
|
|
|
...r,
|
|
|
|
|
status: 'RE_UPLOAD_REQUESTED',
|
|
|
|
|
requestedDocuments: nextRequested,
|
|
|
|
|
documents: (r.documents || []).map((d) => (requestedSet.has(d.name) ? { ...d, state: 'REQUESTED' } : d)),
|
|
|
|
|
updatedAt: now,
|
|
|
|
|
};
|
|
|
|
|
}));
|
|
|
|
|
setViewingCase((prev) => {
|
|
|
|
|
if (!prev || prev.id !== row.id) return prev;
|
|
|
|
|
return {
|
|
|
|
|
...prev,
|
|
|
|
|
status: 'RE_UPLOAD_REQUESTED',
|
|
|
|
|
requestedDocuments: nextRequested,
|
|
|
|
|
documents: (prev.documents || []).map((d) => (requestedSet.has(d.name) ? { ...d, state: 'REQUESTED' } : d)),
|
|
|
|
|
updatedAt: now,
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
setOpenMenuId(null);
|
2026-03-27 05:35:18 +01:00
|
|
|
};
|
|
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const enrichCaseFromSubmission = async (row: VerificationRecord) => {
|
|
|
|
|
if (!row.userId) return;
|
|
|
|
|
setLoadingLinkedDetails(true);
|
|
|
|
|
try {
|
|
|
|
|
const submission = await loadSubmissionDetails(row.userId || '', row.roleKey || '');
|
|
|
|
|
const progress = (submission?.onboarding?.progress_json && typeof submission.onboarding.progress_json === 'object')
|
|
|
|
|
? submission.onboarding.progress_json
|
|
|
|
|
: {};
|
|
|
|
|
const flattened = flattenFields(progress as Record<string, unknown>);
|
|
|
|
|
const getField = (...hints: string[]) => {
|
|
|
|
|
const hit = flattened.find((field) => hints.some((hint) => field.key.toLowerCase().includes(hint)));
|
|
|
|
|
return String(hit?.value || '').trim();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const linkedDocs = flattened
|
|
|
|
|
.filter((field) => {
|
|
|
|
|
const key = field.key.toLowerCase();
|
|
|
|
|
const value = String(field.value || '').toLowerCase();
|
|
|
|
|
const isDocLike = /(document|doc|file|upload|resume|cv|certificate|license|id|proof|portfolio)/.test(key);
|
|
|
|
|
const hasPath = value.startsWith('http') || value.startsWith('/');
|
|
|
|
|
return isDocLike && (hasPath || value.endsWith('.pdf') || value.endsWith('.jpg') || value.endsWith('.jpeg') || value.endsWith('.png'));
|
|
|
|
|
})
|
|
|
|
|
.slice(0, 8)
|
|
|
|
|
.map((field) => {
|
|
|
|
|
const nameParts = field.key.split('.');
|
|
|
|
|
const baseName = nameParts[nameParts.length - 1] || field.key;
|
|
|
|
|
const humanName = toTitle(baseName.replace(/_/g, ' ').replace(/file|upload|url/gi, '').trim() || baseName);
|
|
|
|
|
const fileName = String(field.value || '').split('/').pop() || `${humanName.toLowerCase().replace(/\s+/g, '_')}.pdf`;
|
|
|
|
|
const requestedSet = new Set(row.requestedDocuments || []);
|
|
|
|
|
return {
|
|
|
|
|
name: humanName,
|
|
|
|
|
fileName,
|
|
|
|
|
state: (requestedSet.has(humanName) ? 'REQUESTED' : 'RECEIVED') as 'REQUESTED' | 'RECEIVED',
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const nextRoleTags = Array.from(new Set([
|
|
|
|
|
...extractRoleTags(progress),
|
|
|
|
|
...extractRoleTags({ role_key: submission?.role_key }),
|
|
|
|
|
...(row.roleTags || []),
|
|
|
|
|
])).slice(0, 5);
|
|
|
|
|
const nextPrimary = getField('category', 'profession', 'service', 'role', 'job_title') || row.primaryService || '—';
|
|
|
|
|
const nextArea = getField('location', 'city', 'area', 'place', 'venue') || row.area || '—';
|
|
|
|
|
const nextDocs = linkedDocs.length ? linkedDocs : (row.documents || []);
|
|
|
|
|
|
|
|
|
|
setRows((prev) => prev.map((item) => {
|
|
|
|
|
if (item.id !== row.id) return item;
|
|
|
|
|
return {
|
|
|
|
|
...item,
|
|
|
|
|
roleTags: nextRoleTags.length ? nextRoleTags : item.roleTags,
|
|
|
|
|
primaryService: nextPrimary,
|
|
|
|
|
area: nextArea,
|
|
|
|
|
documents: nextDocs,
|
|
|
|
|
documentsCount: nextDocs.length || item.documentsCount,
|
|
|
|
|
roleKey: item.roleKey || String(submission?.role_key || ''),
|
|
|
|
|
};
|
|
|
|
|
}));
|
|
|
|
|
setViewingCase((prev) => {
|
|
|
|
|
if (!prev || prev.id !== row.id) return prev;
|
|
|
|
|
return {
|
|
|
|
|
...prev,
|
|
|
|
|
roleTags: nextRoleTags.length ? nextRoleTags : prev.roleTags,
|
|
|
|
|
primaryService: nextPrimary,
|
|
|
|
|
area: nextArea,
|
|
|
|
|
documents: nextDocs,
|
|
|
|
|
documentsCount: nextDocs.length || prev.documentsCount,
|
|
|
|
|
roleKey: prev.roleKey || String(submission?.role_key || ''),
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
} finally {
|
|
|
|
|
setLoadingLinkedDetails(false);
|
|
|
|
|
}
|
2026-03-27 05:35:18 +01:00
|
|
|
};
|
|
|
|
|
|
2026-03-30 21:43:49 +02:00
|
|
|
const requestDocumentsPersist = async (row: VerificationRecord, requested: string[], note: string) => {
|
|
|
|
|
if (!row.userId) return false;
|
|
|
|
|
const roleKey = String(row.roleKey || '').toUpperCase();
|
|
|
|
|
const payload = { documents: requested, note: note || 'Please upload the missing documents.' };
|
|
|
|
|
const endpoints: string[] = [];
|
|
|
|
|
if (roleKey && roleKey !== 'COMPANY' && roleKey !== 'CUSTOMER') {
|
|
|
|
|
endpoints.push(`${API}/api/admin/approvals/profiles/professional/${roleKey}/${row.userId}/request-documents`);
|
|
|
|
|
}
|
|
|
|
|
if (roleKey === 'COMPANY') endpoints.push(`${API}/api/admin/approvals/profiles/company/${row.userId}/request-documents`);
|
|
|
|
|
if (roleKey === 'CUSTOMER') endpoints.push(`${API}/api/admin/approvals/profiles/customer/${row.userId}/request-documents`);
|
|
|
|
|
endpoints.push(`${API}/api/admin/approvals/submission/${row.userId}/request-documents`);
|
|
|
|
|
for (const endpoint of endpoints) {
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(endpoint, {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: { 'Content-Type': 'application/json', Accept: 'application/json' },
|
|
|
|
|
body: JSON.stringify(payload),
|
|
|
|
|
});
|
|
|
|
|
if (res.ok) return true;
|
|
|
|
|
} catch {
|
|
|
|
|
// Try the next known endpoint shape.
|
|
|
|
|
}
|
2026-03-27 05:35:18 +01:00
|
|
|
}
|
2026-03-30 21:43:49 +02:00
|
|
|
return false;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const flagCase = (row: VerificationRecord) => {
|
|
|
|
|
setCaseStatus(row.id, 'FLAGGED');
|
|
|
|
|
setOpenMenuId(null);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const toggleRequestedDocument = (name: string) => {
|
|
|
|
|
setRequestingDocs((prev) => (
|
|
|
|
|
prev.includes(name) ? prev.filter((v) => v !== name) : [...prev, name]
|
|
|
|
|
));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const requestMissingDocuments = async (row: VerificationRecord) => {
|
|
|
|
|
const selected = requestingDocs();
|
|
|
|
|
if (!selected.length) return;
|
|
|
|
|
const persisted = await requestDocumentsPersist(row, selected, requestNote().trim());
|
|
|
|
|
requestReuploadCase(row, selected);
|
|
|
|
|
setActionNote(
|
|
|
|
|
persisted
|
|
|
|
|
? 'Document request sent successfully.'
|
|
|
|
|
: 'Document request updated in dashboard state. API endpoint for persistence was not available.',
|
|
|
|
|
);
|
|
|
|
|
setRequestNote('');
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const quickRequestDocuments = async (row: VerificationRecord, selected: string[]) => {
|
|
|
|
|
if (!selected.length) return;
|
|
|
|
|
const persisted = await requestDocumentsPersist(row, selected, '');
|
|
|
|
|
requestReuploadCase(row, selected);
|
|
|
|
|
setActionNote(
|
|
|
|
|
persisted
|
|
|
|
|
? 'Document request sent successfully.'
|
|
|
|
|
: 'Document request updated in dashboard state. API endpoint for persistence was not available.',
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const exportRows = () => {
|
|
|
|
|
const data = filteredRows();
|
|
|
|
|
const headers = ['Verification ID', 'Applicant', 'User Type', 'Verification Type', 'Documents', 'Priority', 'Status', 'Submitted Date'];
|
|
|
|
|
const lines = data.map((row) => [
|
|
|
|
|
row.id,
|
|
|
|
|
row.applicantName || '',
|
|
|
|
|
row.userType,
|
|
|
|
|
row.verificationType,
|
|
|
|
|
String(row.documentsCount || 0),
|
|
|
|
|
row.priority,
|
|
|
|
|
row.status,
|
|
|
|
|
formatDate(row.submittedDate || row.updatedAt),
|
2026-03-27 05:35:18 +01:00
|
|
|
]);
|
2026-03-30 21:43:49 +02:00
|
|
|
const csv = [headers, ...lines]
|
|
|
|
|
.map((line) => line.map((cell) => `"${String(cell).replace(/"/g, '""')}"`).join(','))
|
|
|
|
|
.join('\n');
|
|
|
|
|
const blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });
|
|
|
|
|
const url = URL.createObjectURL(blob);
|
|
|
|
|
const a = document.createElement('a');
|
|
|
|
|
a.href = url;
|
|
|
|
|
a.download = `verification-cases-${new Date().toISOString().slice(0, 10)}.csv`;
|
|
|
|
|
document.body.appendChild(a);
|
|
|
|
|
a.click();
|
|
|
|
|
document.body.removeChild(a);
|
|
|
|
|
URL.revokeObjectURL(url);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const openView = (row: VerificationRecord) => {
|
|
|
|
|
setActionNote('');
|
|
|
|
|
setViewingCase(row);
|
|
|
|
|
setDetailTab('overview');
|
|
|
|
|
setListTab('view');
|
|
|
|
|
setOpenMenuId(null);
|
|
|
|
|
setRequestNote('');
|
|
|
|
|
setRequestingDocs(row.requestedDocuments?.length ? row.requestedDocuments : (row.documents || []).filter((d) => d.state === 'MISSING').map((d) => d.name));
|
|
|
|
|
void enrichCaseFromSubmission(row);
|
2026-03-27 05:35:18 +01:00
|
|
|
};
|
|
|
|
|
|
2026-03-19 14:21:49 +01:00
|
|
|
return (
|
|
|
|
|
<AdminShell>
|
2026-03-27 05:35:18 +01:00
|
|
|
<div class="w-full space-y-6 pb-8">
|
|
|
|
|
|
|
|
|
|
{/* Page header */}
|
|
|
|
|
<div style="margin-bottom: 1.5rem">
|
|
|
|
|
<h1 class="text-[28px] font-bold leading-tight text-[#111827]">Verification Management</h1>
|
|
|
|
|
<p class="mt-1 text-[14px] text-[#6B7280]">Manage user identity and business verification workflows</p>
|
2026-03-19 14:21:49 +01:00
|
|
|
</div>
|
2026-03-30 04:48:09 +02:00
|
|
|
<Show when={error()}>
|
|
|
|
|
<div style="margin-bottom:10px;border-radius:10px;border:1px solid #FECACA;background:#FEF2F2;padding:12px 16px;font-size:13px;color:#B91C1C">
|
|
|
|
|
{error()}
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
2026-03-30 21:43:49 +02:00
|
|
|
<Show when={actionNote()}>
|
|
|
|
|
<div style="margin-bottom:10px;border-radius:10px;border:1px solid #E5E7EB;background:#F9FAFB;padding:12px 16px;font-size:13px;color:#374151">
|
|
|
|
|
{actionNote()}
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
2026-03-27 05:35:18 +01:00
|
|
|
|
|
|
|
|
{/* ── LIST VIEW ── */}
|
2026-03-30 21:43:49 +02:00
|
|
|
<Show when={true}>
|
|
|
|
|
<div style="display:grid;grid-template-columns:220px minmax(0,1fr);gap:20px;align-items:start">
|
|
|
|
|
<div style="position:sticky;top:12px;border:1px solid #E5E7EB;border-radius:12px;background:white;padding:8px;display:flex;flex-direction:column;gap:6px">
|
2026-03-27 05:35:18 +01:00
|
|
|
<button
|
|
|
|
|
type="button"
|
2026-03-30 21:43:49 +02:00
|
|
|
onClick={() => { setListTab('all'); setStatusFilter('all'); setViewingCase(null); }}
|
|
|
|
|
style={`display:flex;align-items:center;justify-content:space-between;height:38px;border-radius:8px;padding:0 12px;font-size:12.5px;font-weight:500;border:none;cursor:pointer;${listTab() === 'all' ? 'background:#FFF3EE;color:#FF5E13' : 'background:white;color:#6B7280'}`}
|
2026-03-27 05:35:18 +01:00
|
|
|
>
|
2026-03-30 21:43:49 +02:00
|
|
|
<span>Verification Queue</span>
|
|
|
|
|
<span style={`font-size:11px;font-weight:700;${listTab() === 'all' ? 'color:#FF5E13' : 'color:#9CA3AF'}`}>{rows().length}</span>
|
2026-03-27 05:35:18 +01:00
|
|
|
</button>
|
2026-03-30 21:43:49 +02:00
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
onClick={() => { setListTab('view'); }}
|
|
|
|
|
style={`display:flex;align-items:center;justify-content:space-between;height:38px;border-radius:8px;padding:0 12px;font-size:12.5px;font-weight:500;border:none;cursor:pointer;${listTab() === 'view' ? 'background:#FFF3EE;color:#FF5E13' : 'background:white;color:#6B7280'}`}
|
|
|
|
|
>
|
|
|
|
|
<span>View Verification</span>
|
|
|
|
|
</button>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
onClick={() => { setListTab('flagged'); setStatusFilter('flagged'); setViewingCase(null); }}
|
|
|
|
|
style={`display:flex;align-items:center;justify-content:space-between;height:38px;border-radius:8px;padding:0 12px;font-size:12.5px;font-weight:500;border:none;cursor:pointer;${listTab() === 'flagged' ? 'background:#FFF3EE;color:#FF5E13' : 'background:white;color:#6B7280'}`}
|
|
|
|
|
>
|
|
|
|
|
<span>Flagged Cases</span>
|
|
|
|
|
<span style={`font-size:11px;font-weight:700;${listTab() === 'flagged' ? 'color:#FF5E13' : 'color:#9CA3AF'}`}>{flaggedCount()}</span>
|
|
|
|
|
</button>
|
|
|
|
|
<div style="margin-top:4px;border-top:1px solid #F3F4F6;padding:8px 6px 2px 6px;font-size:11px;color:#9CA3AF">
|
|
|
|
|
Pending Review: <span style="font-weight:700;color:#6B7280">{reviewCount()}</span>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div>
|
2026-03-27 05:35:18 +01:00
|
|
|
|
|
|
|
|
<Show when={listTab() === 'view'}>
|
|
|
|
|
<Show when={!viewingCase()}>
|
|
|
|
|
<div style="margin-top:24px;border-radius:16px;border:1px solid #E5E7EB;background:white;padding:48px 24px;text-align:center">
|
|
|
|
|
<p style="font-size:15px;font-weight:600;color:#111827">No verification selected</p>
|
|
|
|
|
<p style="margin-top:6px;font-size:13px;color:#6B7280">Click the <strong>⋮</strong> menu on any row and choose <strong>View Verification</strong>.</p>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
<Show when={viewingCase()}>
|
|
|
|
|
<div style="margin-top:24px;border-radius:16px;border:1px solid #E5E7EB;background:white;box-shadow:0 1px 4px rgba(0,0,0,0.06);overflow:hidden">
|
|
|
|
|
<div style="padding:20px 24px;border-bottom:1px solid #E5E7EB;display:flex;align-items:center;justify-content:space-between">
|
|
|
|
|
<div>
|
|
|
|
|
<div style="display:flex;align-items:center;gap:12px">
|
|
|
|
|
<h2 style="font-size:18px;font-weight:700;color:#111827">{viewingCase()!.applicantName}</h2>
|
|
|
|
|
<StatusBadge status={viewingCase()!.status} />
|
|
|
|
|
<PriorityBadge priority={viewingCase()!.priority} />
|
|
|
|
|
</div>
|
|
|
|
|
<p style="margin-top:2px;font-size:13px;color:#6B7280">ID: {viewingCase()!.id} • {viewingCase()!.verificationType} • Submitted: {formatDate(viewingCase()!.submittedDate)}</p>
|
|
|
|
|
</div>
|
|
|
|
|
<div style="display:flex;gap:10px">
|
2026-03-30 21:43:49 +02:00
|
|
|
<button type="button" onClick={() => markVerifiedCase(viewingCase()!)} style="height:36px;border-radius:8px;background:#0D0D2A;padding:0 16px;font-size:13px;font-weight:600;color:white;border:none;cursor:pointer">Mark Verified</button>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
disabled={!requestingDocs().length}
|
|
|
|
|
onClick={() => requestMissingDocuments(viewingCase()!)}
|
|
|
|
|
style={`height:36px;border-radius:8px;border:1px solid #E5E7EB;background:white;padding:0 16px;font-size:13px;font-weight:600;color:#374151;cursor:${requestingDocs().length ? 'pointer' : 'not-allowed'};opacity:${requestingDocs().length ? 1 : 0.55}`}
|
|
|
|
|
>
|
|
|
|
|
Request Documents
|
|
|
|
|
</button>
|
2026-03-27 05:35:18 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div style="display:flex;align-items:center;gap:4px;border-bottom:1px solid #E5E7EB;padding:0 24px;background:#FAFAFA">
|
|
|
|
|
{(['overview', 'documents', 'checklist', 'logs'] as const).map((tab, i) => {
|
|
|
|
|
const labels = ['Overview', 'Documents', 'Checklist', 'Activity Logs'];
|
|
|
|
|
const active = () => detailTab() === tab;
|
|
|
|
|
return (
|
|
|
|
|
<button type="button" onClick={() => setDetailTab(tab)} style={`position:relative;padding:14px 12px;font-size:13px;font-weight:600;background:none;border:none;cursor:pointer;color:${active() ? '#FF5E13' : '#6B7280'}`}>
|
|
|
|
|
{labels[i]}
|
|
|
|
|
<Show when={active()}><span style="position:absolute;left:0;right:0;bottom:0;height:2px;background:#FF5E13;border-radius:2px 2px 0 0" /></Show>
|
|
|
|
|
</button>
|
|
|
|
|
);
|
|
|
|
|
})}
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div style="padding:24px">
|
|
|
|
|
<Show when={detailTab() === 'overview'}>
|
|
|
|
|
<div style="display:grid;grid-template-columns:2fr 1fr;gap:24px">
|
|
|
|
|
<div style="display:flex;flex-direction:column;gap:24px">
|
|
|
|
|
<div style="border:1px solid #E5E7EB;border-radius:12px;padding:20px">
|
|
|
|
|
<h3 style="font-size:14px;font-weight:700;color:#111827;margin-bottom:16px">Case Summary</h3>
|
2026-03-30 21:43:49 +02:00
|
|
|
<Show when={loadingLinkedDetails()}>
|
|
|
|
|
<p style="margin:0 0 12px;font-size:12px;color:#6B7280">Syncing linked submission details...</p>
|
|
|
|
|
</Show>
|
2026-03-27 05:35:18 +01:00
|
|
|
<div style="display:grid;grid-template-columns:1fr 1fr;gap:16px">
|
|
|
|
|
<div><p style="font-size:11px;color:#9CA3AF">Applicant Name</p><p style="font-size:14px;font-weight:600;color:#111827">{viewingCase()!.applicantName}</p></div>
|
|
|
|
|
<div><p style="font-size:11px;color:#9CA3AF">User Type</p><p style="font-size:14px;font-weight:600;color:#111827">{viewingCase()!.userType}</p></div>
|
2026-03-30 21:43:49 +02:00
|
|
|
<div><p style="font-size:11px;color:#9CA3AF">Primary Service</p><p style="font-size:14px;font-weight:600;color:#111827">{viewingCase()!.primaryService || '—'}</p></div>
|
|
|
|
|
<div><p style="font-size:11px;color:#9CA3AF">Area / Place</p><p style="font-size:14px;font-weight:600;color:#111827">{viewingCase()!.area || '—'}</p></div>
|
2026-03-27 05:35:18 +01:00
|
|
|
<div><p style="font-size:11px;color:#9CA3AF">Assigned Verifier</p><p style="font-size:14px;font-weight:600;color:#111827">{viewingCase()!.assignedVerifier}</p></div>
|
|
|
|
|
<div><p style="font-size:11px;color:#9CA3AF">Documents</p><p style="font-size:14px;font-weight:600;color:#111827">{Number(viewingCase()!.documentsCount || 0)}</p></div>
|
|
|
|
|
</div>
|
2026-03-30 21:43:49 +02:00
|
|
|
<Show when={viewingCase()!.roleTags?.length}>
|
|
|
|
|
<div style="margin-top:12px">
|
|
|
|
|
<p style="margin:0 0 8px;font-size:11px;color:#9CA3AF">Registered Roles / Services</p>
|
|
|
|
|
<div style="display:flex;flex-wrap:wrap;gap:6px">
|
|
|
|
|
<For each={viewingCase()!.roleTags || []}>
|
|
|
|
|
{(tag) => (
|
|
|
|
|
<span style="height:24px;padding:0 10px;border-radius:999px;border:1px solid #E5E7EB;background:#F9FAFB;display:inline-flex;align-items:center;font-size:11px;font-weight:600;color:#374151">{tag}</span>
|
|
|
|
|
)}
|
|
|
|
|
</For>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</div>
|
|
|
|
|
<div style="border:1px solid #FFE2D3;border-radius:12px;padding:20px;background:#FFF8F4">
|
|
|
|
|
<h3 style="font-size:14px;font-weight:700;color:#111827;margin-bottom:12px">Request Missing Documents</h3>
|
|
|
|
|
<p style="margin:0 0 10px;font-size:12px;color:#6B7280">Select missing documents and send one clear request to the user.</p>
|
|
|
|
|
<div style="display:grid;grid-template-columns:1fr 1fr;gap:10px">
|
|
|
|
|
<For each={viewingCase()!.documents || []}>
|
|
|
|
|
{(doc) => (
|
|
|
|
|
<label style="display:flex;align-items:center;gap:8px;border:1px solid #E5E7EB;background:white;border-radius:10px;padding:8px 10px;cursor:pointer">
|
|
|
|
|
<input type="checkbox" checked={requestingDocs().includes(doc.name)} onChange={() => toggleRequestedDocument(doc.name)} style="width:14px;height:14px;accent-color:#FF5E13" />
|
|
|
|
|
<span style="font-size:12px;color:#111827;font-weight:600">{doc.name}</span>
|
|
|
|
|
</label>
|
|
|
|
|
)}
|
|
|
|
|
</For>
|
|
|
|
|
</div>
|
|
|
|
|
<textarea
|
|
|
|
|
value={requestNote()}
|
|
|
|
|
onInput={(e) => setRequestNote(e.currentTarget.value)}
|
|
|
|
|
placeholder="Optional note for user (e.g. upload clear PDF/JPG, full document visible)..."
|
|
|
|
|
style="margin-top:10px;width:100%;height:72px;border-radius:8px;border:1px solid #E5E7EB;padding:10px;font-size:12px;resize:none"
|
|
|
|
|
/>
|
|
|
|
|
<div style="margin-top:10px;display:flex;justify-content:flex-end">
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
onClick={() => requestMissingDocuments(viewingCase()!)}
|
|
|
|
|
disabled={!requestingDocs().length}
|
|
|
|
|
style={`height:34px;border-radius:8px;border:none;background:#0D0D2A;color:white;padding:0 14px;font-size:12px;font-weight:700;cursor:${requestingDocs().length ? 'pointer' : 'not-allowed'};opacity:${requestingDocs().length ? 1 : 0.55}`}
|
|
|
|
|
>
|
|
|
|
|
Send Document Request
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
2026-03-27 05:35:18 +01:00
|
|
|
</div>
|
|
|
|
|
<div style="border:1px solid #E5E7EB;border-radius:12px;padding:20px">
|
|
|
|
|
<h3 style="font-size:14px;font-weight:700;color:#111827;margin-bottom:16px">Verification Tracker</h3>
|
|
|
|
|
<div style="display:flex;justify-content:space-between;align-items:center;position:relative">
|
|
|
|
|
<div style="position:absolute;top:10px;left:0;right:0;height:2px;background:#E5E7EB;z-index:1" />
|
|
|
|
|
<div style="position:absolute;top:10px;left:0;width:50%;height:2px;background:#FF5E13;z-index:2" />
|
|
|
|
|
{[
|
|
|
|
|
{ l: 'Submitted', active: true },
|
|
|
|
|
{ l: 'In Review', active: true },
|
|
|
|
|
{ l: 'Verified', active: false },
|
|
|
|
|
].map((step) => (
|
|
|
|
|
<div style="position:relative;z-index:3;text-align:center">
|
|
|
|
|
<div style={`width:20px;height:20px;border-radius:50%;background:${step.active ? '#FF5E13' : 'white'};border:2px solid ${step.active ? '#FF5E13' : '#E5E7EB'};margin:0 auto`} />
|
|
|
|
|
<p style="font-size:11px;margin-top:4px;color:#111827;font-weight:600">{step.l}</p>
|
|
|
|
|
</div>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div style="border:1px solid #E5E7EB;border-radius:12px;padding:20px;background:#F9FAFB">
|
|
|
|
|
<h3 style="font-size:14px;font-weight:700;color:#111827;margin-bottom:16px">Notes</h3>
|
|
|
|
|
<textarea placeholder="Add internal note..." style="width:100%;height:100px;border-radius:8px;border:1px solid #E5E7EB;padding:10px;font-size:13px;resize:none;margin-bottom:12px" />
|
|
|
|
|
<button type="button" style="width:100%;height:34px;background:#0D0D2A;color:white;border-radius:8px;font-size:12px;font-weight:600;border:none">Add Note</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
|
|
|
|
|
<Show when={detailTab() === 'documents'}>
|
|
|
|
|
<div style="border:1px solid #E5E7EB;border-radius:12px;overflow:hidden">
|
|
|
|
|
<table style="width:100%;border-collapse:collapse">
|
|
|
|
|
<thead style="background:#F9FAFB">
|
|
|
|
|
<tr style="text-align:left">
|
|
|
|
|
<th style="padding:12px 16px;font-size:11px;font-weight:600;color:#6B7280;text-transform:uppercase">Document Name</th>
|
|
|
|
|
<th style="padding:12px 16px;font-size:11px;font-weight:600;color:#6B7280;text-transform:uppercase">Status</th>
|
|
|
|
|
<th style="padding:12px 16px;font-size:11px;font-weight:600;color:#6B7280;text-transform:uppercase">Actions</th>
|
|
|
|
|
</tr>
|
|
|
|
|
</thead>
|
|
|
|
|
<tbody>
|
2026-03-30 21:43:49 +02:00
|
|
|
<For each={viewingCase()!.documents || []}>
|
2026-03-27 05:35:18 +01:00
|
|
|
{(doc) => (
|
|
|
|
|
<tr style="border-top:1px solid #E5E7EB">
|
2026-03-30 21:43:49 +02:00
|
|
|
<td style="padding:12px 16px">
|
|
|
|
|
<p style="margin:0;font-size:13px;font-weight:600;color:#111827">{doc.name}</p>
|
|
|
|
|
<p style="margin:2px 0 0;font-size:11px;color:#6B7280">{doc.fileName}</p>
|
|
|
|
|
</td>
|
|
|
|
|
<td style="padding:12px 16px"><StatusBadge status={doc.state === 'APPROVED' ? 'VERIFIED' : doc.state === 'REQUESTED' ? 'RE_UPLOAD_REQUESTED' : doc.state === 'MISSING' ? 'PENDING' : 'IN_REVIEW'} /></td>
|
2026-03-27 05:35:18 +01:00
|
|
|
<td style="padding:12px 16px;display:flex;gap:8px">
|
|
|
|
|
<button type="button" style="font-size:12px;color:#FF5E13;background:none;border:none;cursor:pointer;font-weight:600">Preview</button>
|
2026-03-30 21:43:49 +02:00
|
|
|
<Show when={doc.state === 'MISSING' || doc.state === 'REQUESTED'}>
|
|
|
|
|
<button type="button" onClick={() => toggleRequestedDocument(doc.name)} style="font-size:12px;color:#0D0D2A;background:none;border:none;cursor:pointer;font-weight:600">
|
|
|
|
|
{requestingDocs().includes(doc.name) ? 'Unselect' : 'Request'}
|
|
|
|
|
</button>
|
|
|
|
|
</Show>
|
2026-03-27 05:35:18 +01:00
|
|
|
</td>
|
|
|
|
|
</tr>
|
|
|
|
|
)}
|
|
|
|
|
</For>
|
|
|
|
|
</tbody>
|
|
|
|
|
</table>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
|
|
|
|
|
<Show when={detailTab() === 'checklist'}>
|
|
|
|
|
<div style="display:flex;flex-direction:column;gap:12px">
|
|
|
|
|
{[
|
|
|
|
|
'Identity matches provided documents',
|
|
|
|
|
'Address proof is valid and recent',
|
|
|
|
|
'Business registration is authentic',
|
|
|
|
|
'Contact information is verified',
|
|
|
|
|
].map((item) => (
|
|
|
|
|
<label style="display:flex;align-items:center;gap:12px;padding:12px;border:1px solid #E5E7EB;border-radius:10px;cursor:pointer">
|
|
|
|
|
<input type="checkbox" style="width:16px;height:16px;accent-color:#FF5E13" />
|
|
|
|
|
<span style="font-size:13px;color:#111827;font-weight:500">{item}</span>
|
|
|
|
|
</label>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
|
|
|
|
|
<Show when={detailTab() === 'logs'}>
|
|
|
|
|
<div style="display:flex;flex-direction:column;gap:16px">
|
|
|
|
|
<div style="display:flex;gap:12px">
|
|
|
|
|
<div style="width:8px;height:8px;border-radius:50%;background:#FF5E13;margin-top:4px" />
|
|
|
|
|
<div>
|
|
|
|
|
<p style="font-size:13px;font-weight:600;color:#111827">Case Review Started</p>
|
|
|
|
|
<p style="font-size:12px;color:#6B7280">Verifier started reviewing documents • 2 hours ago</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div style="display:flex;gap:12px">
|
|
|
|
|
<div style="width:8px;height:8px;border-radius:50%;background:#E5E7EB;margin-top:4px" />
|
|
|
|
|
<div>
|
|
|
|
|
<p style="font-size:13px;font-weight:600;color:#111827">Verification Request Submitted</p>
|
|
|
|
|
<p style="font-size:12px;color:#6B7280">System received applicant data • 1 day ago</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div style="display:flex;align-items:center;gap:10px;padding:14px 24px;border-top:1px solid #E5E7EB">
|
|
|
|
|
<button type="button" onClick={() => { setViewingCase(null); setListTab('all'); }} style="height:36px;border-radius:8px;border:1px solid #E5E7EB;background:white;padding:0 16px;font-size:13px;font-weight:600;color:#374151;cursor:pointer">Back to List</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</Show>
|
|
|
|
|
|
|
|
|
|
<div style={`display:${listTab() === 'view' ? 'none' : 'block'}`}>
|
|
|
|
|
<div style="margin-top:1.5rem;margin-left:-24px;margin-right:-24px;border-radius:0;border-left:none;border-right:none;overflow:hidden;border-top:1px solid #E5E7EB;border-bottom:1px solid #E5E7EB;background:white;box-shadow:0 1px 3px rgba(0,0,0,0.06)">
|
|
|
|
|
<div style="display:flex;align-items:center;gap:8px;padding:14px 20px;border-bottom:1px solid #F3F4F6">
|
|
|
|
|
<input
|
|
|
|
|
value={search()}
|
|
|
|
|
onInput={(e) => setSearch(e.currentTarget.value)}
|
|
|
|
|
placeholder="Search verifications..."
|
|
|
|
|
style="height:34px;flex:1;border-radius:8px;border:1px solid #E5E7EB;background:white;padding:0 12px;font-size:13px;color:#111827;outline:none"
|
|
|
|
|
/>
|
|
|
|
|
<div style="position:relative">
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
onClick={() => { setSortMenuOpen((v) => !v); setFilterMenuOpen(false); }}
|
|
|
|
|
style="display:inline-flex;height:34px;align-items:center;gap:6px;border-radius:8px;border:1px solid #E5E7EB;background:white;padding:0 12px;font-size:12px;font-weight:500;color:#374151;cursor:pointer"
|
|
|
|
|
>
|
|
|
|
|
<svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M7 4v13"/><path d="m3 13 4 4 4-4"/><path d="M17 20V7"/><path d="m21 11-4-4-4 4"/></svg>
|
|
|
|
|
Sort
|
|
|
|
|
</button>
|
|
|
|
|
<Show when={sortMenuOpen()}>
|
|
|
|
|
<div style="position:absolute;left:0;top:38px;z-index:30;min-width:220px;border-radius:12px;border:1px solid #E5E7EB;background:white;padding:6px;box-shadow:0 4px 16px rgba(0,0,0,0.1)">
|
|
|
|
|
{(['submitted_desc', 'submitted_asc', 'priority_desc', 'priority_asc'] as const).map((s, i) => (
|
|
|
|
|
<button type="button" onClick={() => { setSortBy(s); setSortMenuOpen(false); }} style={`display:block;width:100%;border-radius:8px;padding:8px 12px;text-align:left;font-size:13px;background:none;border:none;cursor:pointer;color:${sortBy() === s ? '#FF5E13' : '#374151'};background:${sortBy() === s ? '#FFF1EB' : 'transparent'}`}>
|
|
|
|
|
{['Submitted (Newest)', 'Submitted (Oldest)', 'Priority (High-Low)', 'Priority (Low-High)'][i]}
|
|
|
|
|
</button>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</div>
|
|
|
|
|
<div style="position:relative">
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
onClick={() => { setFilterMenuOpen((v) => !v); setSortMenuOpen(false); }}
|
|
|
|
|
style="display:inline-flex;height:34px;align-items:center;gap:6px;border-radius:8px;border:1px solid #E5E7EB;background:white;padding:0 12px;font-size:12px;font-weight:500;color:#374151;cursor:pointer"
|
|
|
|
|
>
|
|
|
|
|
<svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M3 5h18M6 12h12M10 19h4"/></svg>
|
|
|
|
|
Filters
|
|
|
|
|
</button>
|
|
|
|
|
<Show when={filterMenuOpen()}>
|
|
|
|
|
<div style="position:absolute;left:0;top:38px;z-index:30;min-width:200px;border-radius:12px;border:1px solid #E5E7EB;background:white;padding:6px;box-shadow:0 4px 16px rgba(0,0,0,0.1)">
|
|
|
|
|
{(['all', 'pending', 'flagged'] as const).map((s) => (
|
|
|
|
|
<button type="button" onClick={() => { setStatusFilter(s); setFilterMenuOpen(false); }} style={`display:block;width:100%;border-radius:8px;padding:8px 12px;text-align:left;font-size:13px;background:none;border:none;cursor:pointer;color:${statusFilter() === s ? '#FF5E13' : '#374151'};background:${statusFilter() === s ? '#FFF1EB' : 'transparent'}`}>
|
|
|
|
|
{s === 'all' ? 'All Cases' : s === 'pending' ? 'Pending Review' : 'Flagged'}
|
|
|
|
|
</button>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</div>
|
2026-03-30 21:43:49 +02:00
|
|
|
<button type="button" onClick={exportRows} style="display:inline-flex;height:34px;align-items:center;gap:6px;border-radius:8px;background:#0D0D2A;padding:0 12px;font-size:12px;font-weight:600;color:white;border:none;cursor:pointer">
|
2026-03-27 05:35:18 +01:00
|
|
|
<svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="7 10 12 15 17 10"/><line x1="12" y1="15" x2="12" y2="3"/></svg>
|
|
|
|
|
Export
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="overflow-x-auto">
|
|
|
|
|
<table class="min-w-full">
|
|
|
|
|
<thead>
|
|
|
|
|
<tr style="background:#0D0D2A;text-align:left">
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Verification ID</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Applicant</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Type</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Docs</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Priority</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Status</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Submitted</th>
|
|
|
|
|
<th style="padding:10px 20px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:0.05em;color:#FFFFFF;white-space:nowrap">Actions</th>
|
|
|
|
|
</tr>
|
|
|
|
|
</thead>
|
|
|
|
|
<tbody>
|
|
|
|
|
<For each={filteredRows()}>
|
|
|
|
|
{(row) => (
|
|
|
|
|
<tr style="border-bottom:1px solid #F3F4F6" class="hover:bg-[#FAFAFA] transition-colors">
|
|
|
|
|
<td style="padding:12px 20px;font-size:12px;font-family:monospace;color:#6B7280">{row.id}</td>
|
|
|
|
|
<td style="padding:12px 20px">
|
|
|
|
|
<p style="font-size:14px;font-weight:600;color:#111827">{row.applicantName}</p>
|
2026-03-30 21:43:49 +02:00
|
|
|
<p style="font-size:11px;color:#6B7280">{row.userType}{row.area ? ` • ${row.area}` : ''}</p>
|
|
|
|
|
<Show when={row.roleTags?.length}>
|
|
|
|
|
<p style="margin-top:2px;font-size:11px;color:#9CA3AF">{(row.roleTags || []).join(', ')}</p>
|
|
|
|
|
</Show>
|
2026-03-27 05:35:18 +01:00
|
|
|
</td>
|
|
|
|
|
<td style="padding:12px 20px;font-size:13px;color:#6B7280">{row.verificationType}</td>
|
|
|
|
|
<td style="padding:12px 20px;font-size:13px;color:#6B7280">{Number(row.documentsCount || 0)} docs</td>
|
|
|
|
|
<td style="padding:12px 20px"><PriorityBadge priority={row.priority} /></td>
|
|
|
|
|
<td style="padding:12px 20px"><StatusBadge status={row.status} /></td>
|
|
|
|
|
<td style="padding:12px 20px;font-size:13px;color:#6B7280">{formatDate(row.submittedDate || row.updatedAt)}</td>
|
|
|
|
|
<td style="padding:12px 20px;position:relative">
|
|
|
|
|
<button type="button" onClick={() => setOpenMenuId(openMenuId() === row.id ? null : row.id)} style="display:inline-flex;height:32px;width:32px;align-items:center;justify-content:center;border-radius:8px;color:#9CA3AF;background:none;border:none;cursor:pointer">
|
|
|
|
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="1.5"/><circle cx="12" cy="12" r="1.5"/><circle cx="12" cy="19" r="1.5"/></svg>
|
|
|
|
|
</button>
|
|
|
|
|
<Show when={openMenuId() === row.id}>
|
|
|
|
|
<div style="position:absolute;right:20px;top:44px;z-index:20;width:190px;border-radius:12px;border:1px solid #E5E7EB;background:white;padding:6px;box-shadow:0 4px 20px rgba(0,0,0,0.12)">
|
|
|
|
|
<button type="button" onClick={() => openView(row)} style="display:flex;width:100%;align-items:center;gap:10px;border-radius:8px;padding:8px 12px;font-size:13px;color:#374151;background:none;border:none;cursor:pointer;text-align:left">View Verification</button>
|
2026-03-30 21:43:49 +02:00
|
|
|
<button type="button" onClick={() => void quickRequestDocuments(row, (row.documents || []).filter((d) => d.state === 'MISSING' || d.state === 'REQUESTED').map((d) => d.name))} style="display:flex;width:100%;align-items:center;gap:10px;border-radius:8px;padding:8px 12px;font-size:13px;color:#374151;background:none;border:none;cursor:pointer;text-align:left">Request Documents</button>
|
|
|
|
|
<button type="button" onClick={() => markVerifiedCase(row)} style="display:flex;width:100%;align-items:center;gap:10px;border-radius:8px;padding:8px 12px;font-size:13px;color:#374151;background:none;border:none;cursor:pointer;text-align:left">Mark Verified</button>
|
|
|
|
|
<button type="button" onClick={() => flagCase(row)} style="display:flex;width:100%;align-items:center;gap:10px;border-radius:8px;padding:8px 12px;font-size:13px;color:#DC2626;background:none;border:none;cursor:pointer;text-align:left">Flag Case</button>
|
2026-03-27 05:35:18 +01:00
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</td>
|
|
|
|
|
</tr>
|
|
|
|
|
)}
|
|
|
|
|
</For>
|
|
|
|
|
</tbody>
|
|
|
|
|
</table>
|
|
|
|
|
</div>
|
ui(step-4): apply reference layout to remaining 7 pages
- catering-services, fitness-trainers, graphic-designers, social-media-managers,
video-editors, verification, kb: white header, data-table/table-card,
navy buttons, orange tab underlines, inline styles removed
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-24 05:23:57 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
2026-03-27 05:35:18 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
|
ui(step-4): apply reference layout to remaining 7 pages
- catering-services, fitness-trainers, graphic-designers, social-media-managers,
video-editors, verification, kb: white header, data-table/table-card,
navy buttons, orange tab underlines, inline styles removed
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-24 05:23:57 +01:00
|
|
|
</div>
|
2026-03-19 14:21:49 +01:00
|
|
|
</AdminShell>
|
|
|
|
|
);
|
|
|
|
|
}
|