feat: add VideoGallery component to band pages
Some checks failed
Deploy Fediversion / deploy (push) Failing after 1s

- VideoGallery component with modal playback
- YouTube thumbnail extraction
- Responsive grid layout
- Added to band home pages
- Import script for video entities
This commit is contained in:
fullsizemalt 2025-12-30 19:32:35 -08:00
parent 1cb08bc778
commit 6d3b30ed6f
3 changed files with 294 additions and 0 deletions

View file

@ -0,0 +1,79 @@
"""
Import YouTube links from elmeg as Video entities
"""
import csv
import sys
sys.path.insert(0, '/Users/ten/DEV/fediversion/backend')
import psycopg2
from datetime import datetime
# Connect to fediversion database via SSH tunnel
# Run: ssh -L 5433:localhost:5432 nexus-vector
conn = psycopg2.connect(
host="localhost",
port=5433,
database="fediversion",
user="fediversion",
password="fediversion_password"
)
conn.autocommit = True
cur = conn.cursor()
# Get Goose vertical_id
cur.execute("SELECT id FROM vertical WHERE slug = 'goose'")
goose_vertical_id = cur.fetchone()[0]
print(f"Goose vertical_id: {goose_vertical_id}")
# Read CSV
videos_created = 0
links_created = 0
skipped = 0
with open('/tmp/perf_youtube_full.csv', 'r') as f:
reader = csv.DictReader(f)
for row in reader:
youtube_url = row['youtube_link']
show_slug = row['show_slug']
song_title = row['song_title']
show_date = row['show_date']
# Check if video already exists
cur.execute("SELECT id FROM video WHERE url = %s", (youtube_url,))
existing = cur.fetchone()
if existing:
video_id = existing[0]
else:
# Create video
cur.execute("""
INSERT INTO video (url, title, platform, video_type, vertical_id, created_at)
VALUES (%s, %s, 'youtube', 'single_song', %s, NOW())
RETURNING id
""", (youtube_url, f"{song_title} - {show_date}", goose_vertical_id))
video_id = cur.fetchone()[0]
videos_created += 1
# Find the show in fediversion
cur.execute("SELECT id FROM show WHERE slug = %s AND vertical_id = %s", (show_slug, goose_vertical_id))
show_result = cur.fetchone()
if show_result:
show_id = show_result[0]
# Link video to show
cur.execute("""
INSERT INTO videoshow (video_id, show_id)
VALUES (%s, %s)
ON CONFLICT DO NOTHING
""", (video_id, show_id))
links_created += 1
else:
skipped += 1
print(f"\nResults:")
print(f" Videos created: {videos_created}")
print(f" Show links created: {links_created}")
print(f" Shows not found: {skipped}")
cur.close()
conn.close()

View file

@ -5,6 +5,7 @@ import Link from "next/link"
import { Calendar, MapPin, Music, Trophy, Video, Ticket, Building, ChevronRight, Play } from "lucide-react"
import { Card, CardContent } from "@/components/ui/card"
import { Button } from "@/components/ui/button"
import { VideoGallery } from "@/components/videos/video-gallery"
interface Props {
params: Promise<{ vertical: string }>
@ -173,6 +174,15 @@ export default async function VerticalPage({ params }: Props) {
</section>
)}
{/* Videos Section */}
<section className="space-y-4">
<VideoGallery
bandSlug={verticalSlug}
limit={8}
title="Recent Videos"
/>
</section>
{/* Navigation Cards */}
<section className="space-y-4">
<h2 className="text-2xl font-bold tracking-tight">Explore</h2>

View file

