Index.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. import { useState, useMemo, useEffect } from 'react';
  2. import { categories } from '@/data/mockNews';
  3. import { useRealArticles } from '@/hooks/useRealArticles';
  4. import { useAuth } from '@/hooks/useAuth';
  5. import { useIsMobile } from '@/hooks/use-mobile';
  6. import { NewsItem } from '@/types/news';
  7. import Header from '@/components/Header';
  8. import CategoryFilter from '@/components/CategoryFilter';
  9. import NewsCard from '@/components/NewsCard';
  10. import AddFeedModal from '@/components/AddFeedModal';
  11. import ArticleModal from '@/components/ArticleModal';
  12. import { Badge } from '@/components/ui/badge';
  13. import { Button } from '@/components/ui/button';
  14. import { Card, CardContent } from '@/components/ui/card';
  15. import { Drawer, DrawerContent, DrawerTrigger } from '@/components/ui/drawer';
  16. import { RefreshCw, Filter, User, Rss, Plus } from 'lucide-react';
  17. import { toast } from 'sonner';
  18. import { Link } from 'react-router-dom';
  19. const Index = () => {
  20. const {
  21. user
  22. } = useAuth();
  23. const [dateFilter, setDateFilter] = useState<'today' | 'yesterday' | null>(null);
  24. const [showFollowedOnly, setShowFollowedOnly] = useState(false);
  25. const [showReadArticles, setShowReadArticles] = useState(false);
  26. // Reset date filter when switching to "All articles" mode
  27. const handleShowFollowedOnlyChange = (value: boolean) => {
  28. setShowFollowedOnly(value);
  29. if (!value) {
  30. // When switching to "All articles", reset date filter
  31. setDateFilter(null);
  32. }
  33. };
  34. const {
  35. articles,
  36. loading,
  37. togglePin,
  38. markAsRead,
  39. deleteArticle,
  40. refetch
  41. } = useRealArticles(dateFilter, showFollowedOnly, showReadArticles);
  42. const isMobile = useIsMobile();
  43. const [selectedCategory, setSelectedCategory] = useState<string | null>(null);
  44. const [searchQuery, setSearchQuery] = useState('');
  45. const [showFilters, setShowFilters] = useState(true);
  46. const [isAddFeedModalOpen, setIsAddFeedModalOpen] = useState(false);
  47. const [selectedArticle, setSelectedArticle] = useState<NewsItem | null>(null);
  48. const [isArticleModalOpen, setIsArticleModalOpen] = useState(false);
  49. console.log('🏠 Index page - Articles count:', articles.length, 'Loading:', loading, 'User:', !!user);
  50. const filteredNews = useMemo(() => {
  51. let filtered = articles;
  52. if (selectedCategory) {
  53. const category = categories.find(c => c.id === selectedCategory);
  54. if (category) {
  55. filtered = filtered.filter(item => item.category === category.type);
  56. }
  57. }
  58. if (searchQuery) {
  59. filtered = filtered.filter(item => item.title.toLowerCase().includes(searchQuery.toLowerCase()) || item.description.toLowerCase().includes(searchQuery.toLowerCase()) || item.source.toLowerCase().includes(searchQuery.toLowerCase()));
  60. }
  61. return filtered.sort((a, b) => {
  62. if (a.isPinned && !b.isPinned) return -1;
  63. if (!a.isPinned && b.isPinned) return 1;
  64. return new Date(b.publishedAt).getTime() - new Date(a.publishedAt).getTime();
  65. });
  66. }, [articles, selectedCategory, searchQuery]);
  67. const pinnedCount = articles.filter(item => item.isPinned).length;
  68. const unreadCount = articles.filter(item => !item.isRead).length;
  69. // Update document title with unread count
  70. useEffect(() => {
  71. const baseTitle = 'Feeds.Duhaz.fr';
  72. if (unreadCount > 0) {
  73. document.title = `(${unreadCount}) ${baseTitle}`;
  74. } else {
  75. document.title = baseTitle;
  76. }
  77. }, [unreadCount]);
  78. const handleRefresh = () => {
  79. refetch();
  80. toast.success("Flux actualisés");
  81. };
  82. const handleAddFeed = (feedData: any) => {
  83. console.log('Nouveau flux ajouté:', feedData);
  84. toast.success(`Flux "${feedData.name}" ajouté avec succès!`);
  85. };
  86. const handleOpenArticle = (article: NewsItem) => {
  87. setSelectedArticle(article);
  88. setIsArticleModalOpen(true);
  89. };
  90. const handleCloseArticleModal = () => {
  91. setIsArticleModalOpen(false);
  92. setSelectedArticle(null);
  93. };
  94. if (loading) {
  95. return <div className="min-h-screen bg-background flex items-center justify-center">
  96. <div className="flex items-center gap-2">
  97. <Rss className="h-6 w-6 animate-spin text-primary" />
  98. <p>Chargement des articles...</p>
  99. </div>
  100. </div>;
  101. }
  102. return <div className="min-h-screen bg-background">
  103. <Header searchQuery={searchQuery} onSearchChange={setSearchQuery} pinnedCount={pinnedCount} onAddFeedClick={() => setIsAddFeedModalOpen(true)} />
  104. <main className="container mx-auto px-4 py-6">
  105. {/* Message pour les utilisateurs non connectés */}
  106. {!user && (
  107. <div className="mb-6 p-4 bg-blue-50 border border-blue-200 rounded-lg">
  108. <p className="text-blue-800">
  109. Vous consultez les articles en mode visiteur.
  110. <Link to="/auth" className="font-semibold text-blue-600 hover:text-blue-800 ml-1">
  111. Connectez-vous
  112. </Link> pour gérer vos flux et marquer vos articles préférés.
  113. </p>
  114. </div>
  115. )}
  116. {/* Message pour les utilisateurs connectés sans articles suivis */}
  117. {user && articles.length === 0 && (
  118. <div className="mb-6 p-4 bg-amber-50 border border-amber-200 rounded-lg">
  119. <p className="text-amber-800">
  120. Vous ne suivez aucun flux RSS pour le moment.
  121. <Link to="/feeds" className="font-semibold text-amber-600 hover:text-amber-800 ml-1">
  122. Ajoutez des flux
  123. </Link> pour commencer à voir des articles.
  124. </p>
  125. </div>
  126. )}
  127. <div className="grid grid-cols-1 lg:grid-cols-4 gap-6">
  128. {/* Sidebar - Desktop only */}
  129. {!isMobile && (
  130. <div className={`lg:col-span-1 space-y-6 ${!showFilters && 'hidden lg:block'}`}>
  131. <CategoryFilter
  132. categories={categories}
  133. selectedCategory={selectedCategory}
  134. onCategoryChange={setSelectedCategory}
  135. newsCount={articles.length}
  136. pinnedCount={pinnedCount}
  137. articles={articles}
  138. dateFilter={dateFilter}
  139. onDateFilterChange={setDateFilter}
  140. showFollowedOnly={showFollowedOnly}
  141. onShowFollowedOnlyChange={handleShowFollowedOnlyChange}
  142. showReadArticles={showReadArticles}
  143. onShowReadArticlesChange={setShowReadArticles}
  144. />
  145. <div className="bg-card border rounded-lg p-4 space-y-3">
  146. <h3 className="font-semibold text-sm">Statistiques</h3>
  147. <div className="space-y-2 text-sm">
  148. <div className="flex justify-between">
  149. <span className="text-muted-foreground">Articles non lus</span>
  150. <Badge variant="outline">{unreadCount}</Badge>
  151. </div>
  152. <div className="flex justify-between">
  153. <span className="text-muted-foreground">Articles totaux</span>
  154. <Badge variant="outline">{articles.length}</Badge>
  155. </div>
  156. {user && <div className="flex justify-between">
  157. <span className="text-muted-foreground">Épinglés</span>
  158. <Badge variant="secondary">{pinnedCount}</Badge>
  159. </div>}
  160. </div>
  161. </div>
  162. </div>
  163. )}
  164. {/* Main content */}
  165. <div className={`${isMobile ? 'col-span-1' : 'lg:col-span-3'} space-y-6`}>
  166. <div className="flex items-center justify-between">
  167. <div className="flex items-center gap-4">
  168. <h2 className="text-2xl font-bold">
  169. {showReadArticles ? 'Tous les articles' : (user ? 'Articles non lus' : 'Derniers articles')}
  170. </h2>
  171. </div>
  172. <div className="flex items-center gap-2">
  173. {/* Mobile Filter Drawer */}
  174. {isMobile && (
  175. <Drawer>
  176. <DrawerTrigger asChild>
  177. <Button variant="outline" size="sm" className="gap-2">
  178. <Filter className="h-4 w-4" />
  179. Filtres
  180. </Button>
  181. </DrawerTrigger>
  182. <DrawerContent>
  183. <div className="p-4 space-y-6">
  184. <CategoryFilter
  185. categories={categories}
  186. selectedCategory={selectedCategory}
  187. onCategoryChange={setSelectedCategory}
  188. newsCount={articles.length}
  189. pinnedCount={pinnedCount}
  190. articles={articles}
  191. dateFilter={dateFilter}
  192. onDateFilterChange={setDateFilter}
  193. showFollowedOnly={showFollowedOnly}
  194. onShowFollowedOnlyChange={handleShowFollowedOnlyChange}
  195. showReadArticles={showReadArticles}
  196. onShowReadArticlesChange={setShowReadArticles}
  197. />
  198. <div className="bg-card border rounded-lg p-4 space-y-3">
  199. <h3 className="font-semibold text-sm">Statistiques</h3>
  200. <div className="space-y-2 text-sm">
  201. <div className="flex justify-between">
  202. <span className="text-muted-foreground">Articles non lus</span>
  203. <Badge variant="outline">{unreadCount}</Badge>
  204. </div>
  205. <div className="flex justify-between">
  206. <span className="text-muted-foreground">Articles totaux</span>
  207. <Badge variant="outline">{articles.length}</Badge>
  208. </div>
  209. {user && <div className="flex justify-between">
  210. <span className="text-muted-foreground">Épinglés</span>
  211. <Badge variant="secondary">{pinnedCount}</Badge>
  212. </div>}
  213. </div>
  214. </div>
  215. </div>
  216. </DrawerContent>
  217. </Drawer>
  218. )}
  219. {/* Desktop Filter Toggle - keep existing logic */}
  220. {!isMobile && (
  221. <Button variant="outline" size="sm" onClick={() => setShowFilters(!showFilters)} className="lg:hidden gap-2">
  222. <Filter className="h-4 w-4" />
  223. Filtres
  224. </Button>
  225. )}
  226. <Button variant="outline" size="sm" onClick={handleRefresh} className="gap-2">
  227. <RefreshCw className="h-4 w-4" />
  228. Actualiser
  229. </Button>
  230. </div>
  231. </div>
  232. {filteredNews.length === 0 && articles.length > 0 ? <div className="text-center py-12">
  233. <p className="text-muted-foreground text-lg">Aucun article trouvé avec ces filtres</p>
  234. <p className="text-sm text-muted-foreground mt-2">
  235. Essayez de modifier vos filtres ou votre recherche
  236. </p>
  237. </div> : filteredNews.length === 0 ? <div className="text-center py-12">
  238. <Rss className="h-12 w-12 text-muted-foreground mx-auto mb-4" />
  239. <p className="text-muted-foreground text-lg">Aucun article non lu disponible</p>
  240. <p className="text-sm text-muted-foreground mt-2 mb-4">
  241. {user ? 'Bravo ! Tous vos articles sont lus ou suivez des flux RSS pour voir des articles ici' : 'Aucun article public disponible pour le moment'}
  242. </p>
  243. {user && <div className="flex gap-2 justify-center">
  244. <Link to="/feeds">
  245. <Button variant="outline">
  246. Gérer les flux
  247. </Button>
  248. </Link>
  249. <Button onClick={() => setIsAddFeedModalOpen(true)}>
  250. <Plus className="h-4 w-4 mr-2" />
  251. Ajouter un flux
  252. </Button>
  253. </div>}
  254. </div> : <div className="space-y-4">
  255. {filteredNews.map(item => <NewsCard key={item.id} news={item} onTogglePin={togglePin} onMarkAsRead={markAsRead} onDelete={deleteArticle} onOpenArticle={handleOpenArticle} />)}
  256. </div>}
  257. </div>
  258. </div>
  259. </main>
  260. {/* Modals */}
  261. {user && <AddFeedModal isOpen={isAddFeedModalOpen} onClose={() => setIsAddFeedModalOpen(false)} onAddFeed={handleAddFeed} categories={categories} />}
  262. <ArticleModal isOpen={isArticleModalOpen} onClose={handleCloseArticleModal} article={selectedArticle} />
  263. </div>;
  264. };
  265. export default Index;