146 lines
5.9 KiB
TypeScript
146 lines
5.9 KiB
TypeScript
"use client";
|
|
|
|
import { Mic2 } from "lucide-react";
|
|
import Image from "next/image";
|
|
import { ReactNode } from "react";
|
|
|
|
interface PodcastHeroProps {
|
|
title: string;
|
|
author: string;
|
|
description?: string;
|
|
genres?: string[];
|
|
heroImage: string | null;
|
|
colors: any;
|
|
episodeCount: number;
|
|
inProgressCount: number;
|
|
children?: ReactNode;
|
|
}
|
|
|
|
export function PodcastHero({
|
|
title,
|
|
author,
|
|
description,
|
|
genres,
|
|
heroImage,
|
|
colors,
|
|
episodeCount,
|
|
inProgressCount,
|
|
children,
|
|
}: PodcastHeroProps) {
|
|
return (
|
|
<div className="relative">
|
|
{/* Background with VibrantJS gradient */}
|
|
{heroImage ? (
|
|
<div className="absolute inset-0 overflow-hidden">
|
|
<div className="absolute inset-0 scale-110 blur-md opacity-50">
|
|
<Image
|
|
src={heroImage}
|
|
alt={title}
|
|
fill
|
|
sizes="100vw"
|
|
className="object-cover"
|
|
priority
|
|
unoptimized
|
|
/>
|
|
</div>
|
|
<div
|
|
className="absolute inset-0"
|
|
style={{
|
|
background: colors
|
|
? `linear-gradient(to bottom, ${colors.vibrant}30 0%, ${colors.darkVibrant}60 40%, ${colors.darkMuted}90 70%, #0a0a0a 100%)`
|
|
: "linear-gradient(to bottom, rgba(0,0,0,0.5) 0%, rgba(0,0,0,0.7) 40%, #0a0a0a 100%)",
|
|
}}
|
|
/>
|
|
<div className="absolute inset-0 bg-gradient-to-t from-[#0a0a0a] via-transparent to-transparent" />
|
|
</div>
|
|
) : (
|
|
<div
|
|
className="absolute inset-0"
|
|
style={{
|
|
background: colors
|
|
? `linear-gradient(to bottom, ${colors.vibrant}40 0%, ${colors.darkVibrant}80 50%, #0a0a0a 100%)`
|
|
: "linear-gradient(to bottom, #3d2a1e 0%, #1a1a1a 50%, #0a0a0a 100%)",
|
|
}}
|
|
/>
|
|
)}
|
|
|
|
{/* Compact Hero Content - Full Width */}
|
|
<div className="relative px-4 md:px-8 pt-16 pb-6">
|
|
<div className="flex items-end gap-6">
|
|
{/* Cover Art - Circular for podcasts */}
|
|
<div className="w-[140px] h-[140px] md:w-[192px] md:h-[192px] bg-[#282828] rounded-full shadow-2xl shrink-0 overflow-hidden relative">
|
|
{heroImage ? (
|
|
<Image
|
|
src={heroImage}
|
|
alt={title}
|
|
fill
|
|
sizes="(max-width: 768px) 140px, 192px"
|
|
className="object-cover"
|
|
priority
|
|
unoptimized
|
|
/>
|
|
) : (
|
|
<div className="w-full h-full flex items-center justify-center">
|
|
<Mic2 className="w-16 h-16 text-gray-600" />
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Info - Bottom Aligned */}
|
|
<div className="flex-1 min-w-0 pb-1">
|
|
<p className="text-xs font-medium text-white/90 mb-1">
|
|
Podcast
|
|
</p>
|
|
<h1 className="text-2xl md:text-4xl lg:text-5xl font-bold text-white leading-tight line-clamp-2 mb-2">
|
|
{title}
|
|
</h1>
|
|
|
|
{/* Description - truncated */}
|
|
{description && (
|
|
<p className="text-sm text-white/60 line-clamp-2 max-w-3xl mb-2 hidden md:block">
|
|
{description.replace(/<[^>]*>/g, "").substring(0, 150)}...
|
|
</p>
|
|
)}
|
|
|
|
{/* Metadata row */}
|
|
<div className="flex flex-wrap items-center gap-1.5 text-sm text-white/70">
|
|
<span className="font-semibold text-white">{author}</span>
|
|
<span className="mx-1">•</span>
|
|
<span>{episodeCount} {episodeCount === 1 ? "episode" : "episodes"}</span>
|
|
{inProgressCount > 0 && (
|
|
<>
|
|
<span className="mx-1">•</span>
|
|
<span className="text-[#ecb200]">
|
|
{inProgressCount} in progress
|
|
</span>
|
|
</>
|
|
)}
|
|
</div>
|
|
|
|
{/* Genre tags - compact */}
|
|
{genres && genres.length > 0 && (
|
|
<div className="hidden md:flex flex-wrap gap-1.5 mt-3">
|
|
{genres.slice(0, 4).map((genre: string) => (
|
|
<span
|
|
key={genre}
|
|
className="px-2.5 py-0.5 bg-white/10 rounded-full text-xs text-white/80"
|
|
>
|
|
{genre}
|
|
</span>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Action Bar - Full Width */}
|
|
{children && (
|
|
<div className="relative px-4 md:px-8 pb-4">
|
|
{children}
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|