first commit

This commit is contained in:
2025-10-09 20:05:39 +05:30
commit d4fcb658e3
69 changed files with 13582 additions and 0 deletions

View File

@ -0,0 +1,286 @@
'use client'
import React, { useEffect, useState } from 'react';
import Link from 'next/link';
import { ChevronRight } from 'lucide-react';
import { FacultyService, TeamMember } from '../../lib/facultyData';
interface TeamListingProps {
title?: string;
onMemberClick?: (member: TeamMember) => void;
}
const TeamListing: React.FC<TeamListingProps> = ({
title = "Our Faculty",
onMemberClick
}) => {
const [teamMembers, setTeamMembers] = useState<TeamMember[]>([]);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
useEffect(() => {
const loadFacultyData = async () => {
try {
setLoading(true);
const faculty = await FacultyService.getAllFaculty();
setTeamMembers(faculty);
setError(null);
} catch (err) {
console.error('Failed to load faculty data:', err);
setError('Failed to load faculty data. Please try again later.');
} finally {
setLoading(false);
}
};
loadFacultyData();
}, []);
// Filter members by category
const facultyMembers = teamMembers.filter(member => member.category === 'FACULTY');
const supportTeam = teamMembers.filter(member => member.category === 'SUPPORT_TEAM');
const traineesAndFellows = teamMembers.filter(member => member.category === 'TRAINEE_FELLOW');
const handleMemberClick = (member: TeamMember) => {
if (onMemberClick) {
onMemberClick(member);
} else {
window.location.href = `/faculty/${member.id}`;
}
};
if (loading) {
return (
<div className="min-h-screen flex items-center justify-center">
<div className="text-center">
<div className="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-600 mx-auto mb-4"></div>
<p>Loading faculty data...</p>
</div>
</div>
);
}
if (error) {
return (
<div className="min-h-screen flex items-center justify-center">
<div className="text-center">
<p className="text-red-600 mb-4">{error}</p>
<button
onClick={() => window.location.reload()}
className="px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700"
>
Retry
</button>
</div>
</div>
);
}
const TeamMemberCard: React.FC<{ member: TeamMember }> = ({ member }) => {
const [imageError, setImageError] = useState(false);
const [imageLoading, setImageLoading] = useState(true);
const handleImageError = (e: React.SyntheticEvent<HTMLImageElement>) => {
const target = e.target as HTMLImageElement;
console.error('Image failed to load:', member.image);
console.error('Member:', member.name, 'Professor ID:', member.professorId);
if (!imageError) {
setImageError(true);
target.src = '/images/default-avatar.jpg';
}
};
const handleImageLoad = () => {
console.log('Image loaded successfully:', member.image);
setImageLoading(false);
};
return (
<div
className="group cursor-pointer bg-white rounded-lg border border-gray-300 overflow-hidden hover:shadow-lg transition-all duration-300"
onClick={() => handleMemberClick(member)}
>
<div className="relative aspect-square overflow-hidden">
{imageLoading && (
<div className="absolute inset-0 flex items-center justify-center bg-gray-100">
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600"></div>
</div>
)}
<img
src={member.image}
alt={member.name}
className={`w-full h-full object-cover transition-transform duration-300 group-hover:scale-105 ${
imageLoading ? 'opacity-0' : 'opacity-100'
}`}
onError={handleImageError}
onLoad={handleImageLoad}
loading="lazy"
/>
{imageError && (
<div className="absolute top-2 right-2 bg-red-500 text-white text-xs px-2 py-1 rounded">
Default
</div>
)}
</div>
<div className="p-4 sm:p-6">
<h3 className="text-lg font-medium mb-2 group-hover:opacity-70 transition-opacity" style={{ color: '#012068' }}>
{member.name}
</h3>
<p className="text-sm leading-relaxed" style={{ color: '#333' }}>
{member.position}
</p>
{member.department && (
<p className="text-xs mt-1" style={{ color: '#666' }}>
{member.department}
</p>
)}
{member.specialty && (
<p className="text-xs mt-1 font-medium" style={{ color: '#e64838' }}>
{member.specialty}
</p>
)}
</div>
</div>
);
};
return (
<div className="min-h-screen">
{/* Breadcrumb Section */}
<section className="py-4" style={{ backgroundColor: '#f4f4f4' }}>
<div className="max-w-7xl mx-auto px-4">
<nav className="flex items-center space-x-2 text-sm">
<Link
href="/"
className="hover:opacity-70 transition-opacity duration-200"
style={{ color: '#012068' }}
>
Home
</Link>
<ChevronRight className="w-4 h-4" style={{ color: '#012068' }} />
<span className="font-medium" style={{ color: '#e64838' }}>
Faculty
</span>
</nav>
<div className="mt-6">
<div className="flex items-center mb-4">
<h1 className="text-3xl font-bold" style={{ color: '#012068' }}>
{title}
</h1>
</div>
<p className="text-base max-w-3xl leading-relaxed" style={{ color: '#333' }}>
Meet our dedicated team of medical professionals, researchers, and support staff committed to advancing healthcare, education, and patient outcomes at Christian Medical College, Vellore
</p>
</div>
</div>
</section>
{/* Faculty Section */}
{facultyMembers.length > 0 && (
<section className="py-8" style={{ backgroundColor: '#fff' }}>
<div className="max-w-7xl mx-auto px-4">
<div className="mb-6">
<h2 className="text-2xl font-bold mb-2" style={{ color: '#012068' }}>
Faculty Members
</h2>
<p className="text-sm" style={{ color: '#666' }}>
Our experienced faculty members leading medical education and research
</p>
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-6">
{facultyMembers.map((member) => (
<TeamMemberCard key={member.id} member={member} />
))}
</div>
</div>
</section>
)}
{/* Support Team Section */}
{supportTeam.length > 0 && (
<section className="py-8" style={{ backgroundColor: '#f9f9f9' }}>
<div className="max-w-7xl mx-auto px-4">
<div className="mb-8">
<h2 className="text-2xl font-bold mb-4" style={{ color: '#012068' }}>
Support Team
</h2>
<p className="text-base leading-relaxed mb-2" style={{ color: '#012068' }}>
<strong>Clinical Support Staff</strong> - Essential team members providing specialized support services
</p>
<p className="text-sm" style={{ color: '#012068' }}>
<strong>Administrative & Technical Support</strong> - Dedicated professionals ensuring smooth operations
</p>
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-6">
{supportTeam.map((member) => (
<TeamMemberCard key={member.id} member={member} />
))}
</div>
</div>
</section>
)}
{/* Trainees & Fellows Section */}
{traineesAndFellows.length > 0 && (
<section className="py-8" style={{ backgroundColor: '#fff' }}>
<div className="max-w-7xl mx-auto px-4">
<div className="mb-8">
<h2 className="text-2xl font-bold mb-4" style={{ color: '#012068' }}>
Trainees & Fellows
</h2>
<p className="text-sm" style={{ color: '#666' }}>
Medical trainees, residents, and fellows advancing their skills and contributing to patient care
</p>
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-6">
{traineesAndFellows.map((member) => (
<TeamMemberCard key={member.id} member={member} />
))}
</div>
</div>
</section>
)}
{/* Show message if no faculty data */}
{teamMembers.length === 0 && !loading && (
<section className="py-16 text-center">
<div className="max-w-2xl mx-auto px-4">
<h2 className="text-2xl font-bold mb-4" style={{ color: '#012068' }}>
No Faculty Data Available
</h2>
<p className="text-gray-600 mb-6">
Faculty information is currently being updated. Please check back later or contact administration.
</p>
<button
onClick={() => window.location.reload()}
className="px-6 py-3 bg-blue-600 text-white rounded-lg hover:bg-blue-700 transition-colors"
>
Refresh Page
</button>
</div>
</section>
)}
{/* Show message if all categories are empty but we have data */}
{teamMembers.length > 0 && facultyMembers.length === 0 && supportTeam.length === 0 && traineesAndFellows.length === 0 && (
<section className="py-16 text-center">
<div className="max-w-2xl mx-auto px-4">
<h2 className="text-2xl font-bold mb-4" style={{ color: '#012068' }}>
No Categorized Faculty Available
</h2>
<p className="text-gray-600 mb-6">
Faculty members need to be assigned to categories. Please contact administration.
</p>
</div>
</section>
)}
</div>
);
};
export default TeamListing;