"use client"
import { useState } from "react"
import Link from "next/link"
import { Badge } from "@/components/ui/badge"
import { Button } from "@/components/ui/button"
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
import { RatePerformanceDialog } from "@/components/songs/rate-performance-dialog"
import { ArrowUpDown, Star, Calendar, Music } from "lucide-react"
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
export interface Performance {
id: number
show_id: number
show_slug?: string
song_id: number
position: number
set_name: string | null
segue: boolean
notes: string | null
show_date: string
venue_name: string
venue_city: string
venue_state: string | null
avg_rating: number
total_reviews: number
}
// ...
// In JSX:
{new Date(perf.show_date).toLocaleDateString(undefined, {
year: 'numeric',
month: 'short',
day: 'numeric',
weekday: 'short'
})}
{perf.set_name || "Set ?"}
interface PerformanceListProps {
performances: Performance[]
songTitle?: string
}
type SortOption = "date_desc" | "date_asc" | "rating_desc"
export function PerformanceList({ performances, songTitle }: PerformanceListProps) {
const [sort, setSort] = useState("date_desc")
const sortedPerformances = [...performances].sort((a, b) => {
if (sort === "date_desc") {
return new Date(b.show_date).getTime() - new Date(a.show_date).getTime()
}
if (sort === "date_asc") {
return new Date(a.show_date).getTime() - new Date(b.show_date).getTime()
}
if (sort === "rating_desc") {
// Primary: Rating, Secondary: Review Count, Tertiary: Date
if (b.avg_rating !== a.avg_rating) return b.avg_rating - a.avg_rating
if (b.total_reviews !== a.total_reviews) return b.total_reviews - a.total_reviews
return new Date(b.show_date).getTime() - new Date(a.show_date).getTime()
}
return 0
})
return (
Performance History
{performances.length}
Sort by:
{sortedPerformances.length > 0 ? (
{sortedPerformances.map((perf) => (
{new Date(perf.show_date).toLocaleDateString(undefined, {
year: 'numeric',
month: 'short',
day: 'numeric',
weekday: 'short'
})}
{perf.set_name || "Set ?"}
{perf.venue_name} • {perf.venue_city}{perf.venue_state ? `, ${perf.venue_state}` : ""}
{perf.notes && (
"{perf.notes}"
)}
{perf.avg_rating > 0 && (
= 4.5 ? 'border-yellow-500/50 bg-yellow-500/10 text-yellow-600' : 'border-muted'}
`}
>
= 4.5 ? 'fill-yellow-600' : ''}`} />
{perf.avg_rating.toFixed(1)}
{perf.total_reviews} review{perf.total_reviews !== 1 ? 's' : ''}
)}
{
// Optional: trigger refresh
window.location.reload()
}}
/>
))}
) : (
No performances recorded yet.
)}
)
}