379 lines
20 KiB
TypeScript
379 lines
20 KiB
TypeScript
import React, { useState, useEffect, useMemo } from 'react';
|
|
import { useNavigate } from 'react-router-dom';
|
|
import { motion, AnimatePresence } from 'framer-motion';
|
|
import { BookOpen, FileText, Layers, Loader2, Plus, Search, Trash2, Edit2, AlertCircle, Check, Clock, XCircle } from 'lucide-react';
|
|
import { apiClient } from '../../services/apiClient';
|
|
import { templateService } from '../../services/templateService';
|
|
import { AssessmentTemplate } from '../../types';
|
|
import { useToast } from '../../contexts/ToastContext';
|
|
import { useConfirm } from '../../contexts/ConfirmContext';
|
|
import { useLanguage } from '../../contexts/LanguageContext';
|
|
|
|
interface QuestionBankViewProps {
|
|
isAdmin?: boolean;
|
|
}
|
|
|
|
interface QuestionBankItem {
|
|
id: string;
|
|
name: string;
|
|
description?: string;
|
|
status: string;
|
|
templateId?: string;
|
|
createdAt: string;
|
|
}
|
|
|
|
type StatusFilter = 'ALL' | 'DRAFT' | 'PENDING_REVIEW' | 'PUBLISHED';
|
|
|
|
export default function QuestionBankView({ isAdmin: _isAdmin }: QuestionBankViewProps) {
|
|
const navigate = useNavigate();
|
|
const { t } = useLanguage();
|
|
const { showSuccess, showError } = useToast();
|
|
const { confirm } = useConfirm();
|
|
|
|
const [banks, setBanks] = useState<QuestionBankItem[]>([]);
|
|
const [loading, setLoading] = useState(false);
|
|
const [error, setError] = useState('');
|
|
const [showDrawer, setShowDrawer] = useState(false);
|
|
const [formData, setFormData] = useState({ name: '', description: '', templateId: '' });
|
|
const [saving, setSaving] = useState(false);
|
|
const [templates, setTemplates] = useState<AssessmentTemplate[]>([]);
|
|
const [loadingTemplates, setLoadingTemplates] = useState(false);
|
|
const [deletingId, setDeletingId] = useState<string | null>(null);
|
|
const [statusFilter, setStatusFilter] = useState<StatusFilter>('ALL');
|
|
const [searchQuery, setSearchQuery] = useState('');
|
|
|
|
useEffect(() => { fetchData(); }, []);
|
|
|
|
const fetchData = async () => {
|
|
try {
|
|
setLoading(true);
|
|
const res = await apiClient.request('/question-banks', {});
|
|
if (!res.ok) throw new Error(res.status.toString());
|
|
const data = await res.json();
|
|
setBanks(Array.isArray(data) ? data : (data.data || []));
|
|
} catch (err: any) {
|
|
setError(err.message || t('actionFailed'));
|
|
showError(err.message || t('actionFailed'));
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
const openDrawer = () => {
|
|
setFormData({ name: '', description: '', templateId: '' });
|
|
setLoadingTemplates(true);
|
|
templateService.getAll()
|
|
.then(data => setTemplates(data))
|
|
.catch(() => showError(t('actionFailed')))
|
|
.finally(() => setLoadingTemplates(false));
|
|
setShowDrawer(true);
|
|
};
|
|
|
|
const handleCreate = async (e: React.FormEvent) => {
|
|
e.preventDefault();
|
|
if (!formData.name.trim()) return;
|
|
setSaving(true);
|
|
try {
|
|
const payload: any = { name: formData.name, description: formData.description };
|
|
if (formData.templateId) payload.templateId = formData.templateId;
|
|
const res = await apiClient.request('/question-banks', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(payload),
|
|
});
|
|
if (!res.ok) {
|
|
const errBody = await res.text().catch(() => '');
|
|
let msg = res.status.toString();
|
|
try { const parsed = JSON.parse(errBody); if (parsed.message) msg = parsed.message; } catch {}
|
|
throw new Error(msg);
|
|
}
|
|
setShowDrawer(false);
|
|
showSuccess(t('questionBankCreated'));
|
|
fetchData();
|
|
} catch (err: any) {
|
|
showError(err.message || t('actionFailed'));
|
|
} finally {
|
|
setSaving(false);
|
|
}
|
|
};
|
|
|
|
const handleDelete = async (e: React.MouseEvent, bankId: string, bankName: string) => {
|
|
e.stopPropagation();
|
|
const ok = await confirm({ message: t('confirmDeleteBank').replace('$1', bankName), confirmLabel: t('delete'), cancelLabel: t('cancel') });
|
|
if (!ok) return;
|
|
setDeletingId(bankId);
|
|
try {
|
|
const res = await apiClient.request(`/question-banks/${bankId}`, { method: 'DELETE' });
|
|
if (!res.ok) {
|
|
const errBody = await res.text().catch(() => '');
|
|
let msg = res.status.toString();
|
|
try { const parsed = JSON.parse(errBody); if (parsed.message) msg = parsed.message; } catch {}
|
|
throw new Error(msg);
|
|
}
|
|
showSuccess(t('confirm'));
|
|
fetchData();
|
|
} catch (err: any) {
|
|
showError(err.message || t('questionBankDeleteFailed'));
|
|
} finally {
|
|
setDeletingId(null);
|
|
}
|
|
};
|
|
|
|
const handleCardClick = (bank: QuestionBankItem) => {
|
|
navigate(`/question-banks/${bank.id}`);
|
|
};
|
|
|
|
const filteredBanks = useMemo(() => {
|
|
let result = banks;
|
|
if (statusFilter !== 'ALL') result = result.filter(b => b.status === statusFilter);
|
|
if (searchQuery.trim()) {
|
|
const q = searchQuery.toLowerCase();
|
|
result = result.filter(b => b.name.toLowerCase().includes(q) || (b.description || '').toLowerCase().includes(q));
|
|
}
|
|
return result;
|
|
}, [banks, statusFilter, searchQuery]);
|
|
|
|
const STATUS_TABS: { key: StatusFilter; label: string; icon: React.ReactNode; count: (b: QuestionBankItem[]) => number }[] = [
|
|
{ key: 'ALL', label: t('all'), icon: <Layers size={14} />, count: (b) => b.length },
|
|
{ key: 'PUBLISHED', label: t('published'), icon: <Check size={14} />, count: (b) => b.filter(i => i.status === 'PUBLISHED').length },
|
|
{ key: 'DRAFT', label: t('draft'), icon: <FileText size={14} />, count: (b) => b.filter(i => i.status === 'DRAFT').length },
|
|
{ key: 'PENDING_REVIEW', label: t('pendingReview'), icon: <Clock size={14} />, count: (b) => b.filter(i => i.status === 'PENDING_REVIEW').length },
|
|
];
|
|
|
|
const stats = useMemo(() => ({
|
|
total: banks.length,
|
|
published: banks.filter(b => b.status === 'PUBLISHED').length,
|
|
draft: banks.filter(b => b.status === 'DRAFT').length,
|
|
pending: banks.filter(b => b.status === 'PENDING_REVIEW').length,
|
|
}), [banks]);
|
|
|
|
const statusLabels: Record<string, string> = {
|
|
PUBLISHED: t('published'),
|
|
PENDING_REVIEW: t('pendingReview'),
|
|
REJECTED: t('rejected'),
|
|
DRAFT: t('draft'),
|
|
};
|
|
|
|
return (
|
|
<div className="space-y-6 overflow-y-auto h-full">
|
|
<div className="flex items-center justify-between">
|
|
<div>
|
|
<h1 className="text-2xl font-black text-slate-900">{t('questionBankManagement')}</h1>
|
|
<p className="text-sm text-slate-500 mt-1">{t('questionBankManagementDesc')}</p>
|
|
</div>
|
|
<button
|
|
onClick={openDrawer}
|
|
className="px-5 py-3 bg-blue-600 text-white rounded-2xl text-sm font-black uppercase tracking-widest flex items-center gap-2 shadow-lg shadow-blue-600/20 hover:bg-blue-700 transition-all active:scale-[0.98]"
|
|
>
|
|
<Plus size={18} />
|
|
{t('createQuestionBank')}
|
|
</button>
|
|
</div>
|
|
|
|
{!loading && !error && banks.length > 0 && (
|
|
<div className="grid grid-cols-4 gap-4">
|
|
{[
|
|
{ label: t('totalBanks'), value: stats.total, color: 'bg-slate-50 border-slate-200 text-slate-700', icon: <Layers size={16} className="text-slate-500" /> },
|
|
{ label: t('published'), value: stats.published, color: 'bg-emerald-50 border-emerald-200/50 text-emerald-700', icon: <Check size={16} className="text-emerald-500" /> },
|
|
{ label: t('draft'), value: stats.draft, color: 'bg-slate-50 border-slate-200 text-slate-700', icon: <FileText size={16} className="text-slate-500" /> },
|
|
{ label: t('pendingReview'), value: stats.pending, color: 'bg-amber-50 border-amber-200/50 text-amber-700', icon: <Clock size={16} className="text-amber-500" /> },
|
|
].map((stat, i) => (
|
|
<motion.div key={stat.label} initial={{ opacity: 0, y: 10 }} animate={{ opacity: 1, y: 0 }} transition={{ delay: i * 0.05 }}
|
|
className={`${stat.color} rounded-2xl border p-4`}>
|
|
<div className="flex items-center justify-between mb-1">
|
|
<span className="text-[10px] font-black uppercase tracking-widest opacity-70">{stat.label}</span>
|
|
{stat.icon}
|
|
</div>
|
|
<div className="text-2xl font-black">{stat.value}</div>
|
|
</motion.div>
|
|
))}
|
|
</div>
|
|
)}
|
|
|
|
{!loading && !error && banks.length > 0 && (
|
|
<div className="flex items-center gap-4">
|
|
<div className="relative flex-1 max-w-sm">
|
|
<Search size={16} className="absolute left-4 top-1/2 -translate-y-1/2 text-slate-400" />
|
|
<input
|
|
type="text"
|
|
value={searchQuery}
|
|
onChange={(e) => setSearchQuery(e.target.value)}
|
|
placeholder={t('searchQuestionBanksPlaceholder')}
|
|
className="w-full pl-10 pr-4 py-3 bg-slate-50 border border-slate-200 rounded-2xl text-sm font-bold focus:ring-4 focus:ring-blue-500/10 focus:border-blue-500/50 outline-none transition-all placeholder:text-slate-300"
|
|
/>
|
|
</div>
|
|
<div className="flex gap-1 bg-slate-50 rounded-2xl p-1 border border-slate-200">
|
|
{STATUS_TABS.map((tab) => {
|
|
const active = statusFilter === tab.key;
|
|
return (
|
|
<button
|
|
key={tab.key}
|
|
onClick={() => setStatusFilter(tab.key)}
|
|
className={`flex items-center gap-1.5 px-4 py-2 rounded-xl text-xs font-bold transition-all ${
|
|
active
|
|
? 'bg-white text-slate-900 shadow-sm border border-slate-200/50'
|
|
: 'text-slate-500 hover:text-slate-700'
|
|
}`}
|
|
>
|
|
{tab.icon}
|
|
{tab.label}
|
|
<span className={`${active ? 'bg-slate-100 text-slate-600' : 'bg-white/50 text-slate-400'} px-1.5 py-0.5 rounded-lg text-[10px] font-black`}>
|
|
{tab.count(banks)}
|
|
</span>
|
|
</button>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{loading ? (
|
|
<div className="flex items-center justify-center py-20">
|
|
<Loader2 className="w-8 h-8 animate-spin text-blue-600 opacity-20" />
|
|
</div>
|
|
) : error ? (
|
|
<div className="flex items-center gap-3 text-red-500 bg-red-50 rounded-2xl p-6 border border-red-100">
|
|
<AlertCircle size={20} />
|
|
<span className="text-sm font-bold">{t('actionFailed')}</span>
|
|
<button onClick={fetchData} className="ml-auto text-xs font-black text-red-600 hover:text-red-700 uppercase tracking-widest">{t('retry')}</button>
|
|
</div>
|
|
) : banks.length === 0 ? (
|
|
<div className="bg-slate-50 rounded-[2rem] border-2 border-dashed border-slate-200 p-20 text-center">
|
|
<div className="w-16 h-16 bg-slate-100 rounded-3xl flex items-center justify-center mx-auto mb-6">
|
|
<BookOpen size={32} className="text-slate-300" />
|
|
</div>
|
|
<p className="text-slate-400 font-black uppercase tracking-widest text-xs mb-2">{t('noQuestionBanks')}</p>
|
|
<p className="text-slate-300 text-xs mb-6">{t('createFirstBank')}</p>
|
|
<button onClick={openDrawer} className="inline-flex items-center gap-2 px-6 py-3 bg-blue-600 text-white rounded-2xl text-sm font-black uppercase tracking-widest hover:bg-blue-700 transition-all active:scale-[0.98] shadow-lg shadow-blue-600/20">
|
|
<Plus size={18} /> {t('createQuestionBank')}
|
|
</button>
|
|
</div>
|
|
) : filteredBanks.length === 0 ? (
|
|
<div className="bg-slate-50 rounded-[2rem] border-2 border-dashed border-slate-200 p-20 text-center">
|
|
<Search size={32} className="text-slate-300 mx-auto mb-4" />
|
|
<p className="text-slate-400 font-bold text-xs uppercase tracking-widest">{t('noMatchingQuestionBanks')}</p>
|
|
<p className="text-slate-300 text-xs mt-2">{t('tryChangingFilter')}</p>
|
|
</div>
|
|
) : (
|
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
|
<AnimatePresence mode="popLayout">
|
|
{filteredBanks.map((bank) => (
|
|
<motion.div
|
|
key={bank.id}
|
|
layout
|
|
initial={{ opacity: 0, scale: 0.95 }}
|
|
animate={{ opacity: 1, scale: 1 }}
|
|
exit={{ opacity: 0, scale: 0.95 }}
|
|
onClick={() => handleCardClick(bank)}
|
|
className="bg-white border border-slate-200 rounded-3xl p-5 shadow-sm hover:shadow-md transition-all cursor-pointer group relative overflow-hidden"
|
|
>
|
|
<div className={`absolute top-0 right-0 w-32 h-32 rounded-full blur-3xl -mr-16 -mt-16 ${
|
|
bank.status === 'PUBLISHED' ? 'bg-emerald-500/5' :
|
|
bank.status === 'PENDING_REVIEW' ? 'bg-amber-500/5' :
|
|
bank.status === 'REJECTED' ? 'bg-red-500/5' : 'bg-blue-500/5'
|
|
}`} />
|
|
|
|
<div className="relative z-10">
|
|
<div className="flex items-start justify-between mb-3">
|
|
<h3 className="font-black text-base text-slate-900 pr-8 line-clamp-1">{bank.name}</h3>
|
|
<div className="flex gap-0.5 opacity-0 group-hover:opacity-100 transition-opacity absolute top-0 right-0">
|
|
<button onClick={(e) => { e.stopPropagation(); handleCardClick(bank); }}
|
|
className="p-1.5 text-slate-400 hover:text-blue-600 hover:bg-blue-50 rounded-xl transition-all" title={t('edit')}>
|
|
<Edit2 size={13} />
|
|
</button>
|
|
<button onClick={(e) => handleDelete(e, bank.id, bank.name)} disabled={deletingId === bank.id}
|
|
className="p-1.5 text-slate-400 hover:text-red-600 hover:bg-red-50 rounded-xl transition-all disabled:opacity-50" title={t('delete')}>
|
|
{deletingId === bank.id ? <Loader2 size={13} className="animate-spin" /> : <Trash2 size={13} />}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<p className="text-xs text-slate-500 mb-4 line-clamp-2 h-8">{bank.description || t('noDescription')}</p>
|
|
|
|
<div className="flex items-center justify-between pt-3 border-t border-slate-50">
|
|
<span className={`px-2.5 py-1 text-[10px] font-black uppercase tracking-widest rounded-full border ${
|
|
bank.status === 'PUBLISHED' ? 'bg-emerald-50 text-emerald-600 border-emerald-200/50' :
|
|
bank.status === 'PENDING_REVIEW' ? 'bg-amber-50 text-amber-600 border-amber-200/50' :
|
|
bank.status === 'REJECTED' ? 'bg-red-50 text-red-500 border-red-200/50' :
|
|
'bg-slate-50 text-slate-500 border-slate-200/50'
|
|
}`}>
|
|
{statusLabels[bank.status] || bank.status}
|
|
</span>
|
|
<span className="text-[10px] text-slate-400 font-medium">
|
|
{new Date(bank.createdAt).toLocaleDateString('zh-CN')}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</motion.div>
|
|
))}
|
|
</AnimatePresence>
|
|
</div>
|
|
)}
|
|
|
|
<AnimatePresence>
|
|
{showDrawer && (
|
|
<>
|
|
<motion.div initial={{ opacity: 0 }} animate={{ opacity: 1 }} exit={{ opacity: 0 }}
|
|
onClick={() => setShowDrawer(false)} className="fixed inset-0 bg-slate-900/40 backdrop-blur-sm z-40" />
|
|
<motion.div initial={{ x: '100%' }} animate={{ x: 0 }} exit={{ x: '100%' }}
|
|
transition={{ type: 'spring', damping: 25, stiffness: 200 }}
|
|
className="fixed right-0 top-0 h-full w-full max-w-md bg-white shadow-2xl z-50 flex flex-col">
|
|
<div className="flex items-center justify-between px-6 py-4 border-b border-slate-100">
|
|
<div className="flex items-center gap-3">
|
|
<div className="w-10 h-10 bg-blue-50 text-blue-600 rounded-2xl flex items-center justify-center"><Plus size={22} /></div>
|
|
<h2 className="text-lg font-black text-slate-900">{t('createQuestionBank')}</h2>
|
|
</div>
|
|
<button onClick={() => setShowDrawer(false)} className="p-2 text-slate-400 hover:text-slate-600 hover:bg-slate-50 rounded-xl transition-all"><XCircle size={22} /></button>
|
|
</div>
|
|
<div className="flex-1 overflow-y-auto p-6">
|
|
<form id="create-form" onSubmit={handleCreate} className="space-y-5">
|
|
<div className="space-y-1.5">
|
|
<label className="text-[10px] font-black text-slate-400 uppercase tracking-widest px-1 ml-1 flex items-center gap-2">
|
|
<BookOpen size={12} className="text-blue-500" /> {t('name')} <span className="text-red-500">*</span>
|
|
</label>
|
|
<input type="text" value={formData.name} onChange={(e) => setFormData({ ...formData, name: e.target.value })}
|
|
className="w-full px-5 py-4 bg-slate-50 border border-slate-200 rounded-[1.25rem] text-sm font-bold focus:ring-4 focus:ring-blue-500/10 focus:border-blue-500/50 outline-none transition-all placeholder:text-slate-300"
|
|
placeholder={t('name')} required autoFocus />
|
|
</div>
|
|
<div className="space-y-1.5">
|
|
<label className="text-[10px] font-black text-slate-400 uppercase tracking-widest px-1 ml-1 flex items-center gap-2">
|
|
<FileText size={12} className="text-blue-500" /> {t('description')}
|
|
</label>
|
|
<input type="text" value={formData.description} onChange={(e) => setFormData({ ...formData, description: e.target.value })}
|
|
className="w-full px-5 py-4 bg-slate-50 border border-slate-200 rounded-[1.25rem] text-sm font-bold focus:ring-4 focus:ring-blue-500/10 focus:border-blue-500/50 outline-none transition-all placeholder:text-slate-300"
|
|
placeholder={t('description')} />
|
|
</div>
|
|
<div className="space-y-1.5">
|
|
<label className="text-[10px] font-black text-slate-400 uppercase tracking-widest px-1 ml-1 flex items-center gap-2">
|
|
<Layers size={12} className="text-blue-500" /> {t('linkTemplate')}
|
|
</label>
|
|
<select value={formData.templateId} onChange={(e) => setFormData({ ...formData, templateId: e.target.value })}
|
|
className="w-full px-5 py-4 bg-slate-50 border border-slate-200 rounded-[1.25rem] text-sm font-bold focus:ring-4 focus:ring-blue-500/10 focus:border-blue-500/50 outline-none transition-all cursor-pointer"
|
|
disabled={loadingTemplates}>
|
|
<option value="">{t('noTemplate')}</option>
|
|
{templates.map((t) => <option key={t.id} value={t.id}>{t.name}</option>)}
|
|
</select>
|
|
{loadingTemplates && (
|
|
<div className="flex items-center gap-2 px-2 py-1">
|
|
<Loader2 size={12} className="animate-spin text-slate-400" />
|
|
<span className="text-[10px] text-slate-400 font-medium">{t('loading')}</span>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</form>
|
|
</div>
|
|
<div className="p-6 border-t border-slate-100">
|
|
<button type="submit" form="create-form" disabled={saving || !formData.name.trim()}
|
|
className="w-full flex items-center justify-center gap-2 px-6 py-4 bg-blue-600 text-white font-black uppercase tracking-widest text-xs rounded-[1.25rem] hover:bg-blue-700 active:scale-[0.98] transition-all disabled:opacity-50 disabled:cursor-not-allowed shadow-xl shadow-blue-100">
|
|
{saving ? <Loader2 size={18} className="animate-spin" /> : <Plus size={18} />}
|
|
{saving ? t('creating') : t('createQuestionBank')}
|
|
</button>
|
|
</div>
|
|
</motion.div>
|
|
</>
|
|
)}
|
|
</AnimatePresence>
|
|
</div>
|
|
);
|
|
}
|