- Implemented Tooltip component using Radix UI for better accessibility and customization. - Created TooltipProvider, TooltipTrigger, and TooltipContent for modular usage. - Added useIsMobile hook to detect mobile devices based on screen width. - Updated themes with new color variables for better design consistency across the application.
266 lines
10 KiB
TypeScript
266 lines
10 KiB
TypeScript
'use client';
|
|
|
|
import React, { useState, useEffect } from 'react';
|
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
|
import { Button } from "@/components/ui/button";
|
|
import { useNavidrome } from "@/app/components/NavidromeContext";
|
|
import { AlbumArtwork } from "@/app/components/album-artwork";
|
|
import { ArtistIcon } from "@/app/components/artist-icon";
|
|
import { Album, Artist, Song } from "@/lib/navidrome";
|
|
import { Heart, Music, Disc, Mic, Play } from "lucide-react";
|
|
import { useAudioPlayer, Track } from "@/app/components/AudioPlayerContext";
|
|
import Image from "next/image";
|
|
|
|
const FavoritesPage = () => {
|
|
const { api, isConnected } = useNavidrome();
|
|
const { playTrack, addToQueue } = useAudioPlayer();
|
|
const [favoriteAlbums, setFavoriteAlbums] = useState<Album[]>([]);
|
|
const [favoriteSongs, setFavoriteSongs] = useState<Song[]>([]);
|
|
const [favoriteArtists, setFavoriteArtists] = useState<Artist[]>([]);
|
|
const [loading, setLoading] = useState(true);
|
|
|
|
useEffect(() => {
|
|
const loadFavorites = async () => {
|
|
if (!api || !isConnected) return;
|
|
|
|
setLoading(true);
|
|
try {
|
|
const [albums, songs, artists] = await Promise.all([
|
|
api.getAlbums('starred', 100),
|
|
api.getStarred2(),
|
|
api.getArtists()
|
|
]);
|
|
|
|
setFavoriteAlbums(albums);
|
|
|
|
// Filter starred songs and artists from the starred2 response
|
|
if (songs.starred2) {
|
|
setFavoriteSongs(songs.starred2.song || []);
|
|
setFavoriteArtists((songs.starred2.artist || []).filter((artist: Artist) => artist.starred));
|
|
}
|
|
} catch (error) {
|
|
console.error('Failed to load favorites:', error);
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
loadFavorites();
|
|
}, [api, isConnected]);
|
|
|
|
const handlePlaySong = (song: Song) => {
|
|
playTrack({
|
|
id: song.id,
|
|
name: song.title,
|
|
artist: song.artist,
|
|
album: song.album,
|
|
albumId: song.albumId,
|
|
artistId: song.artistId,
|
|
url: api?.getStreamUrl(song.id) || '',
|
|
duration: song.duration,
|
|
coverArt: song.coverArt ? api?.getCoverArtUrl(song.coverArt) : undefined,
|
|
starred: !!song.starred
|
|
});
|
|
};
|
|
|
|
const handlePlayAlbum = async (album: Album) => {
|
|
if (!api) return;
|
|
|
|
try {
|
|
const songs = await api.getAlbumSongs(album.id);
|
|
if (songs.length > 0) {
|
|
const tracks = songs.map((song: Song) => ({
|
|
id: song.id,
|
|
name: song.title,
|
|
artist: song.artist,
|
|
album: song.album,
|
|
albumId: song.albumId,
|
|
artistId: song.artistId,
|
|
url: api.getStreamUrl(song.id),
|
|
duration: song.duration,
|
|
coverArt: song.coverArt ? api.getCoverArtUrl(song.coverArt) : undefined,
|
|
starred: !!song.starred
|
|
}));
|
|
|
|
playTrack(tracks[0]);
|
|
tracks.slice(1).forEach((track: Track) => addToQueue(track));
|
|
}
|
|
} catch (error) {
|
|
console.error('Failed to play album:', error);
|
|
}
|
|
};
|
|
|
|
const toggleFavorite = async (id: string, type: 'song' | 'album' | 'artist', isStarred: boolean) => {
|
|
if (!api) return;
|
|
|
|
try {
|
|
if (isStarred) {
|
|
await api.unstar(id, type);
|
|
} else {
|
|
await api.star(id, type);
|
|
}
|
|
|
|
// Refresh favorites
|
|
if (type === 'album') {
|
|
const albums = await api.getAlbums('starred', 100);
|
|
setFavoriteAlbums(albums);
|
|
} else if (type === 'song') {
|
|
const songs = await api.getStarred2();
|
|
setFavoriteSongs(songs.starred2?.song || []);
|
|
} else if (type === 'artist') {
|
|
const songs = await api.getStarred2();
|
|
setFavoriteArtists((songs.starred2?.artist || []).filter((artist: Artist) => artist.starred));
|
|
}
|
|
} catch (error) {
|
|
console.error('Failed to toggle favorite:', error);
|
|
}
|
|
};
|
|
|
|
if (!isConnected) {
|
|
return (
|
|
<div className="container mx-auto p-6">
|
|
<div className="text-center">
|
|
<p className="text-muted-foreground">Please connect to your Navidrome server to view favorites.</p>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className="container mx-auto p-6 pb-24">
|
|
<div className="space-y-6">
|
|
<div className="flex items-center gap-3">
|
|
<div>
|
|
<h1 className="text-3xl font-semibold tracking-tight">Favorites</h1>
|
|
<p className="text-muted-foreground">Your starred albums, songs, and artists</p>
|
|
</div>
|
|
</div>
|
|
|
|
<Tabs defaultValue="albums" className="space-y-6">
|
|
<TabsList className="grid w-full grid-cols-3">
|
|
<TabsTrigger value="albums" className="flex items-center gap-2">
|
|
<Disc className="w-4 h-4" />
|
|
Albums ({favoriteAlbums.length})
|
|
</TabsTrigger>
|
|
<TabsTrigger value="songs" className="flex items-center gap-2">
|
|
<Music className="w-4 h-4" />
|
|
Songs ({favoriteSongs.length})
|
|
</TabsTrigger>
|
|
<TabsTrigger value="artists" className="flex items-center gap-2">
|
|
<Mic className="w-4 h-4" />
|
|
Artists ({favoriteArtists.length})
|
|
</TabsTrigger>
|
|
</TabsList>
|
|
|
|
<TabsContent value="albums" className="space-y-4">
|
|
{loading ? (
|
|
<div className="text-center py-12">
|
|
<p className="text-muted-foreground">Loading favorite albums...</p>
|
|
</div>
|
|
) : favoriteAlbums.length === 0 ? (
|
|
<div className="text-center py-12">
|
|
<Heart className="w-12 h-12 mx-auto text-muted-foreground mb-4" />
|
|
<p className="text-muted-foreground">No favorite albums yet</p>
|
|
<p className="text-sm text-muted-foreground mt-2">Star albums to see them here</p>
|
|
</div>
|
|
) : (
|
|
<div className="grid gap-4 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-6">
|
|
{favoriteAlbums.map((album) => (
|
|
<AlbumArtwork
|
|
key={album.id}
|
|
album={album}
|
|
className="w-full"
|
|
aspectRatio="square"
|
|
width={200}
|
|
height={200}
|
|
/>
|
|
))}
|
|
</div>
|
|
)}
|
|
</TabsContent>
|
|
|
|
<TabsContent value="songs" className="space-y-4">
|
|
{loading ? (
|
|
<div className="text-center py-12">
|
|
<p className="text-muted-foreground">Loading favorite songs...</p>
|
|
</div>
|
|
) : favoriteSongs.length === 0 ? (
|
|
<div className="text-center py-12">
|
|
<Heart className="w-12 h-12 mx-auto text-muted-foreground mb-4" />
|
|
<p className="text-muted-foreground">No favorite songs yet</p>
|
|
<p className="text-sm text-muted-foreground mt-2">Star songs to see them here</p>
|
|
</div>
|
|
) : (
|
|
<div className="space-y-2">
|
|
{favoriteSongs.map((song, index) => (
|
|
<div key={song.id} className="flex items-center gap-4 p-3 rounded-lg hover:bg-muted/50 group">
|
|
<div className="w-8 text-sm text-muted-foreground text-center">
|
|
{index + 1}
|
|
</div>
|
|
<div className="w-12 h-12 relative shrink-0">
|
|
{song.coverArt && api ? (
|
|
<Image
|
|
src={api.getCoverArtUrl(song.coverArt)}
|
|
alt={song.album}
|
|
fill
|
|
className="rounded object-cover"
|
|
/>
|
|
) : (
|
|
<div className="w-full h-full bg-muted rounded flex items-center justify-center">
|
|
<Music className="w-6 h-6 text-muted-foreground" />
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div className="flex-1 min-w-0">
|
|
<p className="font-medium truncate">{song.title}</p>
|
|
<p className="text-sm text-muted-foreground truncate">{song.artist}</p>
|
|
</div>
|
|
<div className="text-sm text-muted-foreground">{song.album}</div>
|
|
<div className="text-sm text-muted-foreground">
|
|
{Math.floor(song.duration / 60)}:{(song.duration % 60).toString().padStart(2, '0')}
|
|
</div>
|
|
<div className="flex items-center gap-2 opacity-0 group-hover:opacity-100">
|
|
<Button size="sm" variant="ghost" onClick={() => handlePlaySong(song)}>
|
|
Play
|
|
</Button>
|
|
<Button
|
|
size="sm"
|
|
variant="ghost"
|
|
onClick={() => toggleFavorite(song.id, 'song', !!song.starred)}
|
|
>
|
|
<Heart className={`w-4 h-4 ${song.starred ? 'fill-red-500 text-red-500' : ''}`} />
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</TabsContent>
|
|
|
|
<TabsContent value="artists" className="space-y-4">
|
|
{loading ? (
|
|
<div className="text-center py-12">
|
|
<p className="text-muted-foreground">Loading favorite artists...</p>
|
|
</div>
|
|
) : favoriteArtists.length === 0 ? (
|
|
<div className="text-center py-12">
|
|
<Heart className="w-12 h-12 mx-auto text-muted-foreground mb-4" />
|
|
<p className="text-muted-foreground">No favorite artists yet</p>
|
|
<p className="text-sm text-muted-foreground mt-2">Star artists to see them here</p>
|
|
</div>
|
|
) : (
|
|
<div className="grid gap-4 md:grid-cols-3 lg:grid-cols-5 xl:grid-cols-7">
|
|
{favoriteArtists.map((artist) => (
|
|
<ArtistIcon key={artist.id} artist={artist} responsive />
|
|
))}
|
|
</div>
|
|
)}
|
|
</TabsContent>
|
|
</Tabs>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default FavoritesPage;
|