feat: 完善中文心理测评平台

This commit is contained in:
mikemoi
2026-06-22 22:59:01 +02:00
commit 9227c687fc
160 changed files with 16974 additions and 0 deletions
@@ -0,0 +1,156 @@
'use client';
import React from 'react';
import { useScopedI18n } from '@/locales/client';
import { calculateSCL90Results } from '../../test/private/SCL90Calculator';
interface SCL90ResultProps {
answers: string[];
}
export function SCL90Result({ answers }: SCL90ResultProps) {
const t = useScopedI18n('components.scl90Result');
const tCommon = useScopedI18n('common');
// Convert answer format to the format required by calculator
const answersMap: { [key: number]: string } = {};
answers.forEach((answer, index) => {
answersMap[index + 1] = answer;
});
const results = calculateSCL90Results({
answers: answersMap,
questions: []
});
const factorNames = {
somatization: t('factors.somatization'),
obsessive: t('factors.obsessive'),
interpersonal: t('factors.interpersonal'),
depression: t('factors.depression'),
anxiety: t('factors.anxiety'),
hostility: t('factors.hostility'),
phobic: t('factors.phobic'),
paranoid: t('factors.paranoid'),
psychotic: t('factors.psychotic'),
other: t('factors.other')
};
const severityNames = {
normal: tCommon('severity.normal'),
mild: tCommon('severity.mild'),
moderate: tCommon('severity.moderate'),
severe: tCommon('severity.severe')
};
const getSeverityColor = (severity: string) => {
switch (severity) {
case "normal": return "text-green-600";
case "mild": return "text-yellow-600";
case "moderate": return "text-orange-600";
case "severe": return "text-red-600";
default: return "text-gray-600";
}
};
const getFactorSeverity = (score: number) => {
if (score >= 3) return "severe";
if (score >= 2) return "moderate";
if (score >= 1.5) return "mild";
return "normal";
};
return (
<div className="mt-6 space-y-6">
{/* Overall score */}
<div className="bg-white border rounded-lg p-6 shadow-sm">
<h3 className="text-lg font-semibold mb-4">{t('labels.overall_assessment')}</h3>
<div className="grid grid-cols-1 md:grid-cols-4 gap-4">
<MetricCard title={tCommon('labels.total_score')} value={results.totalScore} />
<MetricCard title={t('labels.positive_item_count')} value={results.positiveItemCount} />
<MetricCard title={t('labels.positive_symptom_average')} value={results.positiveItemAverage.toFixed(2)} />
<MetricCard
title={tCommon('labels.severity_level')}
value={severityNames[results.severity as keyof typeof severityNames] || t('warnings.unknown_level')}
className={getSeverityColor(results.severity)}
/>
</div>
</div>
{/* Factor scores */}
<div className="bg-white border rounded-lg p-6 shadow-sm">
<h3 className="text-lg font-semibold mb-4">{t('labels.factor_analysis')}</h3>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
{Object.entries(results.factorScores).map(([factor, score]) => {
const factorSeverity = getFactorSeverity(Number(score));
return (
<div key={factor} className="border rounded-lg p-4">
<div className="flex justify-between items-center mb-2">
<span className="font-medium">{factorNames[factor as keyof typeof factorNames]}</span>
<span className={`text-sm px-2 py-1 rounded ${getSeverityColor(factorSeverity)}`}>
{severityNames[factorSeverity as keyof typeof severityNames]}
</span>
</div>
<div className="text-2xl font-bold text-indigo-600">{Number(score).toFixed(2)}</div>
</div>
);
})}
</div>
</div>
{/* Result interpretation */}
<div className="bg-white border rounded-lg p-6 shadow-sm">
<h3 className="text-lg font-semibold mb-4">{tCommon('labels.result_interpretation')}</h3>
<div className="space-y-3 text-sm text-gray-700">
<div>
<strong>{t('clinical.rating_criteria')}</strong>
<ul className="mt-1 ml-4 space-y-1">
<li> {t('clinical.rating_scale')}</li>
</ul>
</div>
<div>
<strong>{t('clinical.judgment_criteria')}</strong>
<ul className="mt-1 ml-4 space-y-1">
<li> {t('clinical.total_score_criteria')}</li>
<li> {t('clinical.factor_score_2')}</li>
<li> {t('clinical.factor_score_3')}</li>
</ul>
</div>
{results.isSevere && (
<div className="bg-red-50 border border-red-200 rounded p-3 mt-4">
<div className="flex items-center">
<div className="flex-shrink-0">
<svg className="h-5 w-5 text-red-400" viewBox="0 0 20 20" fill="currentColor">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clipRule="evenodd" />
</svg>
</div>
<div className="ml-3">
<div className="text-sm font-medium text-red-800">
{t('warnings.severe_condition')}
</div>
</div>
</div>
</div>
)}
</div>
</div>
</div>
);
}
interface MetricCardProps {
title: string;
value: React.ReactNode;
className?: string;
}
function MetricCard({ title, value, className = '' }: MetricCardProps) {
return (
<div className="bg-gray-50 rounded-lg p-4 flex flex-col items-center">
<span className="text-sm text-gray-500 mb-1">{title}</span>
<span className={`text-2xl font-semibold ${className || 'text-indigo-600'}`}>
{value}
</span>
</div>
);
}