fediversion/frontend/app/shows/[slug]/page.tsx
fullsizemalt 265200b6ad
Some checks failed
Deploy Fediversion / deploy (push) Failing after 1s
fix: change back button from /archive to /shows
2025-12-30 18:53:09 -08:00

395 lines
23 KiB
TypeScript

import { Button } from "@/components/ui/button"
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
import { ArrowLeft, MapPin, Music2, Disc, PlayCircle, Youtube } from "lucide-react"
import Link from "next/link"
import { CommentSection } from "@/components/social/comment-section"
import { EntityRating } from "@/components/social/entity-rating"
import { ShowAttendance } from "@/components/shows/show-attendance"
import { SocialWrapper } from "@/components/social/social-wrapper"
import { notFound } from "next/navigation"
import { SuggestNicknameDialog } from "@/components/shows/suggest-nickname-dialog"
import { EntityReviews } from "@/components/reviews/entity-reviews"
import { getApiUrl } from "@/lib/api-config"
import { YouTubeEmbed } from "@/components/ui/youtube-embed"
import { MarkCaughtButton } from "@/components/chase/mark-caught-button"
import { AddToPlaylistDialog } from "@/components/playlists/add-to-playlist-dialog"
async function getShow(id: string) {
try {
const res = await fetch(`${getApiUrl()}/shows/${id}`, { cache: 'no-store' })
if (!res.ok) return null
return res.json()
} catch (e) {
console.error(e)
return null
}
}
export default async function ShowDetailPage({ params }: { params: Promise<{ slug: string }> }) {
const { slug } = await params
const show = await getShow(slug)
if (!show) {
notFound()
}
// Group by set
const sets: Record<string, any[]> = {};
if (show.performances) {
show.performances.forEach((perf: any) => {
const setName = perf.set_name || "Set 1"; // Default to Set 1 if missing
if (!sets[setName]) sets[setName] = [];
sets[setName].push(perf);
});
}
// Sort keys: Set 1, Set 2, Set 3, Encore, Encore 2...
const sortedKeys = Object.keys(sets).sort((a, b) => {
const aLower = a.toLowerCase();
const bLower = b.toLowerCase();
// Encore always last
if (aLower.includes("encore") && !bLower.includes("encore")) return 1;
if (!aLower.includes("encore") && bLower.includes("encore")) return -1;
// If both have Set, compare numbers
if (aLower.includes("set") && bLower.includes("set")) {
const aNum = parseInt(a.replace(/\D/g, "") || "0");
const bNum = parseInt(b.replace(/\D/g, "") || "0");
return aNum - bNum;
}
return a.localeCompare(b);
});
return (
<div className="flex flex-col gap-6">
<div className="flex flex-col gap-4 md:flex-row md:items-center md:justify-between">
<div className="flex items-center gap-4">
<Link href="/shows">
<Button variant="ghost" size="icon">
<ArrowLeft className="h-4 w-4" />
</Button>
</Link>
<div>
{/* Band Name - Most Important */}
{show.vertical && (
<Link
href={`/bands/${show.vertical.slug}`}
className="inline-flex items-center gap-2 text-sm font-semibold text-primary hover:underline mb-1"
>
<Music2 className="h-4 w-4" />
{show.vertical.name}
</Link>
)}
<h1 className="text-2xl sm:text-3xl font-bold tracking-tight">
{new Date(show.date).toLocaleDateString()}
</h1>
{show.venue && (
<p className="text-base sm:text-lg text-muted-foreground mt-1">
{show.venue.name}, {show.venue.city}, {show.venue.state}
</p>
)}
{show.tags && show.tags.length > 0 && (
<div className="flex flex-wrap gap-2 mt-1">
{show.tags.map((tag: any) => (
<span key={tag.id} className="bg-secondary text-secondary-foreground px-2 py-0.5 rounded-full text-xs font-medium">
#{tag.name}
</span>
))}
</div>
)}
<div className="flex items-center flex-wrap gap-4 mt-2">
{show.tour && (
<p className="text-muted-foreground flex items-center gap-2">
<Music2 className="h-4 w-4" />
<Link href={`/tours/${show.tour.slug || show.tour.id}`} className="hover:underline">
{show.tour.name}
</Link>
</p>
)}
</div>
</div>
</div>
</div>
{show.notes && (
<div className="bg-muted/50 p-4 rounded-lg border text-sm italic">
Note: {show.notes}
</div>
)}
<div className="grid gap-6 md:grid-cols-[2fr_1fr]">
<div className="flex flex-col gap-6">
{/* Full Show Video */}
{show.youtube_link && (
<Card className="border-2 border-red-500/20 bg-gradient-to-br from-red-500/5 to-transparent">
<CardHeader className="pb-2">
<CardTitle className="text-sm font-medium text-muted-foreground flex items-center gap-2">
<Youtube className="h-4 w-4 text-red-500" />
Full Show Video
</CardTitle>
</CardHeader>
<CardContent>
<YouTubeEmbed url={show.youtube_link} title={`${show.date?.split('T')[0]} - ${show.venue?.name}`} />
</CardContent>
</Card>
)}
<Card>
<CardHeader>
<CardTitle>Setlist</CardTitle>
</CardHeader>
<CardContent>
{show.performances && show.performances.length > 0 ? (
<div>
{sortedKeys.map((setName) => (
<div key={setName} className="mb-6 last:mb-0">
<h3 className="font-semibold text-sm uppercase tracking-wider text-muted-foreground mb-3 pl-2 border-b pb-1">
{setName}
</h3>
<div className="space-y-1">
{sets[setName].map((perf: any) => (
<div key={perf.id} className="flex flex-col group py-1.5 hover:bg-muted/50 rounded px-2 -mx-2 transition-colors">
<div className="flex items-center justify-between">
<div className="flex items-center gap-3">
<span className="text-muted-foreground/60 w-6 text-right text-xs font-mono">{perf.position}.</span>
<div className="font-medium flex items-center gap-2">
<Link
href={`/performances/${perf.slug}`}
className="hover:text-primary hover:underline transition-colors"
>
{perf.song?.title || "Unknown Song"}
</Link>
{perf.track_url && (
<a
href={perf.track_url}
target="_blank"
rel="noopener noreferrer"
className="text-muted-foreground hover:text-primary"
title="Listen"
>
<PlayCircle className="h-3.5 w-3.5" />
</a>
)}
{perf.youtube_link && (
<span
className="text-red-500"
title="Video available"
>
<Youtube className="h-3.5 w-3.5" />
</span>
)}
{perf.bandcamp_link && (
<a
href={perf.bandcamp_link}
target="_blank"
rel="noopener noreferrer"
className="text-[#629aa9] hover:text-[#4a7a89]"
title="Listen on Bandcamp"
>
<Disc className="h-3.5 w-3.5" />
</a>
)}
{perf.nugs_link && (
<a
href={perf.nugs_link}
target="_blank"
rel="noopener noreferrer"
className="text-[#ff6b00] hover:text-[#cc5500]"
title="Listen on Nugs.net"
>
<PlayCircle className="h-3.5 w-3.5" />
</a>
)}
{perf.segue && <span className="ml-1 text-muted-foreground">&gt;</span>}
</div>
{/* Nicknames */}
{perf.nicknames && perf.nicknames.length > 0 && (
<div className="flex gap-1 ml-2">
{perf.nicknames.map((nick: any) => (
<span key={nick.id} className="text-[10px] bg-yellow-100/80 text-yellow-800 px-1.5 py-0.5 rounded-full border border-yellow-200" title={nick.description}>
&quot;{nick.nickname}&quot;
</span>
))}
</div>
)}
{/* Suggest Nickname Button */}
<div className="opacity-50 md:opacity-30 md:group-hover:opacity-100 transition-opacity">
<SuggestNicknameDialog
performanceId={perf.id}
songTitle={perf.song?.title || "Song"}
/>
</div>
</div>
{/* Rating Column */}
<SocialWrapper type="ratings">
<EntityRating
entityType="performance"
entityId={perf.id}
compact={true}
/>
</SocialWrapper>
{/* Mark Caught (for chase songs) */}
<MarkCaughtButton
songId={perf.song?.id}
songTitle={perf.song?.title || "Song"}
showId={show.id}
/>
{/* Add to Playlist */}
<div className="opacity-0 group-hover:opacity-100 transition-opacity">
<AddToPlaylistDialog
performanceId={perf.id}
songTitle={perf.song?.title || "Song"}
/>
</div>
</div>
{perf.notes && (
<div className="text-xs text-muted-foreground ml-9 italic mt-0.5">
{perf.notes}
</div>
)}
</div>
))}
</div>
</div>
))}
</div>
) : (
<div className="flex flex-col items-center justify-center py-8 text-center">
<Music2 className="h-12 w-12 text-muted-foreground/30 mb-4" />
<p className="text-muted-foreground font-medium">No Setlist Documented</p>
<p className="text-sm text-muted-foreground/70 mt-1 max-w-sm">
This show&apos;s setlist hasn&apos;t been added yet. Early shows often weren&apos;t documented.
</p>
</div>
)}
</CardContent>
</Card>
<SocialWrapper type="comments">
<CommentSection entityType="show" entityId={show.id} />
</SocialWrapper>
<SocialWrapper type="reviews">
<EntityReviews entityType="show" entityId={show.id} />
</SocialWrapper>
</div>
<div className="flex flex-col gap-4">
{/* Venue Info Card */}
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm">Venue</CardTitle>
</CardHeader>
<CardContent className="space-y-1">
{show.venue ? (
<>
<div className="flex items-center gap-2">
<MapPin className="h-4 w-4 text-muted-foreground" />
<Link href={`/venues/${show.venue.slug}`} className="font-medium hover:underline hover:text-primary">
{show.venue.name}
</Link>
</div>
<p className="text-sm text-muted-foreground pl-6">
{show.venue.city}, {show.venue.state} {show.venue.country}
</p>
</>
) : (
<p className="text-sm text-muted-foreground">Unknown Venue</p>
)}
</CardContent>
</Card>
{/* Listen On Card */}
{(show.nugs_link || show.bandcamp_link) && (
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm">Listen On</CardTitle>
</CardHeader>
<CardContent className="space-y-2">
{show.nugs_link && (
<a
href={show.nugs_link}
target="_blank"
rel="noopener noreferrer"
className="flex items-center gap-3 p-2 rounded-lg bg-orange-500/10 hover:bg-orange-500/20 border border-orange-500/20 transition-colors"
>
<PlayCircle className="h-5 w-5 text-orange-500" />
<div>
<p className="font-medium text-sm">Nugs.net</p>
<p className="text-xs text-muted-foreground">Stream or download</p>
</div>
</a>
)}
{show.bandcamp_link && (
<a
href={show.bandcamp_link}
target="_blank"
rel="noopener noreferrer"
className="flex items-center gap-3 p-2 rounded-lg bg-blue-500/10 hover:bg-blue-500/20 border border-blue-500/20 transition-colors"
>
<Disc className="h-5 w-5 text-blue-500" />
<div>
<p className="font-medium text-sm">Bandcamp</p>
<p className="text-xs text-muted-foreground">Official release</p>
</div>
</a>
)}
</CardContent>
</Card>
)}
{/* Tour Info */}
{show.tour && (
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm">Tour</CardTitle>
</CardHeader>
<CardContent>
<div className="flex items-center gap-2">
<Music2 className="h-4 w-4 text-muted-foreground" />
<Link href={`/tours/${show.tour.slug || show.tour.id}`} className="font-medium hover:underline hover:text-primary">
{show.tour.name}
</Link>
</div>
</CardContent>
</Card>
)}
{/* Attendance */}
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm">I Was There</CardTitle>
</CardHeader>
<CardContent>
<ShowAttendance showId={show.id} />
</CardContent>
</Card>
{/* Rate This Show */}
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm">Rate This Show</CardTitle>
</CardHeader>
<CardContent>
<SocialWrapper type="ratings">
<EntityRating
entityType="show"
entityId={show.id}
compact={false}
/>
</SocialWrapper>
</CardContent>
</Card>
</div>
</div>
</div >
)
}