import { useState, useEffect, useRef, useCallback } from 'react'; import { useToast } from '../contexts/ToastContext.jsx'; import { api } from '../utils/api.js'; import Avatar from '../components/Avatar.jsx'; import UserFooter from '../components/UserFooter.jsx'; import PasswordInput from '../components/PasswordInput.jsx'; const SIDEBAR_W = 320; function isValidEmail(e) { return /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(e); } function isValidPhone(p) { if (!p || !p.trim()) return true; const digits = p.replace(/[\s\-\(\)\+\.x#]/g, ''); return /^\d{7,15}$/.test(digits); } function parseCSV(text) { const lines = text.split(/\r?\n/).map(l => l.trim()).filter(Boolean); const rows = [], invalid = []; for (let i = 0; i < lines.length; i++) { const line = lines[i]; if (i === 0 && /^name\s*,/i.test(line)) continue; const parts = line.split(',').map(p => p.trim()); if (parts.length < 2 || parts.length > 4) { invalid.push({ line, reason: 'Must have 2–4 comma-separated fields' }); continue; } const [name, email, password, role] = parts; if (!name || !/\S+\s+\S+/.test(name)) { invalid.push({ line, reason: 'Name must be two words (First Last)' }); continue; } if (!email || !isValidEmail(email)) { invalid.push({ line, reason: `Invalid email: "${email}"` }); continue; } rows.push({ name: name.trim(), email: email.trim().toLowerCase(), password: (password || '').trim(), role: (role || 'member').trim().toLowerCase() }); } return { rows, invalid }; } // ── User Row (accordion list item) ─────────────────────────────────────────── function UserRow({ u, onUpdated, onEdit }) { const toast = useToast(); const [open, setOpen] = useState(false); const handleSuspend = async () => { if (!confirm(`Suspend ${u.name}?`)) return; try { await api.suspendUser(u.id); toast('User suspended', 'success'); onUpdated(); } catch (e) { toast(e.message, 'error'); } }; const handleActivate = async () => { try { await api.activateUser(u.id); toast('User activated', 'success'); onUpdated(); } catch (e) { toast(e.message, 'error'); } }; const handleDelete = async () => { if (u.role === 'admin') return toast('Demote to member before deleting an admin', 'error'); if (!confirm(`Delete ${u.name}?\n\nThis will:\n• Anonymise their account and free their email for re-use\n• Remove all their messages from conversations\n• Freeze any direct messages they were part of\n• Remove all their group memberships\n\nThis cannot be undone.`)) return; try { await api.deleteUser(u.id); toast('User deleted', 'success'); onUpdated(); } catch (e) { toast(e.message, 'error'); } }; return (
{open && !u.is_default_admin && (
{u.status === 'active' ? ( ) : u.status === 'suspended' ? ( ) : null}
)}
); } // ── User Form (create / edit) ───────────────────────────────────────────────── function UserForm({ user, userPass, onDone, onCancel, isMobile, onIF, onIB }) { const toast = useToast(); const isEdit = !!user; const [firstName, setFirstName] = useState(user?.first_name || ''); const [lastName, setLastName] = useState(user?.last_name || ''); const [email, setEmail] = useState(user?.email || ''); const [phone, setPhone] = useState(user?.phone || ''); const [role, setRole] = useState(user?.role || 'member'); const [isMinor, setIsMinor] = useState(!!user?.is_minor); const [password, setPassword] = useState(''); const [pwEnabled, setPwEnabled] = useState(!isEdit); const [saving, setSaving] = useState(false); const fmtLastLogin = (ts) => { if (!ts) return 'Never'; const d = new Date(ts); const today = new Date(); today.setHours(0,0,0,0); const yesterday = new Date(today); yesterday.setDate(yesterday.getDate() - 1); const dd = new Date(d); dd.setHours(0,0,0,0); if (dd >= today) return 'Today'; if (dd >= yesterday) return 'Yesterday'; return dd.toISOString().slice(0, 10); }; const handleSubmit = async () => { if (!isEdit && (!email.trim() || !isValidEmail(email.trim()))) return toast('Valid email address required', 'error'); if (!firstName.trim()) return toast('First name is required', 'error'); if (!lastName.trim()) return toast('Last name is required', 'error'); if (!isValidPhone(phone)) return toast('Invalid phone number', 'error'); if (!['member', 'admin', 'manager'].includes(role)) return toast('Role is required', 'error'); if (!isEdit && (!password || password.length < 6)) return toast('Password must be at least 6 characters', 'error'); if (isEdit && pwEnabled && (!password || password.length < 6)) return toast('New password must be at least 6 characters', 'error'); setSaving(true); try { if (isEdit) { await api.updateUser(user.id, { firstName: firstName.trim(), lastName: lastName.trim(), phone: phone.trim(), isMinor, role, ...(pwEnabled && password ? { password } : {}), }); toast('User updated', 'success'); } else { await api.createUser({ firstName: firstName.trim(), lastName: lastName.trim(), email: email.trim(), phone: phone.trim(), isMinor, role, password, }); toast('User created', 'success'); } onDone(); } catch (e) { toast(e.message, 'error'); } finally { setSaving(false); } }; const colGrid = isMobile ? '1fr' : '1fr 1fr'; const lbl = (text, required, note) => ( ); return (
{/* Back + title */}
{isEdit ? 'Edit User' : 'Create User'}
{/* Row 1: Login (email) — full width */}
{lbl('Login (email)', !isEdit)} setEmail(e.target.value)} disabled={isEdit} style={{ width:'100%', ...(isEdit ? { opacity:0.6, cursor:'not-allowed' } : {}) }} autoComplete="new-password" onFocus={onIF} onBlur={onIB} />
{/* Row 2: First Name + Last Name */}
{lbl('First Name', true)} setFirstName(e.target.value)} autoComplete="new-password" autoCapitalize="words" onFocus={onIF} onBlur={onIB} />
{lbl('Last Name', true)} setLastName(e.target.value)} autoComplete="new-password" autoCapitalize="words" onFocus={onIF} onBlur={onIB} />
{/* Row 3: Phone + Role */}
{lbl('Phone', false, '(optional)')} setPhone(e.target.value)} autoComplete="new-password" onFocus={onIF} onBlur={onIB} />
{lbl('App Role', true)}
{/* Row 4: Is minor */}
{/* Row 5: Password */}
{lbl('Password', (!isEdit) || (isEdit && pwEnabled), isEdit && !pwEnabled ? '(not changing — click Reset Password to set a new one)' : !isEdit ? `(blank = ${userPass})` : null )}
setPassword(e.target.value)} placeholder={isEdit && !pwEnabled ? '••••••••' : 'Min 6 characters'} disabled={!pwEnabled} autoComplete="new-password" onFocus={onIF} onBlur={onIB} />
{/* Row 6: Buttons */}
{isEdit && !pwEnabled && ( )} {isEdit && pwEnabled && ( )}
{/* Row 7 (edit only): Last login + must change password */} {isEdit && (
Last Login: {fmtLastLogin(user.last_online)} {!!user.must_change_password && ( ⚠ Must change password )}
)}
); } // ── Bulk Import Form ────────────────────────────────────────────────────────── function BulkImportForm({ userPass, onCreated }) { const toast = useToast(); const fileRef = useRef(null); const [csvFile, setCsvFile] = useState(null); const [csvRows, setCsvRows] = useState([]); const [csvInvalid, setCsvInvalid] = useState([]); const [bulkResult, setBulkResult] = useState(null); const [loading, setLoading] = useState(false); const handleFile = e => { const file = e.target.files?.[0]; if (!file) return; setCsvFile(file); setBulkResult(null); const reader = new FileReader(); reader.onload = ev => { const { rows, invalid } = parseCSV(ev.target.result); setCsvRows(rows); setCsvInvalid(invalid); }; reader.readAsText(file); }; const handleImport = async () => { if (!csvRows.length) return; setLoading(true); try { const result = await api.bulkUsers(csvRows); setBulkResult(result); setCsvRows([]); setCsvFile(null); setCsvInvalid([]); if (fileRef.current) fileRef.current.value = ''; onCreated(); } catch(e) { toast(e.message, 'error'); } finally { setLoading(false); } }; return (

