Spaces:
Running
Running
File size: 11,726 Bytes
f46336a 38de65f 6decbb1 08d905b f5b9300 08d905b 929b27b 255baa6 de1321f 08d905b 38de65f 6decbb1 38de65f 08d905b 38de65f 6decbb1 08d905b de1321f 08d905b 38de65f 6decbb1 08d905b 38de65f 6decbb1 929b27b 6decbb1 08d905b 38de65f 1685b02 08d905b 1685b02 08d905b 1685b02 08d905b 1685b02 08d905b 1685b02 08d905b 1685b02 38de65f 255baa6 f46336a 0edd5f0 1b35277 0edd5f0 6decbb1 07ed8c2 08d905b 1b35277 08d905b 1b35277 08d905b f5b9300 08d905b f5b9300 08d905b f5b9300 08d905b 6decbb1 f46336a 1b35277 f46336a 38de65f f46336a 255baa6 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 |
import Header from "@/components/Header";
import FilterBar from "@/components/FilterBar";
import ConferenceCard from "@/components/ConferenceCard";
import conferencesData from "@/data/conferences.yml";
import { Conference } from "@/types/conference";
import { useState, useMemo, useEffect } from "react";
import { Switch } from "@/components/ui/switch"
import { parseISO, isValid, isPast } from "date-fns";
import { extractCountry } from "@/utils/countryExtractor";
import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover";
import { Button } from "@/components/ui/button";
import { Checkbox } from "@/components/ui/checkbox";
import { X, ChevronRight, Filter, Globe } from "lucide-react";
import { getAllCountries } from "@/utils/countryExtractor";
import { getDeadlineInLocalTime } from "@/utils/dateUtils";
import { sortConferencesByDeadline } from "@/utils/conferenceUtils";
const Index = () => {
const [selectedTags, setSelectedTags] = useState<Set<string>>(new Set());
const [selectedCountries, setSelectedCountries] = useState<Set<string>>(new Set());
const [searchQuery, setSearchQuery] = useState("");
const [showPastConferences, setShowPastConferences] = useState(false);
// Category buttons configuration
const categoryButtons = [
{ id: "machine-learning", label: "Machine Learning" },
{ id: "lifelong-learning", label: "Lifelong Learning" },
{ id: "robotics", label: "Robotics" },
{ id: "computer-vision", label: "Computer Vision" },
{ id: "web-search", label: "Web Search" },
{ id: "data-mining", label: "Data Mining" },
{ id: "natural-language-processing", label: "Natural Language Processing" },
{ id: "signal-processing", label: "Signal Processing" },
{ id: "human-computer-interaction", label: "Human Computer Interaction" },
{ id: "computer-graphics", label: "Computer Graphics" },
{ id: "mathematics", label: "Mathematics" },
{ id: "reinforcement-learning", label: "Reinforcement Learning" },
];
const filteredConferences = useMemo(() => {
if (!Array.isArray(conferencesData)) {
console.error("Conferences data is not an array:", conferencesData);
return [];
}
return conferencesData
.filter((conf: Conference) => {
// Filter by deadline (past/future)
const deadlineDate = conf.deadline && conf.deadline !== 'TBD' ? parseISO(conf.deadline) : null;
const isUpcoming = !deadlineDate || !isValid(deadlineDate) || !isPast(deadlineDate);
if (!showPastConferences && !isUpcoming) return false;
// Filter by tags
const matchesTags = selectedTags.size === 0 ||
(Array.isArray(conf.tags) && conf.tags.some(tag => selectedTags.has(tag)));
// Filter by countries
const matchesCountry = selectedCountries.size === 0 ||
(conf.country && selectedCountries.has(conf.country));
// Filter by search query
const matchesSearch = searchQuery === "" ||
conf.title.toLowerCase().includes(searchQuery.toLowerCase()) ||
(conf.full_name && conf.full_name.toLowerCase().includes(searchQuery.toLowerCase()));
return matchesTags && matchesCountry && matchesSearch;
})
.sort((a: Conference, b: Conference) => {
const aDeadline = getDeadlineInLocalTime(a.deadline, a.timezone);
const bDeadline = getDeadlineInLocalTime(b.deadline, b.timezone);
if (aDeadline && bDeadline) {
return aDeadline.getTime() - bDeadline.getTime();
}
// Handle cases where one or both deadlines are invalid
if (!aDeadline && !bDeadline) return 0;
if (!aDeadline) return 1;
if (!bDeadline) return -1;
return 0;
});
}, [selectedTags, selectedCountries, searchQuery, showPastConferences]);
// Update handleTagsChange to handle multiple tags
const handleTagsChange = (newTags: Set<string>) => {
setSelectedTags(newTags);
const searchParams = new URLSearchParams(window.location.search);
if (newTags.size > 0) {
searchParams.set('tags', Array.from(newTags).join(','));
} else {
searchParams.delete('tags');
}
window.history.replaceState({}, '', `${window.location.pathname}?${searchParams}`);
};
const handleCountriesChange = (newCountries: Set<string>) => {
setSelectedCountries(newCountries);
const searchParams = new URLSearchParams(window.location.search);
if (newCountries.size > 0) {
searchParams.set('countries', Array.from(newCountries).join(','));
} else {
searchParams.delete('countries');
}
window.history.replaceState({}, '', `${window.location.pathname}?${searchParams}`);
};
// Toggle a single tag
const toggleTag = (tag: string) => {
const newTags = new Set(selectedTags);
if (newTags.has(tag)) {
newTags.delete(tag);
} else {
newTags.add(tag);
}
handleTagsChange(newTags);
};
// Load filters from URL on initial render
useEffect(() => {
const searchParams = new URLSearchParams(window.location.search);
const tagsParam = searchParams.get('tags');
const countriesParam = searchParams.get('countries');
if (tagsParam) {
const tags = tagsParam.split(',');
setSelectedTags(new Set(tags));
}
if (countriesParam) {
const countries = countriesParam.split(',');
setSelectedCountries(new Set(countries));
}
}, []);
if (!Array.isArray(conferencesData)) {
return <div>Loading conferences...</div>;
}
return (
<div className="min-h-screen bg-neutral-light">
<Header
onSearch={setSearchQuery}
showEmptyMessage={false}
/>
<div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
<div className="space-y-4 py-4">
{/* Category filter buttons */}
<div className="bg-white shadow rounded-lg p-4">
<div className="flex flex-wrap gap-2">
{categoryButtons.map(category => (
<button
key={category.id}
className={`px-4 py-2 rounded-md text-sm font-medium transition-colors ${
selectedTags.has(category.id)
? 'bg-blue-100 text-blue-800 hover:bg-blue-200'
: 'bg-neutral-100 text-neutral-700 hover:bg-neutral-200'
}`}
onClick={() => {
const newTags = new Set(selectedTags);
if (newTags.has(category.id)) {
newTags.delete(category.id);
} else {
newTags.add(category.id);
}
handleTagsChange(newTags);
}}
>
{category.label}
</button>
))}
</div>
</div>
{/* Controls row with past conferences toggle and country filter */}
<div className="flex flex-wrap items-center gap-4">
<div className="flex items-center gap-2 bg-white p-2 rounded-md shadow-sm">
<label htmlFor="show-past" className="text-sm text-neutral-600">
Show past conferences
</label>
<Switch
id="show-past"
checked={showPastConferences}
onCheckedChange={setShowPastConferences}
/>
</div>
<div className="flex items-center gap-2 bg-white p-2 rounded-md shadow-sm">
<Popover>
<PopoverTrigger asChild>
<Button variant="outline" size="sm" className="h-8 gap-1">
<Globe className="h-4 w-4" />
Filter by Country
</Button>
</PopoverTrigger>
<PopoverContent className="w-64 p-4 bg-white" align="start">
<div className="space-y-4">
<div>
<div className="mb-4">
<h4 className="text-sm font-medium text-gray-800">Country</h4>
</div>
<div className="max-h-60 overflow-y-auto space-y-2 bg-white">
{getAllCountries(conferencesData as Conference[]).map(country => (
<div key={country} className="flex items-center space-x-2 hover:bg-gray-50 p-1 rounded">
<Checkbox
id={`country-${country}`}
checked={selectedCountries.has(country)}
onCheckedChange={() => {
const newCountries = new Set(selectedCountries);
if (newCountries.has(country)) {
newCountries.delete(country);
} else {
newCountries.add(country);
}
handleCountriesChange(newCountries);
}}
/>
<label
htmlFor={`country-${country}`}
className="text-sm font-medium text-gray-700 cursor-pointer w-full py-1"
>
{country}
</label>
</div>
))}
</div>
</div>
</div>
</PopoverContent>
</Popover>
{/* Display selected countries */}
{Array.from(selectedCountries).map(country => (
<button
key={country}
className="inline-flex items-center px-3 py-1 rounded-full text-sm bg-blue-100 text-blue-800 hover:bg-blue-200 font-medium"
onClick={() => {
const newCountries = new Set(selectedCountries);
newCountries.delete(country);
handleCountriesChange(newCountries);
}}
>
{country}
<X className="ml-1 h-3 w-3" />
</button>
))}
{/* Clear all filters button */}
{(selectedTags.size > 0 || selectedCountries.size > 0) && (
<Button
variant="ghost"
size="sm"
onClick={() => {
handleTagsChange(new Set());
handleCountriesChange(new Set());
}}
className="text-neutral-500 hover:text-neutral-700"
>
Clear all filters
</Button>
)}
</div>
</div>
</div>
</div>
<main className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
{filteredConferences.length === 0 && (
<div className="bg-amber-50 border border-amber-200 text-amber-800 rounded-md p-4 mb-6">
<p className="text-center">
There are no upcoming conferences for the selected categories - enable "Show past conferences" to see previous ones
</p>
</div>
)}
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
{filteredConferences.map((conference: Conference) => (
<ConferenceCard key={conference.id} {...conference} />
))}
</div>
</main>
</div>
);
};
export default Index;
|