Files
portfolio/src/components/PMRInterface.tsx
T
admin 4104dd32d8 Task 12: Build ReferralsView with clinical referral form
- Created ReferralsView component with clinical referral form UI
- Pre-filled patient info (CHARLWOOD, Andrew; NHS Number)
- Priority toggle: Urgent/Routine/Two-Week Wait with tooltips
- Form fields: Referrer Name/Email/Org, Reason textarea
- Contact method radio: Email/Phone/LinkedIn
- Form validation for required fields
- Loading state with spinner on submit
- Success state with REF-YYYY-MMDD-NNN reference number
- Direct Contact table with clickable email/phone/LinkedIn links
- Responsive two-column layout for form fields
- Consistent clinical system styling with NHS blue accents
2026-02-11 02:27:41 +00:00

88 lines
2.6 KiB
TypeScript

import { useState } from 'react'
import type { ViewId } from '../types/pmr'
import { ClinicalSidebar } from './ClinicalSidebar'
import { PatientBanner } from './PatientBanner'
import { SummaryView } from './views/SummaryView'
import { ConsultationsView } from './views/ConsultationsView'
import { MedicationsView } from './views/MedicationsView'
import { ProblemsView } from './views/ProblemsView'
import { InvestigationsView } from './views/InvestigationsView'
import { DocumentsView } from './views/DocumentsView'
import { ReferralsView } from './views/ReferralsView'
interface PMRInterfaceProps {
children?: React.ReactNode
}
export function PMRInterface({ children }: PMRInterfaceProps) {
const [activeView, setActiveView] = useState<ViewId>(() => {
const hash = window.location.hash.slice(1) as ViewId
const validViews: ViewId[] = [
'summary',
'consultations',
'medications',
'problems',
'investigations',
'documents',
'referrals',
]
return validViews.includes(hash) ? hash : 'summary'
})
const handleViewChange = (view: ViewId) => {
setActiveView(view)
}
const handleNavigate = (view: ViewId, itemId?: string) => {
void itemId
setActiveView(view)
window.location.hash = view
}
const renderView = () => {
switch (activeView) {
case 'summary':
return <SummaryView onNavigate={handleNavigate} />
case 'consultations':
return <ConsultationsView />
case 'medications':
return <MedicationsView />
case 'problems':
return <ProblemsView onNavigate={handleNavigate} />
case 'investigations':
return <InvestigationsView />
case 'documents':
return <DocumentsView />
case 'referrals':
return <ReferralsView />
default:
return (
<div className="bg-white border border-gray-200 rounded p-6">
<h1 className="font-inter font-semibold text-lg text-gray-900 capitalize">
{activeView} View
</h1>
<p className="font-inter text-sm text-gray-500 mt-2">
Content for {activeView} will be implemented in a separate task.
</p>
</div>
)
}
}
return (
<div className="min-h-screen bg-pmr-content">
<PatientBanner />
<div className="flex">
<ClinicalSidebar activeView={activeView} onViewChange={handleViewChange} />
<main
role="main"
aria-label={`${activeView} view`}
className="flex-1 min-h-[calc(100vh-80px)] p-6"
>
{children || renderView()}
</main>
</div>
</div>
)
}