Enhance development standards and UI components

- Updated .rules to enforce stricter UI/UX standards, including exclusive use of Lucide icons and consistent patterns for entity view pages.
- Added new UI components for entity views, including headers, sections, and quick actions to improve layout and reusability.
- Refactored existing pages (experiments, participants, studies, trials) to utilize the new entity view components, enhancing consistency across the dashboard.
- Improved accessibility and user experience by implementing loading states and error boundaries in async operations.
- Updated package dependencies to ensure compatibility and performance improvements.

Features:
- Comprehensive guidelines for component reusability and visual consistency.
- Enhanced user interface with new entity view components for better organization and navigation.

Breaking Changes: None - existing functionality remains intact.
This commit is contained in:
2025-08-05 02:36:44 -04:00
parent 7cdc1a2340
commit 544207e9a2
16 changed files with 3643 additions and 1531 deletions

View File

@@ -0,0 +1,448 @@
"use client";
import { formatDistanceToNow } from "date-fns";
import {
ArrowLeft,
BarChart3,
Bot,
Calendar,
CheckCircle,
Edit,
FileText,
FlaskConical,
Play,
Settings,
Share,
Target,
Users,
AlertTriangle,
XCircle,
} from "lucide-react";
import Link from "next/link";
import { notFound } from "next/navigation";
import { useEffect, useState } from "react";
import { Badge } from "~/components/ui/badge";
import { Button } from "~/components/ui/button";
import {
EntityView,
EntityViewHeader,
EntityViewSection,
EntityViewSidebar,
EmptyState,
InfoGrid,
QuickActions,
StatsGrid,
} from "~/components/ui/entity-view";
import { useBreadcrumbsEffect } from "~/components/ui/breadcrumb-provider";
import { useSession } from "next-auth/react";
import { api } from "~/trpc/react";
interface ExperimentDetailPageProps {
params: Promise<{
id: string;
}>;
}
const statusConfig = {
draft: {
label: "Draft",
variant: "secondary" as const,
icon: "FileText" as const,
},
testing: {
label: "Testing",
variant: "outline" as const,
icon: "FlaskConical" as const,
},
ready: {
label: "Ready",
variant: "default" as const,
icon: "CheckCircle" as const,
},
deprecated: {
label: "Deprecated",
variant: "destructive" as const,
icon: "AlertTriangle" as const,
},
};
export default function ExperimentDetailPage({
params,
}: ExperimentDetailPageProps) {
const { data: session } = useSession();
const [experiment, setExperiment] = useState<any>(null);
const [trials, setTrials] = useState<any[]>([]);
const [loading, setLoading] = useState(true);
const [resolvedParams, setResolvedParams] = useState<{ id: string } | null>(
null,
);
useEffect(() => {
async function resolveParams() {
const resolved = await params;
setResolvedParams(resolved);
}
resolveParams();
}, [params]);
const { data: experimentData } = api.experiments.get.useQuery(
{ id: resolvedParams?.id ?? "" },
{ enabled: !!resolvedParams?.id },
);
const { data: trialsData } = api.trials.list.useQuery(
{ experimentId: resolvedParams?.id ?? "", limit: 10 },
{ enabled: !!resolvedParams?.id },
);
useEffect(() => {
if (experimentData) {
setExperiment(experimentData);
}
if (trialsData) {
setTrials(trialsData);
}
if (experimentData !== undefined) {
setLoading(false);
}
}, [experimentData, trialsData]);
// Set breadcrumbs
useBreadcrumbsEffect([
{ label: "Dashboard", href: "/dashboard" },
{ label: "Experiments", href: "/experiments" },
{ label: experiment?.name || "Experiment" },
]);
if (!session?.user) {
return notFound();
}
if (loading || !experiment) {
return <div>Loading...</div>;
}
const userRole = session.user.roles?.[0]?.role ?? "observer";
const canEdit = ["administrator", "researcher"].includes(userRole);
const statusInfo = statusConfig[experiment.status];
// TODO: Get actual stats from API
const mockStats = {
totalTrials: trials.length,
completedTrials: trials.filter((t) => t.status === "completed").length,
averageDuration: "—",
successRate:
trials.length > 0
? `${Math.round((trials.filter((t) => t.status === "completed").length / trials.length) * 100)}%`
: "—",
};
return (
<EntityView>
{/* Header */}
<EntityViewHeader
title={experiment.name}
subtitle={experiment.description}
icon="FlaskConical"
status={{
label: statusInfo.label,
variant: statusInfo.variant,
icon: statusInfo.icon,
}}
actions={
canEdit ? (
<>
<Button asChild variant="outline">
<Link href={`/experiments/${experiment.id}/edit`}>
<Edit className="mr-2 h-4 w-4" />
Edit
</Link>
</Button>
<Button asChild variant="outline">
<Link href={`/experiments/${experiment.id}/designer`}>
<Settings className="mr-2 h-4 w-4" />
Designer
</Link>
</Button>
<Button asChild>
<Link href={`/trials/new?experimentId=${experiment.id}`}>
<Play className="mr-2 h-4 w-4" />
Start Trial
</Link>
</Button>
</>
) : (
<Button asChild>
<Link href={`/trials/new?experimentId=${experiment.id}`}>
<Play className="mr-2 h-4 w-4" />
Start Trial
</Link>
</Button>
)
}
/>
<div className="grid grid-cols-1 gap-8 lg:grid-cols-3">
{/* Main Content */}
<div className="space-y-8 lg:col-span-2">
{/* Experiment Information */}
<EntityViewSection title="Experiment Information" icon="FlaskConical">
<InfoGrid
items={[
{
label: "Study",
value: experiment.study ? (
<Link
href={`/studies/${experiment.study.id}`}
className="text-primary hover:underline"
>
{experiment.study.name}
</Link>
) : (
"No study assigned"
),
},
{
label: "Robot Platform",
value: experiment.robot?.name || "Not specified",
},
{
label: "Created",
value: formatDistanceToNow(experiment.createdAt, {
addSuffix: true,
}),
},
{
label: "Last Updated",
value: formatDistanceToNow(experiment.updatedAt, {
addSuffix: true,
}),
},
]}
/>
</EntityViewSection>
{/* Protocol Overview */}
<EntityViewSection
title="Protocol Overview"
icon="Target"
actions={
canEdit && (
<Button asChild variant="outline" size="sm">
<Link href={`/experiments/${experiment.id}/designer`}>
<Edit className="mr-2 h-4 w-4" />
Edit Protocol
</Link>
</Button>
)
}
>
{experiment.protocol &&
typeof experiment.protocol === "object" &&
experiment.protocol !== null ? (
<div className="space-y-3">
<div className="bg-muted rounded-lg p-4">
<h4 className="mb-2 font-medium">Protocol Structure</h4>
<p className="text-muted-foreground text-sm">
Visual protocol designed with{" "}
{Array.isArray((experiment.protocol as any).blocks)
? (experiment.protocol as any).blocks.length
: 0}{" "}
blocks
</p>
</div>
</div>
) : (
<EmptyState
icon="Target"
title="No Protocol Defined"
description="Use the experiment designer to create your protocol"
action={
canEdit && (
<Button asChild>
<Link href={`/experiments/${experiment.id}/designer`}>
Open Designer
</Link>
</Button>
)
}
/>
)}
</EntityViewSection>
{/* Recent Trials */}
<EntityViewSection
title="Recent Trials"
icon="Play"
description="Latest experimental sessions"
actions={
<Button asChild variant="outline" size="sm">
<Link href={`/trials/new?experimentId=${experiment.id}`}>
<Play className="mr-2 h-4 w-4" />
Start Trial
</Link>
</Button>
}
>
{trials.length > 0 ? (
<div className="space-y-3">
{trials.slice(0, 5).map((trial) => (
<div
key={trial.id}
className="hover:bg-muted/50 rounded-lg border p-4 transition-colors"
>
<div className="mb-2 flex items-center justify-between">
<Link
href={`/trials/${trial.id}`}
className="font-medium hover:underline"
>
Trial #{trial.id.slice(-6)}
</Link>
<Badge
variant={
trial.status === "completed"
? "default"
: trial.status === "in_progress"
? "secondary"
: trial.status === "failed"
? "destructive"
: "outline"
}
>
{trial.status.replace("_", " ")}
</Badge>
</div>
<div className="text-muted-foreground flex items-center gap-4 text-sm">
<span className="flex items-center gap-1">
<Calendar className="h-4 w-4" />
{trial.createdAt
? formatDistanceToNow(new Date(trial.createdAt), {
addSuffix: true,
})
: "Not scheduled"}
</span>
{trial.participant && (
<span className="flex items-center gap-1">
<Users className="h-4 w-4" />
{trial.participant.name ||
trial.participant.participantCode}
</span>
)}
</div>
</div>
))}
{trials.length > 5 && (
<div className="pt-2 text-center">
<Button variant="outline" size="sm" asChild>
<Link href={`/trials?experimentId=${experiment.id}`}>
View All Trials ({trials.length})
</Link>
</Button>
</div>
)}
</div>
) : (
<EmptyState
icon="Play"
title="No Trials Yet"
description="Start your first trial to begin collecting data"
action={
<Button asChild>
<Link href={`/trials/new?experimentId=${experiment.id}`}>
Start First Trial
</Link>
</Button>
}
/>
)}
</EntityViewSection>
</div>
{/* Sidebar */}
<EntityViewSidebar>
{/* Quick Stats */}
<EntityViewSection title="Statistics" icon="BarChart3">
<StatsGrid
stats={[
{
label: "Total Trials",
value: mockStats.totalTrials,
},
{
label: "Completed",
value: mockStats.completedTrials,
color: "success",
},
{
label: "Success Rate",
value: mockStats.successRate,
color: "success",
},
{
label: "Avg. Duration",
value: mockStats.averageDuration,
},
]}
/>
</EntityViewSection>
{/* Robot Information */}
{experiment.robot && (
<EntityViewSection title="Robot Platform" icon="Bot">
<InfoGrid
columns={1}
items={[
{
label: "Platform",
value: experiment.robot.name,
},
{
label: "Type",
value: experiment.robot.type || "Not specified",
},
{
label: "Connection",
value: experiment.robot.connectionType || "Not configured",
},
]}
/>
</EntityViewSection>
)}
{/* Quick Actions */}
<EntityViewSection title="Quick Actions" icon="Settings">
<QuickActions
actions={[
{
label: "View All Trials",
icon: "Play",
href: `/trials?experimentId=${experiment.id}`,
},
{
label: "Export Data",
icon: "Share",
href: `/experiments/${experiment.id}/export`,
},
...(canEdit
? [
{
label: "Edit Experiment",
icon: "Edit",
href: `/experiments/${experiment.id}/edit`,
},
{
label: "Protocol Designer",
icon: "Settings",
href: `/experiments/${experiment.id}/designer`,
},
]
: []),
]}
/>
</EntityViewSection>
</EntityViewSidebar>
</div>
</EntityView>
);
}

