import { CalendarDays, Globe, Tag, Clock, AlarmClock } from "lucide-react"; import { Conference } from "@/types/conference"; import { formatDistanceToNow, parseISO, isValid, isPast } from "date-fns"; import ConferenceDialog from "./ConferenceDialog"; import { useState } from "react"; import { getDeadlineInLocalTime } from '@/utils/dateUtils'; const ConferenceCard = ({ title, full_name, year, date, deadline, timezone, tags = [], link, note, abstract_deadline, city, country, venue, ...conferenceProps }: Conference) => { const [dialogOpen, setDialogOpen] = useState(false); const deadlineDate = getDeadlineInLocalTime(deadline, timezone); // Add validation before using formatDistanceToNow const getTimeRemaining = () => { if (!deadlineDate || !isValid(deadlineDate)) { return 'TBD'; } if (isPast(deadlineDate)) { return 'Deadline passed'; } try { return formatDistanceToNow(deadlineDate, { addSuffix: true }); } catch (error) { console.error('Error formatting time remaining:', error); return 'Invalid date'; } }; const timeRemaining = getTimeRemaining(); // Create location string by concatenating city and country const location = [city, country].filter(Boolean).join(", "); // Determine countdown color based on days remaining const getCountdownColor = () => { if (!deadlineDate || !isValid(deadlineDate)) return "text-neutral-600"; try { const daysRemaining = Math.ceil((deadlineDate.getTime() - new Date().getTime()) / (1000 * 60 * 60 * 24)); if (daysRemaining <= 7) return "text-red-600"; if (daysRemaining <= 30) return "text-orange-600"; return "text-green-600"; } catch (error) { console.error('Error calculating countdown color:', error); return "text-neutral-600"; } }; const handleCardClick = (e: React.MouseEvent) => { if (!(e.target as HTMLElement).closest('a') && !(e.target as HTMLElement).closest('.tag-button')) { setDialogOpen(true); } }; const handleTagClick = (e: React.MouseEvent, tag: string) => { e.stopPropagation(); const searchParams = new URLSearchParams(window.location.search); const currentTags = searchParams.get('tags')?.split(',') || []; let newTags; if (currentTags.includes(tag)) { newTags = currentTags.filter(t => t !== tag); } else { newTags = [...currentTags, tag]; } if (newTags.length > 0) { searchParams.set('tags', newTags.join(',')); } else { searchParams.delete('tags'); } const newUrl = `${window.location.pathname}${searchParams.toString() ? `?${searchParams.toString()}` : ''}`; window.history.pushState({}, '', newUrl); window.dispatchEvent(new CustomEvent('urlchange', { detail: { tag } })); }; return ( <>

{title} {year}

{link && ( e.stopPropagation()} > )}
{date}
{location && (
{location}
)}
{deadline === 'TBD' ? 'TBD' : deadline}
{timeRemaining}
{Array.isArray(tags) && tags.length > 0 && (
{tags.map((tag) => ( ))}
)}
); }; export default ConferenceCard;