mirror of
https://github.com/JustAnimeCore/JustAnime.git
synced 2026-04-17 22:01:45 +00:00
basic info page DONE
This commit is contained in:
@@ -72,14 +72,9 @@ const NotFoundState = () => (
|
||||
// Main anime content component
|
||||
const AnimeContent = async ({ id }) => {
|
||||
try {
|
||||
console.log('[AnimeInfo] Fetching info for ID:', id);
|
||||
|
||||
const anime = await fetchAnimeInfo(id);
|
||||
|
||||
console.log('[AnimeInfo] API Response received:', anime ? 'success' : 'empty');
|
||||
|
||||
if (!anime || !anime.info) {
|
||||
console.error('[AnimeInfo] Missing required anime data');
|
||||
return <NotFoundState />;
|
||||
}
|
||||
|
||||
@@ -89,7 +84,6 @@ const AnimeContent = async ({ id }) => {
|
||||
</div>
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('[AnimeInfo] Error:', error.message);
|
||||
return <ErrorState error={error.message || 'An error occurred while loading the anime.'} />;
|
||||
}
|
||||
};
|
||||
|
||||
@@ -1,4 +1,6 @@
|
||||
import { Geist, Geist_Mono } from "next/font/google";
|
||||
import { Analytics } from "@vercel/analytics/next";
|
||||
import { SpeedInsights } from "@vercel/speed-insights/next";
|
||||
import "./globals.css";
|
||||
|
||||
const geistSans = Geist({
|
||||
@@ -24,6 +26,8 @@ export default function RootLayout({ children }) {
|
||||
<main className="flex-grow">
|
||||
{children}
|
||||
</main>
|
||||
<Analytics />
|
||||
<SpeedInsights />
|
||||
</body>
|
||||
</html>
|
||||
);
|
||||
|
||||
@@ -13,8 +13,6 @@ export default function AnimeDetails({ anime }) {
|
||||
const [synopsisOverflows, setSynopsisOverflows] = useState(false);
|
||||
const synopsisRef = useRef(null);
|
||||
|
||||
console.log('AnimeDetails received:', anime);
|
||||
|
||||
// Check if synopsis overflows when component mounts or when content changes
|
||||
useEffect(() => {
|
||||
if (synopsisRef.current) {
|
||||
@@ -24,7 +22,6 @@ export default function AnimeDetails({ anime }) {
|
||||
}, [anime?.info?.description, activeTab]);
|
||||
|
||||
if (!anime?.info) {
|
||||
console.error('Invalid anime data structure:', anime);
|
||||
return null;
|
||||
}
|
||||
|
||||
@@ -37,13 +34,13 @@ export default function AnimeDetails({ anime }) {
|
||||
if (!video) return null;
|
||||
|
||||
return (
|
||||
<div className="fixed inset-0 bg-black/70 z-50 flex items-center justify-center p-4 backdrop-blur-sm">
|
||||
<div className="relative max-w-4xl w-full bg-[var(--card)] rounded-lg overflow-hidden shadow-2xl border border-gray-700 animate-fadeIn">
|
||||
<div className="fixed inset-0 bg-black/80 z-50 flex items-center justify-center p-3 backdrop-blur-sm">
|
||||
<div className="relative w-full max-w-4xl bg-[var(--card)] rounded-lg overflow-hidden shadow-2xl border border-gray-700 animate-fadeIn">
|
||||
<button
|
||||
onClick={onClose}
|
||||
className="absolute top-3 right-3 z-10 bg-black/50 rounded-full p-1 hover:bg-black/70 transition-colors"
|
||||
className="absolute top-3 right-3 z-10 bg-black/50 rounded-full p-1.5 hover:bg-black/70 transition-colors"
|
||||
>
|
||||
<svg xmlns="http://www.w3.org/2000/svg" className="h-6 w-6 text-white" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" className="h-5 w-5 sm:h-6 sm:w-6 text-white" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
|
||||
</svg>
|
||||
</button>
|
||||
@@ -75,8 +72,8 @@ export default function AnimeDetails({ anime }) {
|
||||
{/* Video Modal */}
|
||||
{activeVideo && <VideoModal video={activeVideo} onClose={() => setActiveVideo(null)} />}
|
||||
|
||||
{/* Background Image with Gradient Overlay */}
|
||||
<div className="absolute inset-0 h-[200px] sm:h-[250px] md:h-[400px] overflow-hidden -z-10">
|
||||
{/* Background Image with Gradient Overlay - Desktop Only */}
|
||||
<div className="absolute inset-0 h-[180px] md:h-[400px] overflow-hidden -z-10">
|
||||
{info.poster && (
|
||||
<>
|
||||
<Image
|
||||
@@ -92,11 +89,146 @@ export default function AnimeDetails({ anime }) {
|
||||
</div>
|
||||
|
||||
{/* Main Content */}
|
||||
<div className="container mx-auto px-3 sm:px-4 pt-4 sm:pt-6 md:pt-10">
|
||||
{/* Header Section - Title and Basic Info */}
|
||||
<div className="flex flex-col md:flex-row gap-4 md:gap-10 mb-6 md:mb-8">
|
||||
<div className="container mx-auto px-3 md:px-4 pt-3 md:pt-10">
|
||||
{/* MOBILE LAYOUT - Only visible on mobile */}
|
||||
<div className="md:hidden">
|
||||
<div className="flex flex-col mb-5">
|
||||
{/* Mobile Header with Title + Rating */}
|
||||
<div className="flex items-center justify-between mb-3">
|
||||
<h1 className="text-xl font-bold text-white pr-3">{info.name}</h1>
|
||||
{info.stats?.rating && (
|
||||
<div className="flex items-center bg-[var(--card)] px-2 py-1 rounded-md">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
className="h-3.5 w-3.5 text-yellow-400 mr-1"
|
||||
viewBox="0 0 20 20"
|
||||
fill="currentColor"
|
||||
>
|
||||
<path d="M9.049 2.927c.3-.921 1.603-.921 1.902 0l1.07 3.292a1 1 0 00.95.69h3.462c.969 0 1.371 1.24.588 1.81l-2.8 2.034a1 1 0 00-.364 1.118l1.07 3.292c.3.921-.755 1.688-1.54 1.118l-2.8-2.034a1 1 0 00-1.175 0l-2.8 2.034c-.784.57-1.838-.197-1.539-1.118l1.07-3.292a1 1 0 00-.364-1.118L2.98 8.72c-.783-.57-.38-1.81.588-1.81h3.461a1 1 0 00.951-.69l1.07-3.292z" />
|
||||
</svg>
|
||||
<span className="text-white text-xs font-medium">{info.stats.rating}</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Japanese Title */}
|
||||
{moreInfo?.japanese && (
|
||||
<h2 className="text-xs text-gray-300 mt-[-0.25rem] mb-3">{moreInfo.japanese}</h2>
|
||||
)}
|
||||
|
||||
{/* Mobile Two-Column Layout */}
|
||||
<div className="flex gap-3">
|
||||
{/* Left Column - Poster */}
|
||||
<div className="w-2/5 flex-shrink-0">
|
||||
<div className="bg-[var(--card)] rounded-xl overflow-hidden shadow-lg border border-gray-800">
|
||||
<div className="relative aspect-[3/4] w-full">
|
||||
<Image
|
||||
src={info.poster}
|
||||
alt={info.name}
|
||||
fill
|
||||
className="object-cover"
|
||||
priority
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Right Column - Info Card */}
|
||||
<div className="w-3/5 flex flex-col">
|
||||
{/* Type & Episodes on same row */}
|
||||
<div className="flex gap-2 mb-2">
|
||||
{info.stats?.type && (
|
||||
<div className="bg-[var(--card)] px-2 py-1 rounded-md text-[10px] text-white">{info.stats.type}</div>
|
||||
)}
|
||||
|
||||
{info.stats?.episodes && (
|
||||
<div className="bg-[var(--card)] px-2 py-1 rounded-md text-[10px] text-white grow">
|
||||
{info.stats.episodes.sub > 0 && `Sub: ${info.stats.episodes.sub}`}
|
||||
{info.stats.episodes.dub > 0 && info.stats.episodes.sub > 0 && ' • '}
|
||||
{info.stats.episodes.dub > 0 && `Dub: ${info.stats.episodes.dub}`}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Clean Info Layout */}
|
||||
<div className="bg-[var(--card)] rounded-md p-2.5 text-[11px] space-y-1.5 mb-2">
|
||||
{/* Status */}
|
||||
{moreInfo?.status && (
|
||||
<div className="flex">
|
||||
<span className="text-gray-400 w-16">Status:</span>
|
||||
<span className="text-white">{getStatusWithAired()}</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Quality */}
|
||||
{info.stats?.quality && (
|
||||
<div className="flex">
|
||||
<span className="text-gray-400 w-16">Quality:</span>
|
||||
<span className="text-white">{info.stats.quality}</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Duration */}
|
||||
{info.stats?.duration && (
|
||||
<div className="flex">
|
||||
<span className="text-gray-400 w-16">Duration:</span>
|
||||
<span className="text-white">{info.stats.duration}</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Studio */}
|
||||
{moreInfo?.studios && (
|
||||
<div className="flex">
|
||||
<span className="text-gray-400 w-16">Studio:</span>
|
||||
<span className="text-white">{moreInfo.studios}</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Mobile Genres */}
|
||||
{moreInfo?.genres && moreInfo.genres.length > 0 && (
|
||||
<div className="flex flex-wrap gap-1.5">
|
||||
{moreInfo.genres.slice(0, 5).map((genre, index) => (
|
||||
<Link
|
||||
key={index}
|
||||
href={`/genre/${genre.toLowerCase()}`}
|
||||
className="bg-[var(--card)] px-2 py-0.5 rounded-md text-[10px] text-gray-300 hover:text-white"
|
||||
>
|
||||
{genre}
|
||||
</Link>
|
||||
))}
|
||||
{moreInfo.genres.length > 5 && (
|
||||
<span className="text-[10px] text-gray-500 self-center">+{moreInfo.genres.length - 5}</span>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Watch Button - Full Width on Mobile */}
|
||||
{info.stats?.episodes && (info.stats.episodes.sub > 0 || info.stats.episodes.dub > 0) && (
|
||||
<Link
|
||||
href={`/watch/${info.id}?ep=1`}
|
||||
className="bg-[#ffffff] text-[var(--background)] px-4 py-2.5 rounded-xl mt-3 hover:opacity-90 transition-opacity flex items-center justify-center font-medium text-sm w-full shadow-lg"
|
||||
>
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
viewBox="0 0 24 24"
|
||||
fill="currentColor"
|
||||
className="h-4 w-4 mr-1.5"
|
||||
>
|
||||
<path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-2 14.5v-9l6 4.5-6 4.5z" />
|
||||
</svg>
|
||||
<span>Start Watching</span>
|
||||
</Link>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* DESKTOP LAYOUT - Only visible on desktop */}
|
||||
<div className="hidden md:flex md:flex-row gap-10 mb-8">
|
||||
{/* Poster */}
|
||||
<div className="w-[180px] sm:w-[220px] md:w-1/4 max-w-[240px] mx-auto md:mx-0">
|
||||
<div className="w-1/4 max-w-[240px]">
|
||||
<div className="bg-[var(--card)] rounded-xl overflow-hidden shadow-lg border border-gray-800">
|
||||
<div className="relative aspect-[3/4] w-full">
|
||||
<Image
|
||||
@@ -109,17 +241,17 @@ export default function AnimeDetails({ anime }) {
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Watch Button - Mobile & Desktop */}
|
||||
{/* Watch Button - Desktop */}
|
||||
{info.stats?.episodes && (info.stats.episodes.sub > 0 || info.stats.episodes.dub > 0) && (
|
||||
<Link
|
||||
href={`/watch/${info.id}?ep=1`}
|
||||
className="bg-[#ffffff] text-[var(--background)] px-4 sm:px-6 py-2.5 sm:py-3 rounded-xl mt-3 sm:mt-4 hover:opacity-90 transition-opacity flex items-center justify-center font-medium text-sm sm:text-base w-full shadow-lg"
|
||||
className="bg-[#ffffff] text-[var(--background)] px-6 py-3 rounded-xl mt-4 hover:opacity-90 transition-opacity flex items-center justify-center font-medium text-base w-full shadow-lg"
|
||||
>
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
viewBox="0 0 24 24"
|
||||
fill="currentColor"
|
||||
className="h-4 w-4 sm:h-5 sm:w-5 mr-1.5 sm:mr-2"
|
||||
className="h-5 w-5 mr-2"
|
||||
>
|
||||
<path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-2 14.5v-9l6 4.5-6 4.5z" />
|
||||
</svg>
|
||||
@@ -129,56 +261,56 @@ export default function AnimeDetails({ anime }) {
|
||||
</div>
|
||||
|
||||
{/* Title and Metadata */}
|
||||
<div className="flex-1 pt-3 md:pt-2">
|
||||
<div className="flex-1 pt-2">
|
||||
{/* Title Section */}
|
||||
<div className="text-center md:text-left">
|
||||
<h1 className="text-xl sm:text-2xl md:text-3xl lg:text-4xl font-bold text-white mb-1 sm:mb-2">
|
||||
<div className="text-left">
|
||||
<h1 className="text-3xl lg:text-4xl font-bold text-white mb-2">
|
||||
{info.name}
|
||||
</h1>
|
||||
|
||||
{moreInfo?.japanese && (
|
||||
<h2 className="text-sm sm:text-base md:text-lg text-gray-400 mb-1 sm:mb-2">{moreInfo.japanese}</h2>
|
||||
<h2 className="text-base md:text-lg text-gray-400 mb-2">{moreInfo.japanese}</h2>
|
||||
)}
|
||||
|
||||
{/* Synonyms */}
|
||||
{moreInfo?.synonyms && (
|
||||
<div className="mt-1 sm:mt-2 mb-2 sm:mb-4">
|
||||
<p className="text-xs sm:text-sm text-gray-400 italic">{moreInfo.synonyms}</p>
|
||||
<div className="mt-2 mb-4">
|
||||
<p className="text-sm text-gray-400 italic">{moreInfo.synonyms}</p>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Status Badges */}
|
||||
<div className="flex flex-wrap justify-center md:justify-start gap-1.5 sm:gap-2 my-4 sm:my-5">
|
||||
<div className="flex flex-wrap justify-start gap-2 my-4">
|
||||
{info.stats?.rating && (
|
||||
<div className="flex items-center bg-[var(--card)] px-2 sm:px-3 py-1 sm:py-1.5 rounded-full">
|
||||
<div className="flex items-center bg-[var(--card)] px-3 py-1.5 rounded-full">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
className="h-3.5 w-3.5 sm:h-4 sm:w-4 text-yellow-400 mr-1"
|
||||
className="h-4 w-4 text-yellow-400 mr-1"
|
||||
viewBox="0 0 20 20"
|
||||
fill="currentColor"
|
||||
>
|
||||
<path d="M9.049 2.927c.3-.921 1.603-.921 1.902 0l1.07 3.292a1 1 0 00.95.69h3.462c.969 0 1.371 1.24.588 1.81l-2.8 2.034a1 1 0 00-.364 1.118l1.07 3.292c.3.921-.755 1.688-1.54 1.118l-2.8-2.034a1 1 0 00-1.175 0l-2.8 2.034c-.784.57-1.838-.197-1.539-1.118l1.07-3.292a1 1 0 00-.364-1.118L2.98 8.72c-.783-.57-.38-1.81.588-1.81h3.461a1 1 0 00.951-.69l1.07-3.292z" />
|
||||
</svg>
|
||||
<span className="text-white text-xs sm:text-sm font-medium">{info.stats.rating}</span>
|
||||
<span className="text-white text-sm font-medium">{info.stats.rating}</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Status with Aired Date */}
|
||||
{moreInfo?.status && (
|
||||
<div className="bg-[var(--card)] px-2 sm:px-3 py-1 sm:py-1.5 rounded-full text-xs sm:text-sm text-white">
|
||||
<div className="bg-[var(--card)] px-3 py-1.5 rounded-full text-sm text-white">
|
||||
{getStatusWithAired()}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{info.stats?.type && (
|
||||
<div className="bg-[var(--card)] px-2 sm:px-3 py-1 sm:py-1.5 rounded-full text-xs sm:text-sm text-white">
|
||||
<div className="bg-[var(--card)] px-3 py-1.5 rounded-full text-sm text-white">
|
||||
{info.stats.type}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{info.stats?.episodes && (
|
||||
<div className="bg-[var(--card)] px-2 sm:px-3 py-1 sm:py-1.5 rounded-full text-xs sm:text-sm text-white">
|
||||
<div className="bg-[var(--card)] px-3 py-1.5 rounded-full text-sm text-white">
|
||||
{info.stats.episodes.sub > 0 && `SUB ${info.stats.episodes.sub}`}
|
||||
{info.stats.episodes.dub > 0 && info.stats.episodes.sub > 0 && ' | '}
|
||||
{info.stats.episodes.dub > 0 && `DUB ${info.stats.episodes.dub}`}
|
||||
@@ -186,30 +318,30 @@ export default function AnimeDetails({ anime }) {
|
||||
)}
|
||||
|
||||
{info.stats?.quality && (
|
||||
<div className="bg-[var(--card)] px-2 sm:px-3 py-1 sm:py-1.5 rounded-full text-xs sm:text-sm text-white">
|
||||
<div className="bg-[var(--card)] px-3 py-1.5 rounded-full text-sm text-white">
|
||||
{info.stats.quality}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{info.stats?.duration && (
|
||||
<div className="bg-[var(--card)] px-2 sm:px-3 py-1 sm:py-1.5 rounded-full text-xs sm:text-sm text-white">
|
||||
<div className="bg-[var(--card)] px-3 py-1.5 rounded-full text-sm text-white">
|
||||
{info.stats.duration}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Genres & Studios */}
|
||||
<div className="space-y-3 sm:space-y-4 mt-3 sm:mt-5">
|
||||
<div className="space-y-4 mt-4">
|
||||
{/* Genres */}
|
||||
{moreInfo?.genres && moreInfo.genres.length > 0 && (
|
||||
<div>
|
||||
<h3 className="text-white text-sm sm:text-base font-medium mb-2 sm:mb-3 text-center md:text-left">Genres</h3>
|
||||
<div className="flex flex-wrap justify-center md:justify-start gap-1.5 sm:gap-2">
|
||||
<h3 className="text-white text-base font-medium mb-3 text-left">Genres</h3>
|
||||
<div className="flex flex-wrap justify-start gap-2">
|
||||
{moreInfo.genres.map((genre, index) => (
|
||||
<Link
|
||||
key={index}
|
||||
href={`/genre/${genre.toLowerCase()}`}
|
||||
className="px-2 sm:px-3 py-1 sm:py-1.5 bg-[var(--card)] text-gray-300 text-xs sm:text-sm rounded-full whitespace-nowrap hover:text-white transition-colors hover:bg-[var(--card-hover)]"
|
||||
className="px-3 py-1.5 bg-[var(--card)] text-gray-300 text-sm rounded-full whitespace-nowrap hover:text-white transition-colors hover:bg-[var(--card-hover)]"
|
||||
>
|
||||
{genre}
|
||||
</Link>
|
||||
@@ -221,9 +353,9 @@ export default function AnimeDetails({ anime }) {
|
||||
{/* Studios */}
|
||||
{moreInfo?.studios && (
|
||||
<div>
|
||||
<h3 className="text-white text-sm sm:text-base font-medium mb-2 sm:mb-3 text-center md:text-left">Studios</h3>
|
||||
<div className="flex flex-wrap justify-center md:justify-start gap-1.5 sm:gap-2">
|
||||
<div className="px-2 sm:px-3 py-1 sm:py-1.5 bg-[var(--card)] text-gray-300 text-xs sm:text-sm rounded-full hover:text-white">
|
||||
<h3 className="text-white text-base font-medium mb-3 text-left">Studios</h3>
|
||||
<div className="flex flex-wrap justify-start gap-2">
|
||||
<div className="px-3 py-1.5 bg-[var(--card)] text-gray-300 text-sm rounded-full hover:text-white">
|
||||
{moreInfo.studios}
|
||||
</div>
|
||||
</div>
|
||||
@@ -233,51 +365,54 @@ export default function AnimeDetails({ anime }) {
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Details Tabs - Synopsis, Characters, Videos */}
|
||||
<div className="bg-[var(--card)] rounded-xl shadow-lg mb-6 sm:mb-8 border border-gray-800">
|
||||
{/* Tabs Section - Different for Mobile/Desktop */}
|
||||
<div className="bg-[var(--card)] rounded-lg mb-6 shadow-lg border border-gray-800">
|
||||
{/* Tab Navigation */}
|
||||
<div className="flex flex-wrap border-b border-gray-800">
|
||||
<div className="flex border-b border-gray-800">
|
||||
{/* Synopsis Tab */}
|
||||
<button
|
||||
className={`px-3 sm:px-5 py-2.5 sm:py-3 text-sm sm:text-base font-medium transition-colors ${activeTab === 'synopsis' ? 'text-white border-b-2 border-[var(--primary)]' : 'text-gray-400 hover:text-white'}`}
|
||||
className={`px-4 py-2.5 md:py-3 text-sm md:text-base font-medium transition-colors flex-1 md:flex-none ${activeTab === 'synopsis' ? 'text-white border-b-2 border-[var(--primary)]' : 'text-gray-400 hover:text-white'}`}
|
||||
onClick={() => setActiveTab('synopsis')}
|
||||
>
|
||||
Synopsis
|
||||
</button>
|
||||
|
||||
{/* Characters Tab */}
|
||||
{hasCharacters && (
|
||||
<button
|
||||
className={`px-3 sm:px-5 py-2.5 sm:py-3 text-sm sm:text-base font-medium transition-colors ${activeTab === 'characters' ? 'text-white border-b-2 border-[var(--primary)]' : 'text-gray-400 hover:text-white'}`}
|
||||
className={`px-4 py-2.5 md:py-3 text-sm md:text-base font-medium transition-colors flex-1 md:flex-none ${activeTab === 'characters' ? 'text-white border-b-2 border-[var(--primary)]' : 'text-gray-400 hover:text-white'}`}
|
||||
onClick={() => setActiveTab('characters')}
|
||||
>
|
||||
Characters
|
||||
</button>
|
||||
)}
|
||||
|
||||
{/* Videos Tab */}
|
||||
{hasVideos && (
|
||||
<button
|
||||
className={`px-3 sm:px-5 py-2.5 sm:py-3 text-sm sm:text-base font-medium transition-colors ${activeTab === 'videos' ? 'text-white border-b-2 border-[var(--primary)]' : 'text-gray-400 hover:text-white'}`}
|
||||
className={`px-4 py-2.5 md:py-3 text-sm md:text-base font-medium transition-colors flex-1 md:flex-none ${activeTab === 'videos' ? 'text-white border-b-2 border-[var(--primary)]' : 'text-gray-400 hover:text-white'}`}
|
||||
onClick={() => setActiveTab('videos')}
|
||||
>
|
||||
Promotional Videos
|
||||
<span>Videos</span>
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Tab Content */}
|
||||
<div className="p-3 sm:p-5">
|
||||
<div className="p-3 md:p-5">
|
||||
{/* Synopsis Tab */}
|
||||
{activeTab === 'synopsis' && (
|
||||
<div className="relative">
|
||||
<div>
|
||||
<p
|
||||
ref={synopsisRef}
|
||||
className={`text-gray-300 leading-relaxed text-xs sm:text-sm md:text-base ${!isExpanded ? 'line-clamp-4 md:line-clamp-6' : ''}`}
|
||||
className={`text-gray-300 leading-relaxed text-xs md:text-base ${!isExpanded ? 'line-clamp-4 md:line-clamp-6' : ''}`}
|
||||
>
|
||||
{info.description || 'No description available for this anime.'}
|
||||
</p>
|
||||
{synopsisOverflows && (
|
||||
<button
|
||||
onClick={() => setIsExpanded(!isExpanded)}
|
||||
className="text-[var(--primary)] hover:underline text-xs sm:text-sm mt-2 sm:mt-3 font-medium"
|
||||
className="text-[var(--primary)] hover:underline text-xs md:text-sm mt-2 md:mt-3 font-medium"
|
||||
>
|
||||
{isExpanded ? 'Show Less' : 'Read More'}
|
||||
</button>
|
||||
@@ -287,46 +422,44 @@ export default function AnimeDetails({ anime }) {
|
||||
|
||||
{/* Characters Tab */}
|
||||
{activeTab === 'characters' && hasCharacters && (
|
||||
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-3 gap-2 sm:gap-4 max-h-[70vh] overflow-y-auto pr-1">
|
||||
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-3 max-h-[60vh] md:max-h-[70vh] overflow-y-auto">
|
||||
{(info.characterVoiceActor || info.charactersVoiceActors || []).map((item, index) => (
|
||||
<div key={index} className="bg-[var(--background)] rounded overflow-hidden">
|
||||
<div className="flex">
|
||||
{/* Character Image - No padding */}
|
||||
<div className="relative w-[50px] sm:w-[60px] h-[60px] sm:h-[72px] flex-shrink-0">
|
||||
<Image
|
||||
src={item.character.poster}
|
||||
alt={item.character.name}
|
||||
fill
|
||||
className="object-cover"
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Text content in the middle */}
|
||||
<div className="flex-1 py-1.5 sm:py-2.5 px-2 sm:px-3 flex flex-col justify-center min-w-0">
|
||||
<div className="flex justify-between items-center gap-2 sm:gap-3">
|
||||
{/* Character Name */}
|
||||
<div className="min-w-0 flex-1">
|
||||
<p className="text-white font-medium text-xs sm:text-sm truncate">{item.character.name}</p>
|
||||
<p className="text-[10px] sm:text-xs text-gray-400 truncate">{item.character.cast || 'Main'}</p>
|
||||
</div>
|
||||
|
||||
{/* Voice Actor Name */}
|
||||
<div className="min-w-0 flex-1 text-right">
|
||||
<p className="text-white font-medium text-xs sm:text-sm truncate">{item.voiceActor.name}</p>
|
||||
<p className="text-[10px] sm:text-xs text-gray-400 truncate">{item.voiceActor.cast || 'Japanese'}</p>
|
||||
</div>
|
||||
<div key={index} className="bg-[var(--background)] rounded overflow-hidden flex">
|
||||
{/* Character Image */}
|
||||
<div className="relative w-[40px] md:w-[60px] h-[50px] md:h-[72px] flex-shrink-0">
|
||||
<Image
|
||||
src={item.character.poster}
|
||||
alt={item.character.name}
|
||||
fill
|
||||
className="object-cover"
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Text content in the middle */}
|
||||
<div className="flex-1 py-1 md:py-2.5 px-2 md:px-3 flex flex-col justify-center min-w-0">
|
||||
<div className="flex justify-between items-center gap-1 md:gap-3">
|
||||
{/* Character Name */}
|
||||
<div className="min-w-0 flex-1">
|
||||
<p className="text-white font-medium text-xs md:text-sm truncate">{item.character.name}</p>
|
||||
<p className="text-[10px] md:text-xs text-gray-400 truncate">{item.character.cast || 'Main'}</p>
|
||||
</div>
|
||||
|
||||
{/* Voice Actor Name */}
|
||||
<div className="min-w-0 flex-1 text-right">
|
||||
<p className="text-white font-medium text-xs md:text-sm truncate">{item.voiceActor.name}</p>
|
||||
<p className="text-[10px] md:text-xs text-gray-400 truncate">{item.voiceActor.cast || 'Japanese'}</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Voice Actor Image - No padding */}
|
||||
<div className="relative w-[50px] sm:w-[60px] h-[60px] sm:h-[72px] flex-shrink-0">
|
||||
<Image
|
||||
src={item.voiceActor.poster}
|
||||
alt={item.voiceActor.name}
|
||||
fill
|
||||
className="object-cover"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Voice Actor Image */}
|
||||
<div className="relative w-[40px] md:w-[60px] h-[50px] md:h-[72px] flex-shrink-0">
|
||||
<Image
|
||||
src={item.voiceActor.poster}
|
||||
alt={item.voiceActor.name}
|
||||
fill
|
||||
className="object-cover"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
@@ -335,16 +468,16 @@ export default function AnimeDetails({ anime }) {
|
||||
|
||||
{/* Videos Tab */}
|
||||
{activeTab === 'videos' && hasVideos && (
|
||||
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-2 sm:gap-3">
|
||||
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-3">
|
||||
{info.promotionalVideos.map((video, index) => (
|
||||
<div
|
||||
key={index}
|
||||
className="relative aspect-video cursor-pointer group overflow-hidden rounded-md"
|
||||
className="relative aspect-video cursor-pointer group overflow-hidden rounded"
|
||||
onClick={() => setActiveVideo(video)}
|
||||
>
|
||||
<div className="absolute inset-0 bg-black/40 group-hover:bg-black/20 transition-all duration-300 flex items-center justify-center">
|
||||
<div className="w-8 h-8 sm:w-10 sm:h-10 rounded-full bg-[var(--primary)] flex items-center justify-center transform group-hover:scale-110 transition-transform duration-300">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" className="h-4 w-4 sm:h-5 sm:w-5 text-white" viewBox="0 0 20 20" fill="currentColor">
|
||||
<div className="w-8 h-8 md:w-10 md:h-10 rounded-full bg-[var(--primary)] flex items-center justify-center transform group-hover:scale-110 transition-transform duration-300">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" className="h-4 w-4 md:h-5 md:w-5 text-white" viewBox="0 0 20 20" fill="currentColor">
|
||||
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM9.555 7.168A1 1 0 008 8v4a1 1 0 001.555.832l3-2a1 1 0 000-1.664l-3-2z" clipRule="evenodd" />
|
||||
</svg>
|
||||
</div>
|
||||
|
||||
@@ -174,13 +174,11 @@ export const fetchTrending = async () => {
|
||||
export const fetchAnimeInfo = async (id) => {
|
||||
try {
|
||||
if (!id) {
|
||||
console.error('Invalid anime ID provided');
|
||||
return null;
|
||||
}
|
||||
|
||||
const encodedId = encodeURIComponent(id);
|
||||
const url = `${API_BASE_URL}/anime/${encodedId}`;
|
||||
console.log('[API Call] Fetching anime info from:', url);
|
||||
|
||||
// Server-side fetch doesn't need credentials or mode settings
|
||||
const requestOptions = {
|
||||
@@ -192,37 +190,29 @@ export const fetchAnimeInfo = async (id) => {
|
||||
|
||||
// Handle failed requests gracefully
|
||||
if (!response.ok) {
|
||||
console.error(`[API Error] Status: ${response.status}`);
|
||||
return createFallbackAnimeData(id);
|
||||
}
|
||||
|
||||
// Parse the JSON response
|
||||
const data = await response.json();
|
||||
console.log('[API Response]', data);
|
||||
|
||||
// Check if the response is successful
|
||||
if (!data.success && data.status !== 200) {
|
||||
console.error('[API Error] Invalid response format:', data);
|
||||
return createFallbackAnimeData(id);
|
||||
}
|
||||
|
||||
// The data structure might be nested in different ways depending on the API
|
||||
const responseData = data.data || data;
|
||||
|
||||
// Log the data structure for debugging
|
||||
console.log('[API Data Structure]', JSON.stringify(responseData, null, 2));
|
||||
|
||||
// Extract the anime data from the response
|
||||
const animeData = responseData.anime;
|
||||
|
||||
if (!animeData) {
|
||||
console.error('[API Error] Missing anime data in response:', responseData);
|
||||
return createFallbackAnimeData(id);
|
||||
}
|
||||
|
||||
// Create mock characterVoiceActor data if missing
|
||||
if (!animeData.info?.characterVoiceActor || !Array.isArray(animeData.info?.characterVoiceActor) || animeData.info?.characterVoiceActor.length === 0) {
|
||||
console.log('[API Fix] Adding mock characterVoiceActor data');
|
||||
|
||||
// Ensure the info object exists
|
||||
if (!animeData.info) animeData.info = {};
|
||||
@@ -269,7 +259,6 @@ export const fetchAnimeInfo = async (id) => {
|
||||
|
||||
// Check the raw API response structure for characterVoiceActor
|
||||
if (animeData.info) {
|
||||
console.log('[API Debug] Raw info keys:', Object.keys(animeData.info));
|
||||
console.log('[API Debug] Raw charactersVoiceActors type:',
|
||||
animeData.info.charactersVoiceActors ?
|
||||
typeof animeData.info.charactersVoiceActors + ' ' +
|
||||
@@ -300,11 +289,9 @@ export const fetchAnimeInfo = async (id) => {
|
||||
// Explicit validation of charactersVoiceActors data (note the "s" in characters)
|
||||
const charData = animeData.info?.charactersVoiceActors;
|
||||
if (!charData) {
|
||||
console.warn('[API Warning] charactersVoiceActors is missing');
|
||||
return [];
|
||||
}
|
||||
if (!Array.isArray(charData)) {
|
||||
console.warn('[API Warning] charactersVoiceActors is not an array:', typeof charData);
|
||||
return [];
|
||||
}
|
||||
|
||||
@@ -374,7 +361,6 @@ export const fetchAnimeInfo = async (id) => {
|
||||
: []
|
||||
};
|
||||
} catch (error) {
|
||||
console.error('[API Error] Error fetching anime info:', error);
|
||||
return createFallbackAnimeData(id);
|
||||
}
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user