This commit is contained in:
2026-06-06 17:02:45 -04:00
parent 47609e5e42
commit db4c656730
22 changed files with 6195 additions and 326 deletions

View File

@@ -6,6 +6,7 @@ import ConfidenceBadge, { getConfidenceColors } from "@/components/ConfidenceBad
import SymptomChecker from "@/components/SymptomChecker";
import TreatmentTimeline, { treatmentStepsWithUrgency } from "@/components/TreatmentTimeline";
import LookalikeWarning from "@/components/LookalikeWarning";
import FlagButton from "@/components/FlagButton";
import { getLookalikeDiseases } from "@/lib/api/diseases";
/**
@@ -45,15 +46,18 @@ export default function DiseaseCard({
<article
className={`
group/card relative rounded-xl border-2 overflow-hidden transition-all duration-200
${isPrimary
? `${colors.border} ${colors.bg} shadow-md`
: "border-zinc-200 dark:border-zinc-700 bg-white dark:bg-zinc-900 shadow-sm hover:shadow-md"
${
isPrimary
? `${colors.border} ${colors.bg} shadow-md`
: "border-zinc-200 dark:border-zinc-700 bg-white dark:bg-zinc-900 shadow-sm hover:shadow-md"
}
`}
>
{/* Primary diagnosis ribbon */}
{isPrimary && (
<div className={`${colors.accent} text-white text-xs font-bold uppercase tracking-wider px-4 py-1.5 flex items-center gap-2`}>
<div
className={`${colors.accent} text-white text-xs font-bold uppercase tracking-wider px-4 py-1.5 flex items-center gap-2`}
>
<svg className="h-3.5 w-3.5" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
<path d="M9.049 2.927c.3-.921 1.603-.921 1.902 0l1.07 3.292a1 1 0 00.95.69h3.462c.969 0 1.371 1.24.588 1.81l-2.8 2.034a1 1 0 00-.364 1.118l1.07 3.292c.3.921-.755 1.688-1.54 1.118l-2.8-2.034a1 1 0 00-1.175 0l-2.8 2.034c-.784.57-1.838-.197-1.539-1.118l1.07-3.292a1 1 0 00-.364-1.118L2.98 8.72c-.783-.57-.38-1.81.588-1.81h3.461a1 1 0 00.951-.69l1.07-3.292z" />
</svg>
@@ -71,13 +75,16 @@ export default function DiseaseCard({
>
<div className="flex items-start gap-3">
{/* Rank / causal agent icon */}
<div className={`
<div
className={`
flex h-9 w-9 shrink-0 items-center justify-center rounded-lg text-sm font-bold
${isPrimary
? `${colors.accent} text-white`
: "bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400"
${
isPrimary
? `${colors.accent} text-white`
: "bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400"
}
`}>
`}
>
{rank}
</div>
@@ -93,9 +100,7 @@ export default function DiseaseCard({
<p className="mt-0.5 text-xs italic text-zinc-500 dark:text-zinc-400">
{disease.scientificName}
</p>
<p className="mt-1 text-sm text-zinc-600 dark:text-zinc-400 line-clamp-2">
{summary}
</p>
<p className="mt-1 text-sm text-zinc-600 dark:text-zinc-400 line-clamp-2">{summary}</p>
</div>
{/* Expand/collapse chevron */}
@@ -105,7 +110,11 @@ export default function DiseaseCard({
fill="currentColor"
aria-hidden="true"
>
<path fillRule="evenodd" d="M5.22 7.22a.75.75 0 011.06 0L10 10.94l3.72-3.72a.75.75 0 111.06 1.06l-4.25 4.25a.75.75 0 01-1.06 0L5.22 8.28a.75.75 0 010-1.06z" clipRule="evenodd" />
<path
fillRule="evenodd"
d="M5.22 7.22a.75.75 0 011.06 0L10 10.94l3.72-3.72a.75.75 0 111.06 1.06l-4.25 4.25a.75.75 0 01-1.06 0L5.22 8.28a.75.75 0 010-1.06z"
clipRule="evenodd"
/>
</svg>
</div>
</button>
@@ -133,17 +142,47 @@ export default function DiseaseCard({
{/* Symptom checker */}
<div>
<div className="flex items-center justify-between mb-1">
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100">
Symptom Checker
</h4>
<FlagButton
contentType="disease_symptoms"
contentId={disease.id}
fieldName="symptoms"
label="symptoms"
small
/>
</div>
<SymptomChecker symptoms={disease.symptoms} />
</div>
{/* Causes */}
<div>
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100 mb-2 flex items-center gap-2">
<svg className="h-4 w-4 text-zinc-400" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
<path fillRule="evenodd" d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-8-3a1 1 0 00-.867.5 1 1 0 11-1.731-1A3 3 0 0113 8a3.001 3.001 0 01-2 2.83V11a1 1 0 11-2 0v-1a1 1 0 011-1 1 1 0 100-2zm0 8a1 1 0 100-2 1 1 0 000 2z" clipRule="evenodd" />
</svg>
Causes & Contributing Factors
</h4>
<div className="flex items-center justify-between mb-2">
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100 flex items-center gap-2">
<svg
className="h-4 w-4 text-zinc-400"
viewBox="0 0 20 20"
fill="currentColor"
aria-hidden="true"
>
<path
fillRule="evenodd"
d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-8-3a1 1 0 00-.867.5 1 1 0 11-1.731-1A3 3 0 0113 8a3.001 3.001 0 01-2 2.83V11a1 1 0 11-2 0v-1a1 1 0 011-1 1 1 0 100-2zm0 8a1 1 0 100-2 1 1 0 000 2z"
clipRule="evenodd"
/>
</svg>
Causes & Contributing Factors
</h4>
<FlagButton
contentType="disease_causes"
contentId={disease.id}
fieldName="causes"
label="causes"
small
/>
</div>
<ul className="space-y-1.5" role="list">
{disease.causes.map((cause, i) => (
<li key={i} className="flex items-start gap-2">
@@ -156,12 +195,30 @@ export default function DiseaseCard({
{/* Treatment timeline */}
<div>
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100 mb-2 flex items-center gap-2">
<svg className="h-4 w-4 text-leaf-green-600 dark:text-leaf-green-400" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.857-9.809a.75.75 0 00-1.214-.882l-3.483 4.79-1.88-1.88a.75.75 0 10-1.06 1.061l2.5 2.5a.75.75 0 001.137-.089l4-5.5z" clipRule="evenodd" />
</svg>
Treatment Plan
</h4>
<div className="flex items-center justify-between mb-2">
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100 flex items-center gap-2">
<svg
className="h-4 w-4 text-leaf-green-600 dark:text-leaf-green-400"
viewBox="0 0 20 20"
fill="currentColor"
aria-hidden="true"
>
<path
fillRule="evenodd"
d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.857-9.809a.75.75 0 00-1.214-.882l-3.483 4.79-1.88-1.88a.75.75 0 10-1.06 1.061l2.5 2.5a.75.75 0 001.137-.089l4-5.5z"
clipRule="evenodd"
/>
</svg>
Treatment Plan
</h4>
<FlagButton
contentType="disease_treatment"
contentId={disease.id}
fieldName="treatment"
label="treatment"
small
/>
</div>
<TreatmentTimeline
steps={treatmentStepsWithUrgency(disease.treatment)}
severity={disease.severity}
@@ -170,12 +227,30 @@ export default function DiseaseCard({
{/* Prevention tips */}
<div>
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100 mb-2 flex items-center gap-2">
<svg className="h-4 w-4 text-leaf-green-600 dark:text-leaf-green-400" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true">
<path fillRule="evenodd" d="M6.32 2.577a49.255 49.255 0 0111.36 0c1.497.174 2.57 1.46 2.57 2.93V21a.75.75 0 01-1.085.67L10 18.089l-9.165 3.583A.75.75 0 010 21V5.507c0-1.47 1.073-2.756 2.57-2.93a49.254 49.254 0 0111.36 0zM12 9a2 2 0 11-4 0 2 2 0 014 0zm-2 3a1 1 0 00-1 1v1a1 1 0 001 1h0a1 1 0 001-1v-1a1 1 0 00-1-1z" clipRule="evenodd" />
</svg>
Prevention Tips
</h4>
<div className="flex items-center justify-between mb-2">
<h4 className="text-sm font-semibold text-zinc-900 dark:text-zinc-100 flex items-center gap-2">
<svg
className="h-4 w-4 text-leaf-green-600 dark:text-leaf-green-400"
viewBox="0 0 20 20"
fill="currentColor"
aria-hidden="true"
>
<path
fillRule="evenodd"
d="M6.32 2.577a49.255 49.255 0 0111.36 0c1.497.174 2.57 1.46 2.57 2.93V21a.75.75 0 01-1.085.67L10 18.089l-9.165 3.583A.75.75 0 010 21V5.507c0-1.47 1.073-2.756 2.57-2.93a49.254 49.254 0 0111.36 0zM12 9a2 2 0 11-4 0 2 2 0 014 0zm-2 3a1 1 0 00-1 1v1a1 1 0 001 1h0a1 1 0 001-1v-1a1 1 0 00-1-1z"
clipRule="evenodd"
/>
</svg>
Prevention Tips
</h4>
<FlagButton
contentType="disease_prevention"
contentId={disease.id}
fieldName="prevention"
label="prevention tips"
small
/>
</div>
<ul className="space-y-1.5" role="list">
{disease.prevention.map((tip, i) => (
<li key={i} className="flex items-start gap-2">
@@ -187,9 +262,7 @@ export default function DiseaseCard({
</div>
{/* Lookalike warnings */}
{lookalikes.length > 0 && (
<LookalikeWarning disease={disease} lookalikes={lookalikes} />
)}
{lookalikes.length > 0 && <LookalikeWarning disease={disease} lookalikes={lookalikes} />}
{/* Feedback buttons */}
<div className="pt-2 border-t border-zinc-200 dark:border-zinc-700">
@@ -203,9 +276,10 @@ export default function DiseaseCard({
className={`
inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-sm font-medium
transition-colors
${feedback === "yes"
? "bg-leaf-green-100 dark:bg-leaf-green-900/50 text-leaf-green-700 dark:text-leaf-green-300 ring-1 ring-leaf-green-300 dark:ring-leaf-green-700"
: "bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400 hover:bg-zinc-200 dark:hover:bg-zinc-700"
${
feedback === "yes"
? "bg-leaf-green-100 dark:bg-leaf-green-900/50 text-leaf-green-700 dark:text-leaf-green-300 ring-1 ring-leaf-green-300 dark:ring-leaf-green-700"
: "bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400 hover:bg-zinc-200 dark:hover:bg-zinc-700"
}
`}
aria-pressed={feedback === "yes"}
@@ -221,9 +295,10 @@ export default function DiseaseCard({
className={`
inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-sm font-medium
transition-colors
${feedback === "no"
? "bg-red-100 dark:bg-red-900/50 text-red-700 dark:text-red-300 ring-1 ring-red-300 dark:ring-red-700"
: "bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400 hover:bg-zinc-200 dark:hover:bg-zinc-700"
${
feedback === "no"
? "bg-red-100 dark:bg-red-900/50 text-red-700 dark:text-red-300 ring-1 ring-red-300 dark:ring-red-700"
: "bg-zinc-100 dark:bg-zinc-800 text-zinc-600 dark:text-zinc-400 hover:bg-zinc-200 dark:hover:bg-zinc-700"
}
`}
aria-pressed={feedback === "no"}