import { For, Show, createMemo, createSignal, onMount } from 'solid-js'; import AdminShell from '~/components/AdminShell'; 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; 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 ( {label} ); } function PriorityBadge(props: { priority: string }) { const color = props.priority === 'HIGH' ? '#DC2626' : props.priority === 'MEDIUM' ? '#F59E0B' : '#16A34A'; return ( {props.priority} ); } export default function VerificationManagementPage() { const [view, setView] = createSignal<'list' | 'form'>('list'); const [listTab, setListTab] = createSignal<'all' | 'create' | 'view'>('all'); const [detailTab, setDetailTab] = createSignal<'overview' | 'documents' | 'checklist' | 'logs'>('overview'); const [search, setSearch] = createSignal(''); const [rows, setRows] = createSignal([]); const [viewingCase, setViewingCase] = createSignal(null); const [openMenuId, setOpenMenuId] = createSignal(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); const [ruleName, setRuleName] = createSignal(''); const [ruleUserType, setRuleUserType] = createSignal('PROFESSIONAL'); const [ruleVerificationType, setRuleVerificationType] = createSignal('IDENTITY'); const [ruleActive, setRuleActive] = createSignal(true); const [formError, setFormError] = createSignal(''); const [error, setError] = createSignal(''); type VerificationRule = { id: string; name: string; userType: VerificationRecord['userType']; verificationType: VerificationRecord['verificationType']; status: 'ACTIVE' | 'INACTIVE'; updatedAt: string; }; const [rules, setRules] = createSignal([ { id: 'vr1', name: 'Professional Identity Rule', userType: 'PROFESSIONAL', verificationType: 'IDENTITY', status: 'ACTIVE', updatedAt: '2026-03-20' }, { id: 'vr2', name: 'Company Business Verification', userType: 'COMPANY', verificationType: 'BUSINESS', status: 'ACTIVE', updatedAt: '2026-03-18' }, { id: 'vr3', name: 'Jobseeker Profile Check', userType: 'JOBSEEKER', verificationType: 'PROFILE', status: 'INACTIVE', updatedAt: '2026-03-12' }, ]); const load = async () => { 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 : []; const jobCases: VerificationRecord[] = jobs.map((job: any) => ({ 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: 1, assignedVerifier: 'Unassigned', priority: 'HIGH', status: 'IN_REVIEW', updatedAt: String(job.updated_at || job.created_at || ''), })); const requirementCases: VerificationRecord[] = requirements.map((req: any) => ({ 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: 1, assignedVerifier: 'Unassigned', priority: 'MEDIUM', status: 'PENDING', updatedAt: String(req.updated_at || req.created_at || ''), })); setRows([...jobCases, ...requirementCases]); } catch (e: any) { setRows([]); setError(e?.message || 'Could not reach verification API.'); } }; 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; }); const openView = (row: VerificationRecord) => { setViewingCase(row); setDetailTab('overview'); setListTab('view'); setOpenMenuId(null); }; const resetRuleForm = () => { setRuleName(''); setRuleUserType('PROFESSIONAL'); setRuleVerificationType('IDENTITY'); setRuleActive(true); setFormError(''); }; const openCreate = () => { resetRuleForm(); setListTab('create'); setView('form'); }; const saveRule = () => { if (!ruleName().trim()) { setFormError('Rule name is required.'); return; } const now = new Date().toISOString().slice(0, 10); const id = `vr_${Math.random().toString(16).slice(2)}`; setRules((prev) => [ { id, name: ruleName().trim(), userType: ruleUserType(), verificationType: ruleVerificationType(), status: ruleActive() ? 'ACTIVE' : 'INACTIVE', updatedAt: now }, ...prev, ]); setView('list'); setListTab('all'); resetRuleForm(); }; return (
{/* Page header */}

Verification Management

Manage user identity and business verification workflows

{error()}
{/* ── LIST VIEW ── */}
{([ { key: 'all', label: 'All Verifications', action: () => { setListTab('all'); setStatusFilter('all'); } }, { key: 'create', label: 'Create Rule', action: () => openCreate() }, { key: 'view', label: 'View Verification', action: () => { setListTab('view'); } }, ] as const).map((tab) => ( ))}

No verification selected

Click the menu on any row and choose View Verification.

{viewingCase()!.applicantName}

ID: {viewingCase()!.id} • {viewingCase()!.verificationType} • Submitted: {formatDate(viewingCase()!.submittedDate)}

{(['overview', 'documents', 'checklist', 'logs'] as const).map((tab, i) => { const labels = ['Overview', 'Documents', 'Checklist', 'Activity Logs']; const active = () => detailTab() === tab; return ( ); })}

Case Summary

Applicant Name

{viewingCase()!.applicantName}

User Type

{viewingCase()!.userType}

Assigned Verifier

{viewingCase()!.assignedVerifier}

Documents

{Number(viewingCase()!.documentsCount || 0)}

Verification Tracker

{[ { l: 'Submitted', active: true }, { l: 'In Review', active: true }, { l: 'Verified', active: false }, ].map((step) => (

{step.l}

))}

Notes