folx-tv/client/src/pages/videos.tsx
sebastjanartic b320d8b601 Improve website content and fix navigation issues
Update website meta tags, SEO information, and fix a banner overlay bug to ensure all interactive elements are clickable.

Replit-Commit-Author: Agent
Replit-Commit-Session-Id: 23852c00-4779-460a-9e0c-d09fee4b6c92
Replit-Commit-Checkpoint-Type: full_checkpoint
Replit-Commit-Event-Id: 60ed045f-57e0-4c65-bc71-4205e0064bbb
Replit-Commit-Screenshot-Url: https://storage.googleapis.com/screenshot-production-us-central1/f209e72a-0939-48fa-84fc-57854de71967/23852c00-4779-460a-9e0c-d09fee4b6c92/aNXfGlM
Replit-Helium-Checkpoint-Created: true
2026-03-07 15:46:13 +00:00

204 lines
7.1 KiB
TypeScript

import { useQuery } from "@tanstack/react-query";
import { Link, useLocation, useSearch } from "wouter";
import { type Article } from "@shared/schema";
import { format } from "date-fns";
import { de } from "date-fns/locale";
import { usePageMeta } from "@/hooks/use-page-meta";
import { Play, ArrowLeft, ChevronLeft, ChevronRight } from "lucide-react";
import { Button } from "@/components/ui/button";
import { Skeleton } from "@/components/ui/skeleton";
import Header from "@/components/header";
import Footer from "@/components/footer";
import { ArticleCardAd, PageSideAds } from "@/components/adsense";
import { useEffect } from "react";
interface PaginatedResponse {
articles: Article[];
total: number;
page: number;
totalPages: number;
hasMore: boolean;
}
function VideoCard({ article }: { article: Article }) {
const thumbSrc = article.coverImage
? article.coverImage.replace(".webp", "-thumb.webp")
: "/images/article-1.jpg";
return (
<Link href={`/article/${article.slug}`}>
<article
className="group cursor-pointer bg-card rounded-md border border-card-border transition-all duration-300 h-full flex flex-col"
data-testid={`card-video-${article.id}`}
>
<div className="relative rounded-t-md">
<div className="overflow-hidden rounded-t-md">
<img
src={thumbSrc}
alt={article.title}
className="w-full aspect-video object-cover transition-transform duration-500 group-hover:scale-105"
loading="lazy"
/>
</div>
<div className="absolute inset-0 flex items-center justify-center">
<div className="w-12 h-12 rounded-full bg-primary/90 flex items-center justify-center group-hover:bg-primary transition-colors">
<Play className="w-5 h-5 text-white ml-0.5" fill="white" />
</div>
</div>
</div>
<div className="p-4 flex flex-col flex-1">
<h3 className="font-semibold text-card-foreground mb-1 line-clamp-2 group-hover:text-primary transition-colors text-sm leading-snug">
{article.title}
</h3>
<p className="text-xs text-muted-foreground mt-auto">
{format(new Date(article.publishedAt), "d. MMMM yyyy", { locale: de })}
</p>
</div>
</article>
</Link>
);
}
function VideoCardSkeleton() {
return (
<div className="bg-card rounded-md border border-card-border">
<Skeleton className="w-full aspect-video rounded-t-md" />
<div className="p-4 space-y-2">
<Skeleton className="h-4 w-full" />
<Skeleton className="h-3 w-1/2" />
</div>
</div>
);
}
export default function VideosPage() {
usePageMeta("Volksmusik & Schlager Videos", "Volksmusik und Schlager Musikvideos, Live-Auftritte und Konzertmitschnitte bei FOLX TV. Die besten volkstümlichen Hits und Schlager-Stars im Video.");
const searchString = useSearch();
const [, setLocation] = useLocation();
const params = new URLSearchParams(searchString);
const currentPage = Math.max(1, parseInt(params.get("page") || "1"));
const limit = 12;
const { data, isLoading } = useQuery<PaginatedResponse>({
queryKey: [`/api/articles/category/Video?page=${currentPage}&limit=${limit}`],
});
useEffect(() => {
window.scrollTo({ top: 0, behavior: "smooth" });
}, [currentPage]);
const goToPage = (page: number) => {
if (page === 1) {
setLocation("/videos");
} else {
setLocation(`/videos?page=${page}`);
}
};
const totalPages = data?.totalPages || 1;
const articles = data?.articles || [];
const getPageNumbers = () => {
const pages: (number | "...")[] = [];
if (totalPages <= 7) {
for (let i = 1; i <= totalPages; i++) pages.push(i);
} else {
pages.push(1);
if (currentPage > 3) pages.push("...");
const start = Math.max(2, currentPage - 1);
const end = Math.min(totalPages - 1, currentPage + 1);
for (let i = start; i <= end; i++) pages.push(i);
if (currentPage < totalPages - 2) pages.push("...");
pages.push(totalPages);
}
return pages;
};
return (
<div className="min-h-screen bg-background">
<Header />
<PageSideAds />
<main className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-4">
<Link href="/">
<button className="flex items-center gap-2 text-muted-foreground hover:text-foreground transition-colors mb-6 text-sm" data-testid="button-back">
<ArrowLeft className="w-4 h-4" />
Zurück
</button>
</Link>
<h1 className="text-2xl font-bold text-foreground mb-6" data-testid="text-videos-title">
Videos
</h1>
{isLoading ? (
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-5">
{Array.from({ length: 8 }).map((_, i) => (
<VideoCardSkeleton key={i} />
))}
</div>
) : articles.length > 0 ? (
<>
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-5">
{articles.flatMap((article, index) => {
const items: JSX.Element[] = [
<VideoCard key={article.id} article={article} />,
];
if ((index + 1) % 3 === 0) {
items.push(<ArticleCardAd key={`ad-video-${index}`} />);
}
return items;
})}
</div>
{totalPages > 1 && (
<nav className="flex items-center justify-center gap-1 mt-10" data-testid="pagination-videos">
<Button
variant="outline"
size="icon"
onClick={() => goToPage(currentPage - 1)}
disabled={currentPage <= 1}
data-testid="button-prev-page"
>
<ChevronLeft className="w-4 h-4" />
</Button>
{getPageNumbers().map((p, i) =>
p === "..." ? (
<span key={`ellipsis-${i}`} className="px-2 text-muted-foreground">...</span>
) : (
<Button
key={p}
variant={p === currentPage ? "default" : "outline"}
size="sm"
onClick={() => goToPage(p as number)}
data-testid={`button-page-${p}`}
>
{p}
</Button>
)
)}
<Button
variant="outline"
size="icon"
onClick={() => goToPage(currentPage + 1)}
disabled={currentPage >= totalPages}
data-testid="button-next-page"
>
<ChevronRight className="w-4 h-4" />
</Button>
</nav>
)}
</>
) : (
<div className="text-center py-16">
<p className="text-muted-foreground">Noch keine Videos vorhanden.</p>
</div>
)}
</main>
<Footer />
</div>
);
}