CSV Format

{"name,email,password,role\nJane Smith,jane@company.com,,member\nBob Jones,bob@company.com,TempPass1,admin"}

Name and email required. Blank password defaults to {userPass}, blank role defaults to member.

{csvFile && {csvFile.name}{csvRows.length > 0 && ({csvRows.length} valid)}} {csvRows.length > 0 && }
{csvInvalid.length > 0 && (

{csvInvalid.length} line{csvInvalid.length!==1?'s':''} skipped

{csvInvalid.map((e,i) =>
{e.line}— {e.reason}
)}
)} {bulkResult && (

✓ {bulkResult.created.length} user{bulkResult.created.length!==1?'s':''} created

{bulkResult.skipped.length > 0 && ( <>

{bulkResult.skipped.length} skipped:

{bulkResult.skipped.map((s,i) => (
{s.email}{s.reason}
))}
)}
)}
); } // ── Main page ───────────────────────────────────────────────────────────────── export default function UserManagerPage({ isMobile = false, onProfile, onHelp, onAbout }) { const [users, setUsers] = useState([]); const [loading, setLoading] = useState(true); const [loadError, setLoadError] = useState(''); const [search, setSearch] = useState(''); const [view, setView] = useState('list'); // 'list' | 'create' | 'edit' | 'bulk' const [editUser, setEditUser] = useState(null); const [userPass, setUserPass] = useState('user@1234'); const [inputFocused, setInputFocused] = useState(false); const onIF = () => setInputFocused(true); const onIB = () => setInputFocused(false); const load = useCallback(async () => { setLoadError(''); setLoading(true); try { const { users } = await api.getUsers(); setUsers(users || []); } catch(e) { setLoadError(e.message || 'Failed to load users'); } finally { setLoading(false); } }, []); useEffect(() => { load(); api.getSettings().then(({ settings }) => { if (settings.user_pass) setUserPass(settings.user_pass); }).catch(() => {}); }, [load]); const filtered = users .filter(u => !search || u.name?.toLowerCase().includes(search.toLowerCase()) || u.display_name?.toLowerCase().includes(search.toLowerCase()) || u.email?.toLowerCase().includes(search.toLowerCase()) ) .sort((a, b) => a.name.localeCompare(b.name)); const goList = () => { setView('list'); setEditUser(null); }; const goCreate = () => { setView('create'); setEditUser(null); }; const goEdit = (u) => { setView('edit'); setEditUser(u); }; const goBulk = () => { setView('bulk'); setEditUser(null); }; const navItem = (label, active, onClick) => ( ); const isFormView = view === 'create' || view === 'edit'; return (
{/* ── Desktop sidebar ── */} {!isMobile && (
User Manager
View
{navItem(`All Users${!loading ? ` (${users.length})` : ''}`, view === 'list' || view === 'edit', goList)} {navItem('+ Create User', view === 'create', goCreate)} {navItem('Bulk Import CSV', view === 'bulk', goBulk)}
)} {/* ── Right panel ── */}
{/* Mobile tab bar */} {isMobile && (
Users
)} {/* Content */}
{/* LIST VIEW */} {view === 'list' && ( <>
setSearch(e.target.value)} onFocus={onIF} onBlur={onIB} autoComplete="new-password" autoCorrect="off" spellCheck={false} style={{ width:'100%', maxWidth: isMobile ? '100%' : 400 }} />
{loading ? (
) : loadError ? (
⚠ {loadError}
) : filtered.length === 0 ? (
{search ? 'No users match your search.' : 'No users yet.'}
) : ( filtered.map(u => ) )}
)} {/* CREATE / EDIT FORM */} {isFormView && (
{ load(); goList(); }} onCancel={goList} isMobile={isMobile} onIF={onIF} onIB={onIB} />
)} {/* BULK IMPORT */} {view === 'bulk' && (
)}
{/* Mobile footer — fixed, hidden when keyboard is up */} {isMobile && !inputFocused && (
)}
); }