feat: 完善中文心理测评平台
This commit is contained in:
@@ -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>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user