@ -0,0 +1,205 @@
"use client"
import { useState, useEffect } from "react"
import { Card, CardContent } from "@/components/ui/card"
import { Button } from "@/components/ui/button"
import { Play, ExternalLink, Video, Music } from "lucide-react"
import { getApiUrl } from "@/lib/api-config"
interface VideoItem {
id: number
url: string
title: string | null
description: string | null
platform: string
video_type: string
duration_seconds: number | null
thumbnail_url: string | null
external_id: string | null
}
interface VideoGalleryProps {
bandSlug?: string
showId?: number
songId?: number
limit?: number
title?: string
}
function extractYouTubeId(url: string): string | null {
const patterns = [
/(?:youtube\.com\/watch\?v=|youtu\.be\/|youtube\.com\/embed\/)([a-zA-Z0-9_-]{11})/,
]
for (const pattern of patterns) {
const match = url.match(pattern)
if (match) return match[1]
}
return null
}
function getThumbnailUrl(video: VideoItem): string {
if (video.thumbnail_url) return video.thumbnail_url
if (video.platform === "youtube") {
const videoId = extractYouTubeId(video.url)
if (videoId) return `https://img.youtube.com/vi/${videoId}/mqdefault.jpg`
}
return "/placeholder-video.jpg"
}
export function VideoGallery({
bandSlug,
showId,
songId,
limit = 12,
title = "Videos"
}: VideoGalleryProps) {
const [videos, setVideos] = useState<VideoItem[]>([])
const [loading, setLoading] = useState(true)
const [selectedVideo, setSelectedVideo] = useState<VideoItem | null>(null)
useEffect(() => {
async function fetchVideos() {
try {
let endpoint = ""
if (bandSlug) {
endpoint = `${getApiUrl()}/videos/by-band/${bandSlug}?limit=${limit}`
} else if (showId) {
endpoint = `${getApiUrl()}/videos/by-show/${showId}`
} else if (songId) {
endpoint = `${getApiUrl()}/videos/by-song/${songId}`
} else {
endpoint = `${getApiUrl()}/videos/?limit=${limit}`
}
const res = await fetch(endpoint)
if (res.ok) {
const data = await res.json()
setVideos(data)
}
} catch (error) {
console.error("Failed to fetch videos:", error)
} finally {
setLoading(false)
}
}
fetchVideos()
}, [bandSlug, showId, songId, limit])
if (loading) {
return (
<div className="space-y-4">
<h2 className="text-xl font-semibold flex items-center gap-2">
<Video className="h-5 w-5 text-primary" />
{title}
</h2>
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">
{[...Array(4)].map((_, i) => (
<div key={i} className="aspect-video bg-muted animate-pulse rounded-lg" />
))}
</div>
</div>
)
}
if (videos.length === 0) {
return null // Don't show empty section
}
return (
<div className="space-y-4">
<h2 className="text-xl font-semibold flex items-center gap-2">
<Video className="h-5 w-5 text-primary" />
{title}
<span className="text-sm font-normal text-muted-foreground">({videos.length})</span>
</h2>
{/* Video modal */}
{selectedVideo && (
<div
className="fixed inset-0 bg-black/80 z-50 flex items-center justify-center p-4"
onClick={() => setSelectedVideo(null)}
>
<div className="w-full max-w-4xl" onClick={(e) => e.stopPropagation()}>
<div className="aspect-video bg-black rounded-lg overflow-hidden">
{selectedVideo.platform === "youtube" && (
<iframe
src={`https://www.youtube.com/embed/${extractYouTubeId(selectedVideo.url)}?autoplay=1`}
className="w-full h-full"
allowFullScreen
allow="autoplay"
/>
)}
</div>
<div className="mt-4 text-white">
<h3 className="text-lg font-semibold">{selectedVideo.title || "Video"}</h3>
<Button
variant="ghost"
className="mt-2 text-white hover:text-white/80"
onClick={() => setSelectedVideo(null)}
>
Close
</Button>
</div>
</div>
</div>
)}
{/* Video grid */}
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">
{videos.map((video) => (
<Card
key={video.id}
className="group cursor-pointer hover:border-primary/50 transition-colors overflow-hidden"
onClick={() => setSelectedVideo(video)}
>
<div className="aspect-video relative bg-muted">
{/* Thumbnail */}
<img
src={getThumbnailUrl(video)}
alt={video.title || "Video"}
className="w-full h-full object-cover"
/>
{/* Play overlay */}
<div className="absolute inset-0 bg-black/30 opacity-0 group-hover:opacity-100 transition-opacity flex items-center justify-center">
<div className="w-12 h-12 rounded-full bg-primary/90 flex items-center justify-center">
<Play className="h-6 w-6 text-white fill-white" />
</div>
</div>
{/* Platform badge */}
<div className="absolute top-2 right-2 px-2 py-1 rounded text-xs font-medium bg-black/60 text-white">
{video.platform === "youtube" ? "YouTube" : video.platform}
</div>
</div>
<CardContent className="p-3">
<h3 className="text-sm font-medium line-clamp-2">
{video.title || "Untitled Video"}
</h3>
<div className="flex items-center gap-1 mt-1 text-xs text-muted-foreground">
<Music className="h-3 w-3" />
<span>{video.video_type === "single_song" ? "Song" : video.video_type}</span>
</div>
</CardContent>
</Card>
))}
</div>
{/* View all link */}
{videos.length >= limit && bandSlug && (
<div className="text-center">
<Button variant="outline" asChild>
<a href={`/${bandSlug}/videos`}>
View All Videos
<ExternalLink className="ml-2 h-4 w-4" />
</a>
</Button>
</div>
)}
</div>
)
}