643 lines
35 KiB
TypeScript
643 lines
35 KiB
TypeScript
import { Button } from "@/components/ui/button";
|
|
import { Card, CardContent } from "@/components/ui/card";
|
|
import { Download, FolderOpen, ImageDown, FileText, BadgeCheck, XCircle, Filter } from "lucide-react";
|
|
import { Spinner } from "@/components/ui/spinner";
|
|
import { Tooltip, TooltipContent, TooltipTrigger } from "@/components/ui/tooltip";
|
|
import { SearchAndSort } from "./SearchAndSort";
|
|
import { TrackList } from "./TrackList";
|
|
import { DownloadProgress } from "./DownloadProgress";
|
|
import type { TrackMetadata, TrackAvailability } from "@/types/api";
|
|
import { downloadHeader, downloadGalleryImage, downloadAvatar } from "@/lib/api";
|
|
import { getSettings } from "@/lib/settings";
|
|
import { toastWithSound as toast } from "@/lib/toast-with-sound";
|
|
import { useState, useMemo } from "react";
|
|
import { Checkbox } from "@/components/ui/checkbox";
|
|
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogTrigger } from "@/components/ui/dialog";
|
|
import { ScrollArea } from "@/components/ui/scroll-area";
|
|
interface ArtistInfoProps {
|
|
artistInfo: {
|
|
name: string;
|
|
images: string;
|
|
header?: string;
|
|
gallery?: string[];
|
|
followers: number;
|
|
total_albums?: number;
|
|
genres: string[];
|
|
biography?: string;
|
|
verified?: boolean;
|
|
listeners?: number;
|
|
rank?: number;
|
|
};
|
|
albumList: Array<{
|
|
id: string;
|
|
name: string;
|
|
images: string;
|
|
release_date: string;
|
|
album_type: string;
|
|
external_urls: string;
|
|
total_tracks?: number;
|
|
}>;
|
|
trackList: TrackMetadata[];
|
|
searchQuery: string;
|
|
sortBy: string;
|
|
selectedTracks: string[];
|
|
downloadedTracks: Set<string>;
|
|
failedTracks: Set<string>;
|
|
skippedTracks: Set<string>;
|
|
downloadingTrack: string | null;
|
|
isDownloading: boolean;
|
|
bulkDownloadType: "all" | "selected" | null;
|
|
downloadProgress: number;
|
|
currentDownloadInfo: {
|
|
name: string;
|
|
artists: string;
|
|
} | null;
|
|
currentPage: number;
|
|
itemsPerPage: number;
|
|
downloadedLyrics?: Set<string>;
|
|
failedLyrics?: Set<string>;
|
|
skippedLyrics?: Set<string>;
|
|
downloadingLyricsTrack?: string | null;
|
|
checkingAvailabilityTrack?: string | null;
|
|
availabilityMap?: Map<string, TrackAvailability>;
|
|
downloadedCovers?: Set<string>;
|
|
failedCovers?: Set<string>;
|
|
skippedCovers?: Set<string>;
|
|
downloadingCoverTrack?: string | null;
|
|
isBulkDownloadingCovers?: boolean;
|
|
isBulkDownloadingLyrics?: boolean;
|
|
isMetadataLoading?: boolean;
|
|
onSearchChange: (value: string) => void;
|
|
onSortChange: (value: string) => void;
|
|
onToggleTrack: (id: string) => void;
|
|
onToggleSelectAll: (tracks: TrackMetadata[]) => void;
|
|
onDownloadTrack: (id: string, name: string, artists: string, albumName: string, spotifyId?: string, folderName?: string, durationMs?: number, position?: number, albumArtist?: string, releaseDate?: string, coverUrl?: string, spotifyTrackNumber?: number, spotifyDiscNumber?: number, spotifyTotalTracks?: number, spotifyTotalDiscs?: number, copyright?: string, publisher?: string) => void;
|
|
onDownloadLyrics?: (spotifyId: string, name: string, artists: string, albumName: string, folderName?: string, isArtistDiscography?: boolean, position?: number, albumArtist?: string, releaseDate?: string, discNumber?: number) => void;
|
|
onDownloadCover?: (coverUrl: string, trackName: string, artistName: string, albumName: string, folderName?: string, isArtistDiscography?: boolean, position?: number, trackId?: string, albumArtist?: string, releaseDate?: string, discNumber?: number) => void;
|
|
onCheckAvailability?: (spotifyId: string) => void;
|
|
onDownloadAllLyrics?: () => void;
|
|
onDownloadAllCovers?: () => void;
|
|
onDownloadAll: () => void;
|
|
onDownloadSelected: () => void;
|
|
onStopDownload: () => void;
|
|
onOpenFolder: () => void;
|
|
onAlbumClick: (album: {
|
|
id: string;
|
|
name: string;
|
|
external_urls: string;
|
|
}) => void;
|
|
onArtistClick: (artist: {
|
|
id: string;
|
|
name: string;
|
|
external_urls: string;
|
|
}) => void;
|
|
onPageChange: (page: number) => void;
|
|
onTrackClick?: (track: TrackMetadata) => void;
|
|
onBack?: () => void;
|
|
}
|
|
export function ArtistInfo({ artistInfo, albumList, trackList, searchQuery, sortBy, selectedTracks, downloadedTracks, failedTracks, skippedTracks, downloadingTrack, isDownloading, bulkDownloadType, downloadProgress, currentDownloadInfo, currentPage, itemsPerPage, downloadedLyrics, failedLyrics, skippedLyrics, downloadingLyricsTrack, checkingAvailabilityTrack, availabilityMap, downloadedCovers, failedCovers, skippedCovers, downloadingCoverTrack, isBulkDownloadingCovers, isBulkDownloadingLyrics, isMetadataLoading = false, onSearchChange, onSortChange, onToggleTrack, onToggleSelectAll, onDownloadTrack, onDownloadLyrics, onDownloadCover, onCheckAvailability, onDownloadAllLyrics, onDownloadAllCovers, onDownloadAll, onDownloadSelected, onStopDownload, onOpenFolder, onAlbumClick, onArtistClick, onPageChange, onTrackClick, onBack, }: ArtistInfoProps) {
|
|
const [downloadingHeader, setDownloadingHeader] = useState(false);
|
|
const [downloadingAvatar, setDownloadingAvatar] = useState(false);
|
|
const [downloadingGalleryIndex, setDownloadingGalleryIndex] = useState<number | null>(null);
|
|
const [downloadingAllGallery, setDownloadingAllGallery] = useState(false);
|
|
const [activeTab, setActiveTab] = useState<"albums" | "tracks" | "gallery">("albums");
|
|
const [activeAlbumFilter, setActiveAlbumFilter] = useState<string>("all");
|
|
const displayedAlbumCount = artistInfo.total_albums || albumList.length;
|
|
const fetchedAlbumCount = albumList.length;
|
|
const totalAlbumCount = artistInfo.total_albums || fetchedAlbumCount;
|
|
const totalTrackCount = albumList.reduce((sum, album) => sum + (album.total_tracks || 0), 0);
|
|
const fetchedTrackCount = trackList.length;
|
|
const albumCountLabel = isMetadataLoading && totalAlbumCount > 0 && fetchedAlbumCount < totalAlbumCount
|
|
? `${fetchedAlbumCount.toLocaleString()} / ${totalAlbumCount.toLocaleString()} albums`
|
|
: `${displayedAlbumCount.toLocaleString()} ${displayedAlbumCount === 1 ? "album" : "albums"}`;
|
|
const resolvedTrackCount = totalTrackCount > 0 ? totalTrackCount : fetchedTrackCount;
|
|
const trackCountLabel = isMetadataLoading && totalTrackCount > 0 && fetchedTrackCount < totalTrackCount
|
|
? `${fetchedTrackCount.toLocaleString()} / ${totalTrackCount.toLocaleString()} tracks`
|
|
: `${resolvedTrackCount.toLocaleString()} ${resolvedTrackCount === 1 ? "track" : "tracks"}`;
|
|
const albumFilterCounts = useMemo(() => {
|
|
const counts = new Map<string, number>();
|
|
counts.set("all", (albumList || []).length);
|
|
for (const album of albumList || []) {
|
|
const type = (album.album_type || "").trim().toLowerCase();
|
|
if (!type)
|
|
continue;
|
|
counts.set(type, (counts.get(type) || 0) + 1);
|
|
}
|
|
return counts;
|
|
}, [albumList]);
|
|
const albumFilters = useMemo(() => {
|
|
const uniqueTypes = Array.from(new Set((albumList || [])
|
|
.map((album) => (album.album_type || "").trim().toLowerCase())
|
|
.filter(Boolean)));
|
|
return ["all", ...uniqueTypes];
|
|
}, [albumList]);
|
|
const filteredAlbums = useMemo(() => {
|
|
if (activeAlbumFilter === "all") {
|
|
return albumList || [];
|
|
}
|
|
return (albumList || []).filter((album) => (album.album_type || "").trim().toLowerCase() === activeAlbumFilter);
|
|
}, [albumList, activeAlbumFilter]);
|
|
const filteredAlbumGroups = useMemo(() => {
|
|
const albumTypeMap = new Map(albumList.map(a => [a.name, a.album_type]));
|
|
const albumGroups = trackList.reduce((acc, track) => {
|
|
if (!track.album_name)
|
|
return acc;
|
|
if (!acc[track.album_name]) {
|
|
acc[track.album_name] = {
|
|
count: 0,
|
|
tracks: [],
|
|
type: albumTypeMap.get(track.album_name) || "unknown"
|
|
};
|
|
}
|
|
acc[track.album_name].count++;
|
|
acc[track.album_name].tracks.push(track);
|
|
return acc;
|
|
}, {} as Record<string, {
|
|
count: number;
|
|
tracks: TrackMetadata[];
|
|
type: string;
|
|
}>);
|
|
return Object.entries(albumGroups).sort((a, b) => {
|
|
const dateA = a[1].tracks[0]?.release_date || "";
|
|
const dateB = b[1].tracks[0]?.release_date || "";
|
|
return dateB.localeCompare(dateA);
|
|
});
|
|
}, [trackList, albumList]);
|
|
const formatAlbumFilterLabel = (value: string) => {
|
|
const count = albumFilterCounts.get(value) || 0;
|
|
if (value === "all")
|
|
return `All (${count})`;
|
|
const label = value
|
|
.split(/[_\s]+/)
|
|
.filter(Boolean)
|
|
.map((part) => part.charAt(0).toUpperCase() + part.slice(1))
|
|
.join(" ");
|
|
return `${label} (${count})`;
|
|
};
|
|
const handleDownloadHeader = async () => {
|
|
if (!artistInfo.header)
|
|
return;
|
|
setDownloadingHeader(true);
|
|
try {
|
|
const settings = getSettings();
|
|
const response = await downloadHeader({
|
|
header_url: artistInfo.header,
|
|
artist_name: artistInfo.name,
|
|
output_dir: settings.downloadPath,
|
|
});
|
|
if (response.success) {
|
|
if (response.already_exists) {
|
|
toast.info("Header already exists");
|
|
}
|
|
else {
|
|
toast.success("Header downloaded successfully");
|
|
}
|
|
}
|
|
else {
|
|
toast.error(response.error || "Failed to download header");
|
|
}
|
|
}
|
|
catch (error) {
|
|
toast.error(`Error downloading header: ${error}`);
|
|
}
|
|
finally {
|
|
setDownloadingHeader(false);
|
|
}
|
|
};
|
|
const handleDownloadAvatar = async () => {
|
|
if (!artistInfo.images)
|
|
return;
|
|
setDownloadingAvatar(true);
|
|
try {
|
|
const settings = getSettings();
|
|
const response = await downloadAvatar({
|
|
avatar_url: artistInfo.images,
|
|
artist_name: artistInfo.name,
|
|
output_dir: settings.downloadPath,
|
|
});
|
|
if (response.success) {
|
|
if (response.already_exists) {
|
|
toast.info("Avatar already exists");
|
|
}
|
|
else {
|
|
toast.success("Avatar downloaded successfully");
|
|
}
|
|
}
|
|
else {
|
|
toast.error(response.error || "Failed to download avatar");
|
|
}
|
|
}
|
|
catch (error) {
|
|
toast.error(`Error downloading avatar: ${error}`);
|
|
}
|
|
finally {
|
|
setDownloadingAvatar(false);
|
|
}
|
|
};
|
|
const handleDownloadGalleryImage = async (imageUrl: string, index: number) => {
|
|
setDownloadingGalleryIndex(index);
|
|
try {
|
|
const settings = getSettings();
|
|
const response = await downloadGalleryImage({
|
|
image_url: imageUrl,
|
|
artist_name: artistInfo.name,
|
|
image_index: index,
|
|
output_dir: settings.downloadPath,
|
|
});
|
|
if (response.success) {
|
|
if (response.already_exists) {
|
|
toast.info(`Gallery image ${index + 1} already exists`);
|
|
}
|
|
else {
|
|
toast.success(`Gallery image ${index + 1} downloaded successfully`);
|
|
}
|
|
}
|
|
else {
|
|
toast.error(response.error || `Failed to download gallery image ${index + 1}`);
|
|
}
|
|
}
|
|
catch (error) {
|
|
toast.error(`Error downloading gallery image ${index + 1}: ${error}`);
|
|
}
|
|
finally {
|
|
setDownloadingGalleryIndex(null);
|
|
}
|
|
};
|
|
const handleDownloadAllGallery = async () => {
|
|
if (!artistInfo.gallery || artistInfo.gallery.length === 0)
|
|
return;
|
|
setDownloadingAllGallery(true);
|
|
try {
|
|
const settings = getSettings();
|
|
let successCount = 0;
|
|
let existsCount = 0;
|
|
let failCount = 0;
|
|
for (let index = 0; index < artistInfo.gallery.length; index++) {
|
|
const imageUrl = artistInfo.gallery[index];
|
|
try {
|
|
const response = await downloadGalleryImage({
|
|
image_url: imageUrl,
|
|
artist_name: artistInfo.name,
|
|
image_index: index,
|
|
output_dir: settings.downloadPath,
|
|
});
|
|
if (response.success) {
|
|
if (response.already_exists) {
|
|
existsCount++;
|
|
}
|
|
else {
|
|
successCount++;
|
|
}
|
|
}
|
|
else {
|
|
failCount++;
|
|
}
|
|
}
|
|
catch (error) {
|
|
failCount++;
|
|
}
|
|
}
|
|
if (failCount === 0) {
|
|
if (existsCount > 0 && successCount > 0) {
|
|
toast.success(`${successCount} images downloaded, ${existsCount} already existed`);
|
|
}
|
|
else if (existsCount > 0) {
|
|
toast.info(`All ${existsCount} images already exist`);
|
|
}
|
|
else {
|
|
toast.success(`All ${successCount} gallery images downloaded successfully`);
|
|
}
|
|
}
|
|
else {
|
|
toast.error(`${failCount} images failed to download`);
|
|
}
|
|
}
|
|
catch (error) {
|
|
toast.error(`Error downloading gallery images: ${error}`);
|
|
}
|
|
finally {
|
|
setDownloadingAllGallery(false);
|
|
}
|
|
};
|
|
const hasGallery = artistInfo.gallery && artistInfo.gallery.length > 0;
|
|
return (<div className="space-y-6">
|
|
<Card className="overflow-hidden p-0 relative">
|
|
{artistInfo.header ? (<>
|
|
<div className="relative w-full h-64 bg-cover bg-center">
|
|
<div className="absolute inset-0 bg-cover bg-center" style={{ backgroundImage: `url(${artistInfo.header})` }}/>
|
|
<div className="absolute inset-0 bg-gradient-to-t from-black via-black/50 to-transparent"/>
|
|
{onBack && (<div className="absolute top-4 right-4 z-10">
|
|
<Button variant="ghost" size="icon" onClick={onBack} className="text-white hover:bg-white/20 hover:text-white">
|
|
<XCircle className="h-5 w-5"/>
|
|
</Button>
|
|
</div>)}
|
|
<div className="absolute bottom-4 right-4 z-10">
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={handleDownloadHeader} size="sm" variant="secondary" disabled={downloadingHeader} className="bg-white/10 hover:bg-white/20 text-white border-white/20">
|
|
{downloadingHeader ? (<Spinner className="h-4 w-4"/>) : (<ImageDown className="h-4 w-4"/>)}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download Header</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
<div className="relative px-6 pt-6 pb-20">
|
|
<div className="flex gap-6 items-start">
|
|
{artistInfo.images && (<div className="relative group">
|
|
<img src={artistInfo.images} alt={artistInfo.name} className="w-48 h-48 rounded-full shadow-lg object-cover border-4 border-white"/>
|
|
<div className="absolute inset-0 bg-black/0 group-hover:bg-black/50 transition-colors rounded-full flex items-center justify-center">
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={handleDownloadAvatar} size="sm" variant="secondary" disabled={downloadingAvatar} className="opacity-0 group-hover:opacity-100 transition-opacity bg-white/10 hover:bg-white/20 text-white border-white/20">
|
|
{downloadingAvatar ? (<Spinner className="h-4 w-4"/>) : (<ImageDown className="h-4 w-4"/>)}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download Avatar</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
</div>)}
|
|
<div className="flex-1 space-y-2">
|
|
<p className="text-sm font-medium text-white/80">Artist</p>
|
|
<div className="flex items-center gap-2">
|
|
<h2 className="text-4xl font-bold text-white">{artistInfo.name}</h2>
|
|
{artistInfo.verified && (<BadgeCheck className="h-6 w-6 text-white fill-blue-400 shrink-0"/>)}
|
|
</div>
|
|
{artistInfo.biography && (<p className="text-sm text-white/90 line-clamp-4">{artistInfo.biography}</p>)}
|
|
<div className="flex items-center gap-2 text-sm flex-wrap text-white/90">
|
|
{artistInfo.rank && (<>
|
|
<span>#{artistInfo.rank} rank</span>
|
|
<span>•</span>
|
|
</>)}
|
|
<span>{artistInfo.followers.toLocaleString()} {artistInfo.followers === 1 ? "follower" : "followers"}</span>
|
|
{artistInfo.listeners && (<>
|
|
<span>•</span>
|
|
<span>{artistInfo.listeners.toLocaleString()} {artistInfo.listeners === 1 ? "listener" : "listeners"}</span>
|
|
</>)}
|
|
</div>
|
|
<div className="flex items-center gap-2 text-sm flex-wrap text-white/90">
|
|
<span>{albumCountLabel}</span>
|
|
<span>•</span>
|
|
<span>{trackCountLabel}</span>
|
|
{artistInfo.genres.length > 0 && (<>
|
|
<span>•</span>
|
|
<span>{artistInfo.genres.join(", ")}</span>
|
|
</>)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>) : (<CardContent className="px-6 py-6">
|
|
{onBack && (<div className="absolute top-4 right-4 z-10">
|
|
<Button variant="ghost" size="icon" onClick={onBack}>
|
|
<XCircle className="h-5 w-5"/>
|
|
</Button>
|
|
</div>)}
|
|
<div className="flex gap-6 items-start">
|
|
{artistInfo.images && (<div className="relative group">
|
|
<img src={artistInfo.images} alt={artistInfo.name} className="w-48 h-48 rounded-full shadow-lg object-cover border-4 border-white"/>
|
|
<div className="absolute inset-0 bg-black/0 group-hover:bg-black/50 transition-colors rounded-full flex items-center justify-center">
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={handleDownloadAvatar} size="sm" variant="secondary" disabled={downloadingAvatar} className="opacity-0 group-hover:opacity-100 transition-opacity bg-white/10 hover:bg-white/20 text-white border-white/20">
|
|
{downloadingAvatar ? (<Spinner className="h-4 w-4"/>) : (<ImageDown className="h-4 w-4"/>)}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download Avatar</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
</div>)}
|
|
<div className="flex-1 space-y-2">
|
|
<p className="text-sm font-medium">Artist</p>
|
|
<div className="flex items-center gap-2">
|
|
<h2 className="text-4xl font-bold">{artistInfo.name}</h2>
|
|
{artistInfo.verified && (<BadgeCheck className="h-6 w-6 text-white fill-blue-500 shrink-0"/>)}
|
|
</div>
|
|
{artistInfo.biography && (<p className="text-sm text-muted-foreground line-clamp-4">{artistInfo.biography}</p>)}
|
|
<div className="flex items-center gap-2 text-sm flex-wrap">
|
|
{artistInfo.rank && (<>
|
|
<span>#{artistInfo.rank} rank</span>
|
|
<span>•</span>
|
|
</>)}
|
|
<span>{artistInfo.followers.toLocaleString()} {artistInfo.followers === 1 ? "follower" : "followers"}</span>
|
|
{artistInfo.listeners && (<>
|
|
<span>•</span>
|
|
<span>{artistInfo.listeners.toLocaleString()} {artistInfo.listeners === 1 ? "listener" : "listeners"}</span>
|
|
</>)}
|
|
</div>
|
|
<div className="flex items-center gap-2 text-sm flex-wrap">
|
|
<span>{albumCountLabel}</span>
|
|
<span>•</span>
|
|
<span>{trackCountLabel}</span>
|
|
{artistInfo.genres.length > 0 && (<>
|
|
<span>•</span>
|
|
<span>{artistInfo.genres.join(", ")}</span>
|
|
</>)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</CardContent>)}
|
|
</Card>
|
|
|
|
<div className="border-b">
|
|
<div className="flex gap-6">
|
|
<button onClick={() => setActiveTab("albums")} className={`pb-3 text-sm font-medium transition-colors border-b-2 -mb-px hover:text-foreground ${activeTab === "albums" ? "border-primary text-foreground" : "border-transparent text-muted-foreground"}`}>
|
|
Albums
|
|
</button>
|
|
<button onClick={() => setActiveTab("tracks")} className={`pb-3 text-sm font-medium transition-colors border-b-2 -mb-px hover:text-foreground ${activeTab === "tracks" ? "border-primary text-foreground" : "border-transparent text-muted-foreground"}`}>
|
|
All Tracks
|
|
</button>
|
|
{hasGallery && (<button onClick={() => setActiveTab("gallery")} className={`pb-3 text-sm font-medium transition-colors border-b-2 -mb-px hover:text-foreground ${activeTab === "gallery" ? "border-primary text-foreground" : "border-transparent text-muted-foreground"}`}>
|
|
Gallery
|
|
</button>)}
|
|
</div>
|
|
</div>
|
|
|
|
{activeTab === "gallery" && hasGallery && (<div className="space-y-4">
|
|
<div className="flex items-center justify-between">
|
|
<h3 className="text-2xl font-bold">Gallery ({artistInfo.gallery!.length.toLocaleString()})</h3>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={handleDownloadAllGallery} size="sm" variant="outline" disabled={downloadingAllGallery}>
|
|
{downloadingAllGallery ? <Spinner className="h-4 w-4"/> : <ImageDown className="h-4 w-4"/>}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download All Gallery</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5 gap-4">
|
|
{artistInfo.gallery!.map((imageUrl, index) => (<div key={index} className="relative group">
|
|
<div className="relative aspect-square rounded-md overflow-hidden shadow-md">
|
|
<img src={imageUrl} alt={`${artistInfo.name} gallery ${index + 1}`} className="w-full h-full object-cover"/>
|
|
<div className="absolute inset-0 bg-black/0 group-hover:bg-black/50 transition-colors flex items-center justify-center">
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={() => handleDownloadGalleryImage(imageUrl, index)} size="sm" variant="secondary" disabled={downloadingGalleryIndex === index} className="opacity-0 group-hover:opacity-100 transition-opacity bg-white/10 hover:bg-white/20 text-white border-white/20">
|
|
{downloadingGalleryIndex === index ? (<Spinner className="h-4 w-4"/>) : (<ImageDown className="h-4 w-4"/>)}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download Image {index + 1}</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</div>
|
|
</div>
|
|
</div>))}
|
|
</div>
|
|
</div>)}
|
|
|
|
{activeTab === "albums" && albumList.length > 0 && (<div className="space-y-4">
|
|
<div className="flex items-center justify-between flex-wrap gap-2">
|
|
<h3 className="text-2xl font-bold">Discography</h3>
|
|
<div className="flex gap-2">
|
|
<Button onClick={onDownloadAll} size="sm" disabled={isDownloading}>
|
|
{isDownloading && bulkDownloadType === "all" ? (<Spinner />) : (<Download className="h-4 w-4"/>)}
|
|
Download Discography
|
|
</Button>
|
|
{selectedTracks.length > 0 && (<Button onClick={onDownloadSelected} size="sm" variant="secondary" disabled={isDownloading}>
|
|
{isDownloading && bulkDownloadType === "selected" ? (<Spinner />) : (<Download className="h-4 w-4"/>)}
|
|
Download Selected ({selectedTracks.length})
|
|
</Button>)}
|
|
</div>
|
|
</div>
|
|
{albumFilters.length > 1 && (<div className="flex flex-wrap gap-2">
|
|
{albumFilters.map((filter) => (<Button key={filter} size="sm" variant={activeAlbumFilter === filter ? "default" : "outline"} onClick={() => setActiveAlbumFilter(filter)}>
|
|
{formatAlbumFilterLabel(filter)}
|
|
</Button>))}
|
|
</div>)}
|
|
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-4">
|
|
{filteredAlbums.map((album) => {
|
|
const albumTracks = trackList.filter(t => t.album_name === album.name);
|
|
const tracksWithId = albumTracks.filter(t => t.spotify_id);
|
|
const isSelected = tracksWithId.length > 0 && tracksWithId.every(t => selectedTracks.includes(t.spotify_id!));
|
|
const hasTracks = tracksWithId.length > 0;
|
|
return (<div key={album.id} className="group cursor-pointer relative" onClick={() => onAlbumClick({
|
|
id: album.id,
|
|
name: album.name,
|
|
external_urls: album.external_urls,
|
|
})}>
|
|
<div className="relative mb-2">
|
|
|
|
{hasTracks && (<div className={`absolute top-2 left-2 z-20 ${isSelected ? 'opacity-100' : 'opacity-0 group-hover:opacity-100'} transition-opacity`} onClick={(e) => e.stopPropagation()}>
|
|
<Checkbox checked={isSelected} onCheckedChange={() => onToggleSelectAll(albumTracks)} className="bg-black/50 border-white/70 data-[state=checked]:bg-primary data-[state=checked]:border-primary"/>
|
|
</div>)}
|
|
{album.images && (<img src={album.images} alt={album.name} className="w-full aspect-square object-cover rounded-md shadow-md transition-shadow group-hover:shadow-xl"/>)}
|
|
<div className="absolute bottom-2 right-2">
|
|
<span className="text-[10px] uppercase font-bold px-1.5 py-0.5 rounded bg-black/60 text-white backdrop-blur-[2px]">
|
|
{album.album_type}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<h4 className="font-semibold truncate text-sm">{album.name}</h4>
|
|
<div className="flex items-center gap-2 text-xs text-muted-foreground mt-1">
|
|
<span>{album.release_date?.split("-")[0]}</span>
|
|
{album.total_tracks && (<>
|
|
<span>•</span>
|
|
<span>{album.total_tracks} {album.total_tracks === 1 ? "track" : "tracks"}</span>
|
|
</>)}
|
|
</div>
|
|
</div>);
|
|
})}
|
|
</div>
|
|
{filteredAlbums.length === 0 && (<div className="rounded-lg border border-dashed border-border p-6 text-sm text-muted-foreground">
|
|
No releases found for the selected discography filter.
|
|
</div>)}
|
|
</div>)}
|
|
|
|
{activeTab === "tracks" && trackList.length > 0 && (<div className="space-y-4">
|
|
<div className="flex items-center justify-between flex-wrap gap-2">
|
|
<h3 className="text-2xl font-bold">All Tracks</h3>
|
|
<div className="flex gap-2 flex-wrap">
|
|
<Dialog>
|
|
<DialogTrigger asChild>
|
|
<Button variant="outline" size="sm">
|
|
<Filter className="h-4 w-4"/>
|
|
Filter Albums
|
|
</Button>
|
|
</DialogTrigger>
|
|
<DialogContent className="sm:max-w-[500px] h-[80vh] flex flex-col">
|
|
<DialogHeader>
|
|
<DialogTitle>Select Albums</DialogTitle>
|
|
</DialogHeader>
|
|
<ScrollArea className="flex-1 pr-4">
|
|
<div className="space-y-4">
|
|
{filteredAlbumGroups.map(([albumName, data]) => {
|
|
const tracksWithId = data.tracks.filter(t => t.spotify_id);
|
|
const isSelected = tracksWithId.length > 0 && tracksWithId.every(t => selectedTracks.includes(t.spotify_id!));
|
|
return (<div key={albumName} className="flex items-start space-x-3 p-2 hover:bg-muted/50 rounded-md transition-colors">
|
|
<Checkbox id={`album-select-${albumName}`} checked={isSelected} onCheckedChange={() => onToggleSelectAll(data.tracks)} className="mt-1"/>
|
|
<div className="grid gap-1.5 leading-none flex-1">
|
|
<label htmlFor={`album-select-${albumName}`} className="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70 cursor-pointer">
|
|
{albumName}
|
|
</label>
|
|
<div className="flex items-center gap-2 text-xs text-muted-foreground">
|
|
<span className="capitalize bg-muted px-1.5 py-0.5 rounded text-[10px] font-semibold border">
|
|
{data.type}
|
|
</span>
|
|
<span>•</span>
|
|
<span>{data.count} tracks</span>
|
|
<span>•</span>
|
|
<span>{data.tracks[0]?.release_date?.split('-')[0] || 'Unknown Year'}</span>
|
|
</div>
|
|
</div>
|
|
</div>);
|
|
})}
|
|
</div>
|
|
</ScrollArea>
|
|
</DialogContent>
|
|
</Dialog>
|
|
<Button onClick={onDownloadAll} size="sm" disabled={isDownloading}>
|
|
{isDownloading && bulkDownloadType === "all" ? (<Spinner />) : (<Download className="h-4 w-4"/>)}
|
|
Download All
|
|
</Button>
|
|
{selectedTracks.length > 0 && (<Button onClick={onDownloadSelected} size="sm" variant="secondary" disabled={isDownloading}>
|
|
{isDownloading && bulkDownloadType === "selected" ? (<Spinner />) : (<Download className="h-4 w-4"/>)}
|
|
Download Selected ({selectedTracks.length.toLocaleString()})
|
|
</Button>)}
|
|
{onDownloadAllLyrics && (<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={onDownloadAllLyrics} size="sm" variant="outline" disabled={isBulkDownloadingLyrics}>
|
|
{isBulkDownloadingLyrics ? <Spinner /> : <FileText className="h-4 w-4"/>}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download All Lyrics</p>
|
|
</TooltipContent>
|
|
</Tooltip>)}
|
|
{onDownloadAllCovers && (<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={onDownloadAllCovers} size="sm" variant="outline" disabled={isBulkDownloadingCovers}>
|
|
{isBulkDownloadingCovers ? <Spinner /> : <ImageDown className="h-4 w-4"/>}
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Download All Separate Covers</p>
|
|
</TooltipContent>
|
|
</Tooltip>)}
|
|
{downloadedTracks.size > 0 && (<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button onClick={onOpenFolder} size="icon" variant="outline">
|
|
<FolderOpen className="h-4 w-4"/>
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>Open Folder</p>
|
|
</TooltipContent>
|
|
</Tooltip>)}
|
|
</div>
|
|
</div>
|
|
{isDownloading && (<DownloadProgress progress={downloadProgress} currentTrack={currentDownloadInfo} onStop={onStopDownload}/>)}
|
|
<SearchAndSort searchQuery={searchQuery} sortBy={sortBy} onSearchChange={onSearchChange} onSortChange={onSortChange}/>
|
|
<TrackList tracks={trackList} searchQuery={searchQuery} sortBy={sortBy} selectedTracks={selectedTracks} downloadedTracks={downloadedTracks} failedTracks={failedTracks} skippedTracks={skippedTracks} downloadingTrack={downloadingTrack} isDownloading={isDownloading} currentPage={currentPage} itemsPerPage={itemsPerPage} showCheckboxes={true} hideAlbumColumn={false} folderName={artistInfo.name} isArtistDiscography={true} downloadedLyrics={downloadedLyrics} failedLyrics={failedLyrics} skippedLyrics={skippedLyrics} downloadingLyricsTrack={downloadingLyricsTrack} checkingAvailabilityTrack={checkingAvailabilityTrack} availabilityMap={availabilityMap} onToggleTrack={onToggleTrack} onToggleSelectAll={onToggleSelectAll} onDownloadTrack={onDownloadTrack} onDownloadLyrics={onDownloadLyrics} onDownloadCover={onDownloadCover} downloadedCovers={downloadedCovers} failedCovers={failedCovers} skippedCovers={skippedCovers} downloadingCoverTrack={downloadingCoverTrack} onCheckAvailability={onCheckAvailability} onPageChange={onPageChange} onAlbumClick={onAlbumClick} onArtistClick={onArtistClick} onTrackClick={onTrackClick}/>
|
|
</div>)}
|
|
</div>);
|
|
}
|