View File

@@ -1,8 +1,11 @@
"use client";
import { formatDistanceToNow } from "date-fns";
import {
AlertCircle,
ArrowLeft,
Calendar,
CheckCircle,
Edit,
FileText,
Mail,
@@ -10,21 +13,26 @@ import {
Shield,
Trash2,
Users,
XCircle,
} from "lucide-react";
import Link from "next/link";
import { notFound } from "next/navigation";
import { useEffect, useState } from "react";
import { Alert, AlertDescription } from "~/components/ui/alert";
import { Badge } from "~/components/ui/badge";
import { Button } from "~/components/ui/button";
import {
Card,
CardContent,
CardDescription,
CardHeader,
CardTitle,
} from "~/components/ui/card";
import { auth } from "~/server/auth";
import { api } from "~/trpc/server";
EntityView,
EntityViewHeader,
EntityViewSection,
EntityViewSidebar,
EmptyState,
InfoGrid,
QuickActions,
} from "~/components/ui/entity-view";
import { useBreadcrumbsEffect } from "~/components/ui/breadcrumb-provider";
import { useSession } from "next-auth/react";
import { api } from "~/trpc/react";
interface ParticipantDetailPageProps {
params: Promise<{
@@ -32,415 +40,372 @@ interface ParticipantDetailPageProps {
}>;
}
export default async function ParticipantDetailPage({
export default function ParticipantDetailPage({
params,
}: ParticipantDetailPageProps) {
const resolvedParams = await params;
const session = await auth();
const { data: session } = useSession();
const [participant, setParticipant] = useState<any>(null);
const [trials, setTrials] = useState<any[]>([]);
const [loading, setLoading] = useState(true);
const [resolvedParams, setResolvedParams] = useState<{ id: string } | null>(
null,
);
useEffect(() => {
async function resolveParams() {
const resolved = await params;
setResolvedParams(resolved);
}
resolveParams();
}, [params]);
const { data: participantData } = api.participants.get.useQuery(
{ id: resolvedParams?.id ?? "" },
{ enabled: !!resolvedParams?.id },
);
const { data: trialsData } = api.trials.list.useQuery(
{ participantId: resolvedParams?.id ?? "", limit: 10 },
{ enabled: !!resolvedParams?.id },
);
useEffect(() => {
if (participantData) {
setParticipant(participantData);
}
if (trialsData) {
setTrials(trialsData);
}
if (participantData !== undefined) {
setLoading(false);
}
}, [participantData, trialsData]);
// Set breadcrumbs
useBreadcrumbsEffect([
{ label: "Dashboard", href: "/dashboard" },
{ label: "Participants", href: "/participants" },
{
label: participant?.name || participant?.participantCode || "Participant",
},
]);
if (!session?.user) {
return notFound();
}
try {
const participant = await api.participants.get({ id: resolvedParams.id });
if (loading || !participant) {
return <div>Loading...</div>;
}
if (!participant) {
return notFound();
}
const userRole = session.user.roles?.[0]?.role ?? "observer";
const canEdit = ["administrator", "researcher"].includes(userRole);
const userRole = session.user.roles?.[0]?.role ?? "observer";
const canEdit = ["administrator", "researcher"].includes(userRole);
// canDelete removed - not used in component
return (
<EntityView>
{/* Header */}
<EntityViewHeader
title={participant.name ?? participant.participantCode}
subtitle={
participant.name
? `Code: ${participant.participantCode}`
: "Participant"
}
icon="Users"
actions={
canEdit && (
<>
<Button variant="outline" asChild>
<Link href={`/participants/${resolvedParams.id}/edit`}>
<Edit className="mr-2 h-4 w-4" />
Edit
</Link>
</Button>
<Button variant="destructive" size="sm">
<Trash2 className="mr-2 h-4 w-4" />
Delete
</Button>
</>
)
}
/>
// Get participant's trials
const trials = await api.trials.list({
participantId: resolvedParams.id,
limit: 10,
});
<div className="grid gap-6 lg:grid-cols-3">
{/* Main Content */}
<div className="space-y-6 lg:col-span-2">
{/* Participant Information */}
<EntityViewSection title="Participant Information" icon="FileText">
<InfoGrid
items={[
{
label: "Participant Code",
value: (
<code className="bg-muted rounded px-2 py-1 font-mono text-sm">
{participant.participantCode}
</code>
),
},
{
label: "Name",
value: participant.name || "Not provided",
},
{
label: "Email",
value: participant.email ? (
<div className="flex items-center gap-2">
<Mail className="h-4 w-4" />
<a
href={`mailto:${participant.email}`}
className="hover:underline"
>
{participant.email}
</a>
</div>
) : (
"Not provided"
),
},
{
label: "Study",
value: participant.study ? (
<Link
href={`/studies/${participant.study.id}`}
className="text-primary hover:underline"
>
{participant.study.name}
</Link>
) : (
"No study assigned"
),
},
]}
/>
return (
<div className="container mx-auto max-w-6xl px-4 py-8">
{/* Header */}
<div className="mb-8">
<div className="mb-4 flex items-center gap-4">
<Button variant="ghost" size="sm" asChild>
<Link href="/participants">
<ArrowLeft className="mr-2 h-4 w-4" />
Back to Participants
</Link>
</Button>
</div>
{/* Demographics */}
{participant.demographics &&
typeof participant.demographics === "object" &&
participant.demographics !== null &&
Object.keys(participant.demographics).length > 0 && (
<div className="border-t pt-4">
<h4 className="text-muted-foreground mb-3 text-sm font-medium">
Demographics
</h4>
<InfoGrid
items={(() => {
const demo = participant.demographics as Record<
string,
unknown
>;
return [
demo.age && {
label: "Age",
value:
typeof demo.age === "number"
? demo.age.toString()
: String(demo.age),
},
demo.gender && {
label: "Gender",
value: String(demo.gender),
},
].filter(Boolean) as Array<{
label: string;
value: string;
}>;
})()}
/>
</div>
)}
<div className="flex items-center justify-between">
<div className="flex items-center gap-4">
<div className="bg-primary text-primary-foreground flex h-16 w-16 items-center justify-center rounded-lg">
<Users className="h-8 w-8" />
</div>
<div>
<h1 className="text-foreground text-3xl font-bold">
{participant.name ?? participant.participantCode}
</h1>
<p className="text-muted-foreground text-lg">
{participant.name
? `Code: ${participant.participantCode}`
: "Participant"}
</p>
</div>
</div>
{canEdit && (
<div className="flex gap-2">
<Button variant="outline" asChild>
<Link href={`/participants/${resolvedParams.id}/edit`}>
<Edit className="mr-2 h-4 w-4" />
Edit
</Link>
</Button>
<Button variant="destructive" size="sm">
<Trash2 className="mr-2 h-4 w-4" />
Delete
</Button>
{/* Notes */}
{participant.notes && (
<div className="border-t pt-4">
<h4 className="text-muted-foreground mb-2 text-sm font-medium">
Notes
</h4>
<div className="bg-muted rounded p-3 text-sm whitespace-pre-wrap">
{participant.notes}
</div>
</div>
)}
</div>
</div>
</EntityViewSection>
<div className="grid gap-6 lg:grid-cols-3">
{/* Main Content */}
<div className="space-y-6 lg:col-span-2">
{/* Participant Information */}
<Card>
<CardHeader>
<CardTitle className="flex items-center gap-2">
<FileText className="h-5 w-5" />
Participant Information
</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<div className="grid gap-4 md:grid-cols-2">
<div>
<h4 className="text-muted-foreground text-sm font-medium">
Participant Code
</h4>
<p className="bg-muted rounded px-2 py-1 font-mono text-sm">
{participant.participantCode}
</p>
</div>
{participant.name && (
<div>
<h4 className="text-muted-foreground text-sm font-medium">
Name
</h4>
<p className="text-sm">{participant.name}</p>
</div>
)}
{participant.email && (
<div>
<h4 className="text-muted-foreground text-sm font-medium">
Email
</h4>
<p className="flex items-center gap-2 text-sm">
<Mail className="h-4 w-4" />
<a
href={`mailto:${participant.email}`}
className="hover:underline"
>
{participant.email}
</a>
</p>
</div>
)}
<div>
<h4 className="text-muted-foreground text-sm font-medium">
Study
</h4>
<p className="text-sm">
{/* Trial History */}
<EntityViewSection
title="Trial History"
icon="Play"
description="Experimental sessions for this participant"
actions={
canEdit && (
<Button size="sm" asChild>
<Link href={`/trials/new?participantId=${resolvedParams.id}`}>
Schedule Trial
</Link>
</Button>
)
}
>
{trials.length > 0 ? (
<div className="space-y-3">
{trials.map((trial) => (
<div
key={trial.id}
className="hover:bg-muted/50 rounded-lg border p-4 transition-colors"
>
<div className="mb-2 flex items-center justify-between">
<Link
href={`/studies/${participant.study?.id}`}
className="text-primary hover:underline"
href={`/trials/${trial.id}`}
className="font-medium hover:underline"
>
{participant.study?.name}
{trial.experiment?.name || "Trial"}
</Link>
</p>
</div>
</div>
{participant.demographics &&
typeof participant.demographics === "object" &&
participant.demographics !== null &&
Object.keys(participant.demographics).length > 0 ? (
<div className="border-t pt-4">
<h4 className="text-muted-foreground mb-2 text-sm font-medium">
Demographics
</h4>
<div className="grid gap-4 md:grid-cols-2">
{(() => {
const demo = participant.demographics as Record<
string,
unknown
>;
return (
<>
{demo.age && (
<div>
<span className="text-sm font-medium">
Age:
</span>{" "}
<span className="text-sm">
{typeof demo.age === "number"
? demo.age.toString()
: String(demo.age)}
</span>
</div>
)}
{demo.gender && (
<div>
<span className="text-sm font-medium">
Gender:
</span>{" "}
<span className="text-sm">
{String(demo.gender)}
</span>
</div>
)}
</>
);
})()}
<Badge
variant={
trial.status === "completed"
? "default"
: trial.status === "in_progress"
? "secondary"
: trial.status === "failed"
? "destructive"
: "outline"
}
>
{trial.status.replace("_", " ")}
</Badge>
</div>
<div className="text-muted-foreground flex items-center gap-4 text-sm">
<span className="flex items-center gap-1">
<Calendar className="h-4 w-4" />
{trial.createdAt
? formatDistanceToNow(new Date(trial.createdAt), {
addSuffix: true,
})
: "Not scheduled"}
</span>
{trial.duration && (
<span>{Math.round(trial.duration / 60)} minutes</span>
)}
</div>
</div>
) : null}
{/* Notes */}
{participant.notes && (
<div className="border-t pt-4">
<h4 className="text-muted-foreground mb-2 text-sm font-medium">
Notes
</h4>
<p className="bg-muted rounded p-3 text-sm whitespace-pre-wrap">
{participant.notes}
</p>
</div>
)}
</CardContent>
</Card>
{/* Trial History */}
<Card>
<CardHeader>
<div className="flex items-center justify-between">
<CardTitle className="flex items-center gap-2">
<Play className="h-5 w-5" />
Trial History
</CardTitle>
{canEdit && (
<Button size="sm" asChild>
))}
</div>
) : (
<EmptyState
icon="Play"
title="No Trials Yet"
description="This participant hasn't been assigned to any trials."
action={
canEdit && (
<Button asChild>
<Link
href={`/trials/new?participantId=${resolvedParams.id}`}
>
Schedule Trial
Schedule First Trial
</Link>
</Button>
)}
</div>
<CardDescription>
Experimental sessions for this participant
</CardDescription>
</CardHeader>
<CardContent>
{trials.length > 0 ? (
<div className="space-y-3">
{trials.map((trial) => (
<div
key={trial.id}
className="hover:bg-muted/50 rounded-lg border p-4 transition-colors"
>
<div className="mb-2 flex items-center justify-between">
<Link
href={`/trials/${trial.id}`}
className="font-medium hover:underline"
>
{trial.experiment?.name || "Trial"}
</Link>
<Badge
variant={
trial.status === "completed"
? "default"
: trial.status === "in_progress"
? "secondary"
: trial.status === "failed"
? "destructive"
: "outline"
}
>
{trial.status.replace("_", " ")}
</Badge>
</div>
<div className="text-muted-foreground flex items-center gap-4 text-sm">
<span className="flex items-center gap-1">
<Calendar className="h-4 w-4" />
{trial.createdAt
? formatDistanceToNow(new Date(trial.createdAt), {
addSuffix: true,
})
: "Not scheduled"}
</span>
{trial.duration && (
<span>
{Math.round(trial.duration / 60)} minutes
</span>
)}
</div>
</div>
))}
</div>
) : (
<div className="py-8 text-center">
<Play className="text-muted-foreground mx-auto mb-4 h-12 w-12" />
<h3 className="mb-2 font-medium">No Trials Yet</h3>
<p className="text-muted-foreground mb-4 text-sm">
This participant hasn&apos;t been assigned to any trials.
</p>
{canEdit && (
<Button asChild>
<Link
href={`/trials/new?participantId=${resolvedParams.id}`}
>
Schedule First Trial
</Link>
</Button>
)}
</div>
)}
</CardContent>
</Card>
</div>
{/* Sidebar */}
<div className="space-y-6">
{/* Consent Status */}
<Card>
<CardHeader>
<CardTitle className="flex items-center gap-2 text-base">
<Shield className="h-4 w-4" />
Consent Status
</CardTitle>
</CardHeader>
<CardContent>
<div className="space-y-3">
<div className="flex items-center justify-between">
<span className="text-sm">Informed Consent</span>
<Badge
variant={
participant.consentGiven ? "default" : "destructive"
}
>
{participant.consentGiven ? "Given" : "Not Given"}
</Badge>
</div>
{participant.consentDate && (
<div className="text-muted-foreground text-sm">
Consented:{" "}
{formatDistanceToNow(participant.consentDate, {
addSuffix: true,
})}
</div>
)}
{!participant.consentGiven && (
<Alert>
<AlertCircle className="h-4 w-4" />
<AlertDescription className="text-sm">
Consent required before trials can be conducted.
</AlertDescription>
</Alert>
)}
</div>
</CardContent>
</Card>
{/* Registration Details */}
<Card>
<CardHeader>
<CardTitle className="text-base">
Registration Details
</CardTitle>
</CardHeader>
<CardContent className="space-y-3">
<div>
<h4 className="text-muted-foreground text-sm font-medium">
Registered
</h4>
<p className="text-sm">
{formatDistanceToNow(participant.createdAt, {
addSuffix: true,
})}
</p>
</div>
{participant.updatedAt &&
participant.updatedAt !== participant.createdAt && (
<div>
<h4 className="text-muted-foreground text-sm font-medium">
Last Updated
</h4>
<p className="text-sm">
{formatDistanceToNow(participant.updatedAt, {
addSuffix: true,
})}
</p>
</div>
)}
</CardContent>
</Card>
{/* Quick Actions */}
{canEdit && (
<Card>
<CardHeader>
<CardTitle className="text-base">Quick Actions</CardTitle>
</CardHeader>
<CardContent className="space-y-2">
<Button
variant="outline"
className="w-full justify-start"
asChild
>
<Link
href={`/trials/new?participantId=${resolvedParams.id}`}
>
<Play className="mr-2 h-4 w-4" />
Schedule Trial
</Link>
</Button>
<Button
variant="outline"
className="w-full justify-start"
asChild
>
<Link href={`/participants/${resolvedParams.id}/edit`}>
<Edit className="mr-2 h-4 w-4" />
Edit Information
</Link>
</Button>
<Button variant="outline" className="w-full justify-start">
<FileText className="mr-2 h-4 w-4" />
Export Data
</Button>
</CardContent>
</Card>
)
}
/>
)}
</div>
</EntityViewSection>
</div>
{/* Sidebar */}
<EntityViewSidebar>
{/* Consent Status */}
<EntityViewSection title="Consent Status" icon="Shield">
<div className="space-y-3">
<div className="flex items-center justify-between">
<span className="text-sm">Informed Consent</span>
<Badge
variant={participant.consentGiven ? "default" : "destructive"}
>
{participant.consentGiven ? (
<>
<CheckCircle className="mr-1 h-3 w-3" />
Given
</>
) : (
<>
<XCircle className="mr-1 h-3 w-3" />
Not Given
</>
)}
</Badge>
</div>
{participant.consentDate && (
<div className="text-muted-foreground text-sm">
Consented:{" "}
{formatDistanceToNow(participant.consentDate, {
addSuffix: true,
})}
</div>
)}
{!participant.consentGiven && (
<Alert>
<AlertCircle className="h-4 w-4" />
<AlertDescription className="text-sm">
Consent required before trials can be conducted.
</AlertDescription>
</Alert>
)}
</div>
</EntityViewSection>
{/* Registration Details */}
<EntityViewSection title="Registration Details" icon="Calendar">
<InfoGrid
columns={1}
items={[
{
label: "Registered",
value: formatDistanceToNow(participant.createdAt, {
addSuffix: true,
}),
},
...(participant.updatedAt &&
participant.updatedAt !== participant.createdAt
? [
{
label: "Last Updated",
value: formatDistanceToNow(participant.updatedAt, {
addSuffix: true,
}),
},
]
: []),
]}
/>
</EntityViewSection>
{/* Quick Actions */}
{canEdit && (
<EntityViewSection title="Quick Actions" icon="Edit">
<QuickActions
actions={[
{
label: "Schedule Trial",
icon: "Play",
href: `/trials/new?participantId=${resolvedParams.id}`,
},
{
label: "Edit Information",
icon: "Edit",
href: `/participants/${resolvedParams.id}/edit`,
},
{
label: "Export Data",
icon: "FileText",
href: `/participants/${resolvedParams.id}/export`,
},
]}
/>
</EntityViewSection>
)}
</EntityViewSidebar>
</div>
);
} catch {
return notFound();
}
</EntityView>
);
}

View File

@@ -1,27 +1,41 @@
"use client";
import { formatDistanceToNow } from "date-fns";
import {
ArrowLeft,
BarChart3,
Building,
Calendar,
CheckCircle,
Clock,
Edit,
FileText,
FlaskConical,
Plus,
Settings,
Shield,
Users,
XCircle,
} from "lucide-react";
import Link from "next/link";
import { notFound } from "next/navigation";
import { useEffect, useState } from "react";
import { Badge } from "~/components/ui/badge";
import { Button } from "~/components/ui/button";
import {
Card,
CardContent,
CardDescription,
CardHeader,
CardTitle,
} from "~/components/ui/card";
EntityView,
EntityViewHeader,
EntityViewSection,
EntityViewSidebar,
EmptyState,
InfoGrid,
QuickActions,
StatsGrid,
} from "~/components/ui/entity-view";
import { Separator } from "~/components/ui/separator";
import { api } from "~/trpc/server";
import { useBreadcrumbsEffect } from "~/components/ui/breadcrumb-provider";
import { useSession } from "next-auth/react";
import { api } from "~/trpc/react";
interface StudyDetailPageProps {
params: Promise<{
@@ -32,326 +46,281 @@ interface StudyDetailPageProps {
const statusConfig = {
draft: {
label: "Draft",
className: "bg-gray-100 text-gray-800",
icon: "📝",
variant: "secondary" as const,
icon: "FileText" as const,
},
active: {
label: "Active",
className: "bg-green-100 text-green-800",
icon: "🟢",
variant: "default" as const,
icon: "CheckCircle" as const,
},
completed: {
label: "Completed",
className: "bg-blue-100 text-blue-800",
icon: "✅",
variant: "outline" as const,
icon: "CheckCircle" as const,
},
archived: {
label: "Archived",
className: "bg-orange-100 text-orange-800",
icon: "📦",
variant: "destructive" as const,
icon: "XCircle" as const,
},
};
export default async function StudyDetailPage({
params,
}: StudyDetailPageProps) {
try {
const resolvedParams = await params;
const study = await api.studies.get({ id: resolvedParams.id });
const members = await api.studies.getMembers({
studyId: resolvedParams.id,
});
export default function StudyDetailPage({ params }: StudyDetailPageProps) {
const { data: session } = useSession();
const [study, setStudy] = useState<any>(null);
const [members, setMembers] = useState<any[]>([]);
const [loading, setLoading] = useState(true);
const [resolvedParams, setResolvedParams] = useState<{ id: string } | null>(
null,
);
if (!study) {
notFound();
useEffect(() => {
async function resolveParams() {
const resolved = await params;
setResolvedParams(resolved);
}
resolveParams();
}, [params]);
const statusInfo = statusConfig[study.status];
const { data: studyData } = api.studies.get.useQuery(
{ id: resolvedParams?.id ?? "" },
{ enabled: !!resolvedParams?.id },
);
return (
<div className="p-8">
{/* Header */}
<div className="mb-8">
<div className="mb-4 flex items-center space-x-2 text-sm text-slate-600">
<Link href="/studies" className="hover:text-slate-900">
Studies
</Link>
<span>/</span>
<span className="text-slate-900">{study.name}</span>
</div>
const { data: membersData } = api.studies.getMembers.useQuery(
{ studyId: resolvedParams?.id ?? "" },
{ enabled: !!resolvedParams?.id },
);
<div className="flex items-start justify-between">
<div className="min-w-0 flex-1">
<div className="mb-2 flex items-center space-x-3">
<h1 className="truncate text-3xl font-bold text-slate-900">
{study.name}
</h1>
<Badge className={statusInfo.className} variant="secondary">
<span className="mr-1">{statusInfo.icon}</span>
{statusInfo.label}
</Badge>
</div>
<p className="text-lg text-slate-600">{study.description}</p>
</div>
useEffect(() => {
if (studyData) {
setStudy(studyData);
}
if (membersData) {
setMembers(membersData);
}
if (studyData !== undefined) {
setLoading(false);
}
}, [studyData, membersData]);
<div className="ml-4 flex items-center space-x-2">
<Button asChild variant="outline">
<Link href={`/studies/${study.id}/edit`}>
<Settings className="mr-2 h-4 w-4" />
Edit Study
</Link>
</Button>
<Button asChild>
<Link href={`/studies/${study.id}/experiments/new`}>
<Plus className="mr-2 h-4 w-4" />
New Experiment
</Link>
</Button>
</div>
</div>
</div>
// Set breadcrumbs
useBreadcrumbsEffect([
{ label: "Dashboard", href: "/dashboard" },
{ label: "Studies", href: "/studies" },
{ label: study?.name || "Study" },
]);
<div className="grid grid-cols-1 gap-8 lg:grid-cols-3">
{/* Main Content */}
<div className="space-y-8 lg:col-span-2">
{/* Study Information */}
<Card>
<CardHeader>
<CardTitle className="flex items-center space-x-2">
<Building className="h-5 w-5" />
<span>Study Information</span>
</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<div className="grid grid-cols-1 gap-4 md:grid-cols-2">
<div>
<label className="text-sm font-medium text-slate-700">
Institution
</label>
<p className="text-slate-900">{study.institution}</p>
</div>
{study.irbProtocol && (
<div>
<label className="text-sm font-medium text-slate-700">
IRB Protocol
</label>
<p className="text-slate-900">{study.irbProtocol}</p>
</div>
)}
<div>
<label className="text-sm font-medium text-slate-700">
Created
</label>
<p className="text-slate-900">
{formatDistanceToNow(study.createdAt, {
addSuffix: true,
})}
</p>
</div>
<div>
<label className="text-sm font-medium text-slate-700">
Last Updated
</label>
<p className="text-slate-900">
{formatDistanceToNow(study.updatedAt, {
addSuffix: true,
})}
</p>
</div>
</div>
</CardContent>
</Card>
{/* Experiments */}
<Card>
<CardHeader>
<div className="flex items-center justify-between">
<CardTitle className="flex items-center space-x-2">
<FlaskConical className="h-5 w-5" />
<span>Experiments</span>
</CardTitle>
<Button asChild variant="outline" size="sm">
<Link href={`/studies/${study.id}/experiments/new`}>
<Plus className="mr-2 h-4 w-4" />
Add Experiment
</Link>
</Button>
</div>
<CardDescription>
Design and manage experimental protocols for this study
</CardDescription>
</CardHeader>
<CardContent>
{/* Placeholder for experiments list */}
<div className="py-8 text-center">
<FlaskConical className="mx-auto mb-4 h-12 w-12 text-slate-400" />
<h3 className="mb-2 text-lg font-semibold text-slate-900">
No Experiments Yet
</h3>
<p className="mb-4 text-slate-600">
Create your first experiment to start designing research
protocols
</p>
<Button asChild>
<Link href={`/studies/${study.id}/experiments/new`}>
Create First Experiment
</Link>
</Button>
</div>
</CardContent>
</Card>
{/* Recent Activity */}
<Card>
<CardHeader>
<CardTitle className="flex items-center space-x-2">
<BarChart3 className="h-5 w-5" />
<span>Recent Activity</span>
</CardTitle>
</CardHeader>
<CardContent>
<div className="py-8 text-center">
<Calendar className="mx-auto mb-4 h-12 w-12 text-slate-400" />
<h3 className="mb-2 text-lg font-semibold text-slate-900">
No Recent Activity
</h3>
<p className="text-slate-600">
Activity will appear here once you start working on this
study
</p>
</div>
</CardContent>
</Card>
</div>
{/* Sidebar */}
<div className="space-y-6">
{/* Team Members */}
<Card>
<CardHeader>
<div className="flex items-center justify-between">
<CardTitle className="flex items-center space-x-2">
<Users className="h-5 w-5" />
<span>Team</span>
</CardTitle>
<Button variant="outline" size="sm">
<Plus className="mr-2 h-4 w-4" />
Invite
</Button>
</div>
<CardDescription>
{members.length} team member{members.length !== 1 ? "s" : ""}
</CardDescription>
</CardHeader>
<CardContent>
<div className="space-y-3">
{members.map((member) => (
<div
key={member.user.id}
className="flex items-center space-x-3"
>
<div className="flex h-8 w-8 items-center justify-center rounded-full bg-blue-100">
<span className="text-sm font-medium text-blue-600">
{(member.user.name ?? member.user.email)
.charAt(0)
.toUpperCase()}
</span>
</div>
<div className="min-w-0 flex-1">
<p className="truncate text-sm font-medium text-slate-900">
{member.user.name ?? member.user.email}
</p>
<p className="text-xs text-slate-500 capitalize">
{member.role}
</p>
</div>
{member.role === "owner" && (
<Shield className="h-4 w-4 text-amber-600" />
)}
</div>
))}
</div>
</CardContent>
</Card>
{/* Quick Stats */}
<Card>
<CardHeader>
<CardTitle>Quick Stats</CardTitle>
</CardHeader>
<CardContent>
<div className="space-y-3">
<div className="flex justify-between">
<span className="text-sm text-slate-600">Experiments:</span>
<span className="font-medium">0</span>
</div>
<div className="flex justify-between">
<span className="text-sm text-slate-600">
Total Trials:
</span>
<span className="font-medium">0</span>
</div>
<div className="flex justify-between">
<span className="text-sm text-slate-600">
Participants:
</span>
<span className="font-medium">0</span>
</div>
<Separator />
<div className="flex justify-between">
<span className="text-sm text-slate-600">
Completion Rate:
</span>
<span className="font-medium text-green-600"></span>
</div>
</div>
</CardContent>
</Card>
{/* Quick Actions */}
<Card>
<CardHeader>
<CardTitle>Quick Actions</CardTitle>
</CardHeader>
<CardContent className="space-y-2">
<Button
asChild
variant="outline"
className="w-full justify-start"
>
<Link href={`/studies/${study.id}/participants`}>
<Users className="mr-2 h-4 w-4" />
Manage Participants
</Link>
</Button>
<Button
asChild
variant="outline"
className="w-full justify-start"
>
<Link href={`/studies/${study.id}/trials`}>
<Calendar className="mr-2 h-4 w-4" />
Schedule Trials
</Link>
</Button>
<Button
asChild
variant="outline"
className="w-full justify-start"
>
<Link href={`/studies/${study.id}/analytics`}>
<BarChart3 className="mr-2 h-4 w-4" />
View Analytics
</Link>
</Button>
</CardContent>
</Card>
</div>
</div>
</div>
);
} catch (error) {
console.error("Error loading study:", error);
notFound();
if (!session?.user) {
return notFound();
}
if (loading || !study) {
return <div>Loading...</div>;
}
const statusInfo = statusConfig[study.status];
// TODO: Get actual stats from API
const mockStats = {
experiments: 0,
totalTrials: 0,
participants: 0,
completionRate: "—",
};
return (
<EntityView>
{/* Header */}
<EntityViewHeader
title={study.name}
subtitle={study.description}
icon="Building"
status={{
label: statusInfo.label,
variant: statusInfo.variant,
icon: statusInfo.icon,
}}
actions={
<>
<Button asChild variant="outline">
<Link href={`/studies/${study.id}/edit`}>
<Settings className="mr-2 h-4 w-4" />
Edit Study
</Link>
</Button>
<Button asChild>
<Link href={`/experiments/new?studyId=${study.id}`}>
<Plus className="mr-2 h-4 w-4" />
New Experiment
</Link>
</Button>
</>
}
/>
<div className="grid grid-cols-1 gap-8 lg:grid-cols-3">
{/* Main Content */}
<div className="space-y-8 lg:col-span-2">
{/* Study Information */}
<EntityViewSection title="Study Information" icon="Building">
<InfoGrid
items={[
{
label: "Institution",
value: study.institution,
},
{
label: "IRB Protocol",
value: study.irbProtocol || "Not specified",
},
{
label: "Created",
value: formatDistanceToNow(study.createdAt, {
addSuffix: true,
}),
},
{
label: "Last Updated",
value: formatDistanceToNow(study.updatedAt, {
addSuffix: true,
}),
},
]}
/>
</EntityViewSection>
{/* Experiments */}
<EntityViewSection
title="Experiments"
icon="FlaskConical"
description="Design and manage experimental protocols for this study"
actions={
<Button asChild variant="outline" size="sm">
<Link href={`/experiments/new?studyId=${study.id}`}>
<Plus className="mr-2 h-4 w-4" />
Add Experiment
</Link>
</Button>
}
>
<EmptyState
icon="FlaskConical"
title="No Experiments Yet"
description="Create your first experiment to start designing research protocols"
action={
<Button asChild>
<Link href={`/experiments/new?studyId=${study.id}`}>
Create First Experiment
</Link>
</Button>
}
/>
</EntityViewSection>
{/* Recent Activity */}
<EntityViewSection title="Recent Activity" icon="BarChart3">
<EmptyState
icon="Calendar"
title="No Recent Activity"
description="Activity will appear here once you start working on this study"
/>
</EntityViewSection>
</div>
{/* Sidebar */}
<EntityViewSidebar>
{/* Team Members */}
<EntityViewSection
title="Team"
icon="Users"
description={`${members.length} team member${members.length !== 1 ? "s" : ""}`}
actions={
<Button variant="outline" size="sm">
<Plus className="mr-2 h-4 w-4" />
Invite
</Button>
}
>
<div className="space-y-3">
{members.map((member) => (
<div
key={member.user.id}
className="flex items-center space-x-3"
>
<div className="flex h-8 w-8 items-center justify-center rounded-full bg-blue-100">
<span className="text-sm font-medium text-blue-600">
{(member.user.name ?? member.user.email)
.charAt(0)
.toUpperCase()}
</span>
</div>
<div className="min-w-0 flex-1">
<p className="truncate text-sm font-medium">
{member.user.name ?? member.user.email}
</p>
<p className="text-muted-foreground text-xs capitalize">
{member.role}
</p>
</div>
{member.role === "owner" && (
<Shield className="h-4 w-4 text-amber-600" />
)}
</div>
))}
</div>
</EntityViewSection>
{/* Quick Stats */}
<EntityViewSection title="Quick Stats" icon="BarChart3">
<StatsGrid
stats={[
{
label: "Experiments",
value: mockStats.experiments,
},
{
label: "Total Trials",
value: mockStats.totalTrials,
},
{
label: "Participants",
value: mockStats.participants,
},
{
label: "Completion Rate",
value: mockStats.completionRate,
color: "success",
},
]}
/>
</EntityViewSection>
{/* Quick Actions */}
<EntityViewSection title="Quick Actions" icon="Settings">
<QuickActions
actions={[
{
label: "Manage Participants",
icon: "Users",
href: `/participants?studyId=${study.id}`,
},
{
label: "Schedule Trials",
icon: "Calendar",
href: `/trials?studyId=${study.id}`,
},
{
label: "View Analytics",
icon: "BarChart3",
href: `/analytics?studyId=${study.id}`,
},
]}
/>
</EntityViewSection>
</EntityViewSidebar>
</div>
</EntityView>
);
}

File diff suppressed because it is too large Load Diff