feat: update NEXT_PUBLIC_COMMIT_SHA and add WhatsNewPopup component with changelog functionality
This commit is contained in:
@@ -1 +1 @@
|
|||||||
NEXT_PUBLIC_COMMIT_SHA=591faca
|
NEXT_PUBLIC_COMMIT_SHA=f721213
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { NavidromeProvider, useNavidrome } from "../components/NavidromeContext"
|
|||||||
import { NavidromeConfigProvider } from "../components/NavidromeConfigContext";
|
import { NavidromeConfigProvider } from "../components/NavidromeConfigContext";
|
||||||
import { ThemeProvider } from "../components/ThemeProvider";
|
import { ThemeProvider } from "../components/ThemeProvider";
|
||||||
import { PostHogProvider } from "../components/PostHogProvider";
|
import { PostHogProvider } from "../components/PostHogProvider";
|
||||||
|
import { WhatsNewPopup } from "../components/WhatsNewPopup";
|
||||||
import Ihateserverside from "./ihateserverside";
|
import Ihateserverside from "./ihateserverside";
|
||||||
import DynamicViewportTheme from "./DynamicViewportTheme";
|
import DynamicViewportTheme from "./DynamicViewportTheme";
|
||||||
import { LoginForm } from "./start-screen";
|
import { LoginForm } from "./start-screen";
|
||||||
@@ -43,6 +44,7 @@ export default function RootLayoutClient({ children }: { children: React.ReactNo
|
|||||||
<Ihateserverside>
|
<Ihateserverside>
|
||||||
{children}
|
{children}
|
||||||
</Ihateserverside>
|
</Ihateserverside>
|
||||||
|
<WhatsNewPopup />
|
||||||
</AudioPlayerProvider>
|
</AudioPlayerProvider>
|
||||||
</NavidromeErrorBoundary>
|
</NavidromeErrorBoundary>
|
||||||
</NavidromeProvider>
|
</NavidromeProvider>
|
||||||
|
|||||||
139
app/components/WhatsNewPopup.tsx
Normal file
139
app/components/WhatsNewPopup.tsx
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState, useEffect } from 'react';
|
||||||
|
import { Dialog, DialogContent, DialogHeader, DialogTitle } from '@/components/ui/dialog';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { Badge } from '@/components/ui/badge';
|
||||||
|
import { ScrollArea } from '@/components/ui/scroll-area';
|
||||||
|
import { X } from 'lucide-react';
|
||||||
|
|
||||||
|
// Current app version from package.json
|
||||||
|
const APP_VERSION = '1.0.0';
|
||||||
|
|
||||||
|
// Changelog data - add new versions at the top
|
||||||
|
const CHANGELOG = [
|
||||||
|
{
|
||||||
|
version: '1.0.0',
|
||||||
|
date: '2024-01-10',
|
||||||
|
title: 'Initial Release',
|
||||||
|
changes: [
|
||||||
|
'Complete redesign with modern UI',
|
||||||
|
'Added Favorites functionality for albums, songs, and artists',
|
||||||
|
'Integrated standalone Last.fm scrobbling support',
|
||||||
|
'Added collapsible sidebar with icon-only mode',
|
||||||
|
'Improved search and browsing experience',
|
||||||
|
'Added history tracking for played songs',
|
||||||
|
'Enhanced audio player with better controls',
|
||||||
|
'Added settings page for customization options'
|
||||||
|
],
|
||||||
|
breaking: [],
|
||||||
|
fixes: []
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
export function WhatsNewPopup() {
|
||||||
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// Check if we've shown the popup for this version
|
||||||
|
const lastShownVersion = localStorage.getItem('whats-new-last-shown');
|
||||||
|
|
||||||
|
if (lastShownVersion !== APP_VERSION) {
|
||||||
|
setIsOpen(true);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleClose = () => {
|
||||||
|
// Mark this version as shown
|
||||||
|
localStorage.setItem('whats-new-last-shown', APP_VERSION);
|
||||||
|
setIsOpen(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const currentVersionChangelog = CHANGELOG.find(entry => entry.version === APP_VERSION);
|
||||||
|
|
||||||
|
if (!currentVersionChangelog) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={isOpen} onOpenChange={handleClose}>
|
||||||
|
<DialogContent className="max-w-2xl max-h-[80vh]">
|
||||||
|
<DialogHeader className="flex flex-row items-center justify-between space-y-0 pb-4">
|
||||||
|
<div>
|
||||||
|
<DialogTitle className="text-2xl font-bold flex items-center gap-2">
|
||||||
|
What's New in mice
|
||||||
|
<Badge variant="outline">{currentVersionChangelog.version}</Badge>
|
||||||
|
</DialogTitle>
|
||||||
|
<p className="text-sm text-muted-foreground mt-1">
|
||||||
|
Released on {currentVersionChangelog.date}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
<ScrollArea className="max-h-[60vh] pr-4">
|
||||||
|
<div className="space-y-6">
|
||||||
|
{currentVersionChangelog.title && (
|
||||||
|
<div>
|
||||||
|
<h3 className="text-lg font-semibold mb-2">{currentVersionChangelog.title}</h3>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{currentVersionChangelog.changes.length > 0 && (
|
||||||
|
<div>
|
||||||
|
<h4 className="text-md font-medium mb-3 flex items-center gap-2">
|
||||||
|
✨ New Features & Improvements
|
||||||
|
</h4>
|
||||||
|
<ul className="space-y-2">
|
||||||
|
{currentVersionChangelog.changes.map((change, index) => (
|
||||||
|
<li key={index} className="flex items-start gap-2">
|
||||||
|
<span className="text-green-500 mt-1">•</span>
|
||||||
|
<span className="text-sm">{change}</span>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{currentVersionChangelog.fixes.length > 0 && (
|
||||||
|
<div>
|
||||||
|
<h4 className="text-md font-medium mb-3 flex items-center gap-2">
|
||||||
|
🐛 Bug Fixes
|
||||||
|
</h4>
|
||||||
|
<ul className="space-y-2">
|
||||||
|
{currentVersionChangelog.fixes.map((fix, index) => (
|
||||||
|
<li key={index} className="flex items-start gap-2">
|
||||||
|
<span className="text-blue-500 mt-1">•</span>
|
||||||
|
<span className="text-sm">{fix}</span>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{currentVersionChangelog.breaking.length > 0 && (
|
||||||
|
<div>
|
||||||
|
<h4 className="text-md font-medium mb-3 flex items-center gap-2">
|
||||||
|
⚠️ Breaking Changes
|
||||||
|
</h4>
|
||||||
|
<ul className="space-y-2">
|
||||||
|
{currentVersionChangelog.breaking.map((breaking, index) => (
|
||||||
|
<li key={index} className="flex items-start gap-2">
|
||||||
|
<span className="text-red-500 mt-1">•</span>
|
||||||
|
<span className="text-sm">{breaking}</span>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
|
||||||
|
<div className="flex justify-end pt-4">
|
||||||
|
<Button onClick={handleClose}>
|
||||||
|
Got it!
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -16,18 +16,32 @@ interface SidebarProps extends React.HTMLAttributes<HTMLDivElement> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function Sidebar({ className, playlists, collapsed = false, onToggle }: SidebarProps) {
|
export function Sidebar({ className, playlists, collapsed = false, onToggle }: SidebarProps) {
|
||||||
const isRoot = usePathname() === "/";
|
const pathname = usePathname();
|
||||||
const isBrowse = usePathname() === "/browse";
|
|
||||||
const isSearch = usePathname() === "/search";
|
// Define all routes and their active states
|
||||||
const isAlbums = usePathname() === "/library/albums";
|
const routes = {
|
||||||
const isArtists = usePathname() === "/library/artists";
|
isRoot: pathname === "/",
|
||||||
const isQueue = usePathname() === "/queue";
|
isBrowse: pathname === "/browse",
|
||||||
const isRadio = usePathname() === "/radio";
|
isSearch: pathname === "/search",
|
||||||
const isHistory = usePathname() === "/history";
|
isQueue: pathname === "/queue",
|
||||||
const isSongs = usePathname() === "/library/songs";
|
isRadio: pathname === "/radio",
|
||||||
const isPlaylists = usePathname() === "/library/playlists";
|
isPlaylists: pathname === "/library/playlists",
|
||||||
const isFavorites = usePathname() === "/favorites";
|
isSongs: pathname === "/library/songs",
|
||||||
const isNew = usePathname() === "/new";
|
isArtists: pathname === "/library/artists",
|
||||||
|
isAlbums: pathname === "/library/albums",
|
||||||
|
isHistory: pathname === "/history",
|
||||||
|
isFavorites: pathname === "/favorites",
|
||||||
|
isSettings: pathname === "/settings",
|
||||||
|
// Handle dynamic routes
|
||||||
|
isAlbumPage: pathname.startsWith("/album/"),
|
||||||
|
isArtistPage: pathname.startsWith("/artist/"),
|
||||||
|
isPlaylistPage: pathname.startsWith("/playlist/"),
|
||||||
|
isNewPage: pathname === "/new",
|
||||||
|
};
|
||||||
|
|
||||||
|
// Helper function to determine if any sidebar route is active
|
||||||
|
// This prevents highlights on pages not defined in sidebar
|
||||||
|
const isAnySidebarRouteActive = Object.values(routes).some(Boolean);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cn("pb-6 relative", className)}>
|
<div className={cn("pb-6 relative", className)}>
|
||||||
@@ -49,7 +63,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<Link href="/">
|
<Link href="/">
|
||||||
<Button
|
<Button
|
||||||
variant={isRoot ? "secondary" : "ghost"}
|
variant={routes.isRoot ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Listen Now" : undefined}
|
title={collapsed ? "Listen Now" : undefined}
|
||||||
>
|
>
|
||||||
@@ -71,7 +85,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/browse">
|
<Link href="/browse">
|
||||||
<Button
|
<Button
|
||||||
variant={isBrowse ? "secondary" : "ghost"}
|
variant={routes.isBrowse ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Browse" : undefined}
|
title={collapsed ? "Browse" : undefined}
|
||||||
>
|
>
|
||||||
@@ -95,7 +109,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/search">
|
<Link href="/search">
|
||||||
<Button
|
<Button
|
||||||
variant={isSearch ? "secondary" : "ghost"}
|
variant={routes.isSearch ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Search" : undefined}
|
title={collapsed ? "Search" : undefined}
|
||||||
>
|
>
|
||||||
@@ -117,7 +131,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/queue">
|
<Link href="/queue">
|
||||||
<Button
|
<Button
|
||||||
variant={isQueue ? "secondary" : "ghost"}
|
variant={routes.isQueue ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Queue" : undefined}
|
title={collapsed ? "Queue" : undefined}
|
||||||
>
|
>
|
||||||
@@ -138,7 +152,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/radio">
|
<Link href="/radio">
|
||||||
<Button
|
<Button
|
||||||
variant={isRadio ? "secondary" : "ghost"}
|
variant={routes.isRadio ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Radio" : undefined}
|
title={collapsed ? "Radio" : undefined}
|
||||||
>
|
>
|
||||||
@@ -171,7 +185,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<Link href="/library/playlists">
|
<Link href="/library/playlists">
|
||||||
<Button
|
<Button
|
||||||
variant={isPlaylists ? "secondary" : "ghost"}
|
variant={routes.isPlaylists ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-1", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-1", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Playlists" : undefined}
|
title={collapsed ? "Playlists" : undefined}
|
||||||
>
|
>
|
||||||
@@ -196,7 +210,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/library/songs">
|
<Link href="/library/songs">
|
||||||
<Button
|
<Button
|
||||||
variant={isSongs ? "secondary" : "ghost"}
|
variant={routes.isSongs ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Songs" : undefined}
|
title={collapsed ? "Songs" : undefined}
|
||||||
>
|
>
|
||||||
@@ -218,7 +232,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/library/artists">
|
<Link href="/library/artists">
|
||||||
<Button
|
<Button
|
||||||
variant={isArtists ? "secondary" : "ghost"}
|
variant={routes.isArtists ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Artists" : undefined}
|
title={collapsed ? "Artists" : undefined}
|
||||||
>
|
>
|
||||||
@@ -240,7 +254,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/library/albums">
|
<Link href="/library/albums">
|
||||||
<Button
|
<Button
|
||||||
variant={isAlbums ? "secondary" : "ghost"}
|
variant={routes.isAlbums ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Albums" : undefined}
|
title={collapsed ? "Albums" : undefined}
|
||||||
>
|
>
|
||||||
@@ -264,7 +278,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/history">
|
<Link href="/history">
|
||||||
<Button
|
<Button
|
||||||
variant={isHistory ? "secondary" : "ghost"}
|
variant={routes.isHistory ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "History" : undefined}
|
title={collapsed ? "History" : undefined}
|
||||||
>
|
>
|
||||||
@@ -286,7 +300,7 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</Link>
|
</Link>
|
||||||
<Link href="/favorites">
|
<Link href="/favorites">
|
||||||
<Button
|
<Button
|
||||||
variant={isFavorites ? "secondary" : "ghost"}
|
variant={routes.isFavorites ? "secondary" : "ghost"}
|
||||||
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
title={collapsed ? "Favorites" : undefined}
|
title={collapsed ? "Favorites" : undefined}
|
||||||
>
|
>
|
||||||
@@ -308,6 +322,32 @@ export function Sidebar({ className, playlists, collapsed = false, onToggle }: S
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div className="px-3 py-2 mt-4">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Link href="/settings">
|
||||||
|
<Button
|
||||||
|
variant={routes.isSettings ? "secondary" : "ghost"}
|
||||||
|
className={cn("w-full justify-start mb-2", collapsed && "justify-center px-2")}
|
||||||
|
title={collapsed ? "Settings" : undefined}
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className={cn("h-4 w-4", !collapsed && "mr-2")}
|
||||||
|
>
|
||||||
|
<path d="M12.22 2h-.44a2 2 0 0 0-2 2v.18a2 2 0 0 1-1 1.73l-.43.25a2 2 0 0 1-2 0l-.15-.08a2 2 0 0 0-2.73.73l-.22.38a2 2 0 0 0 .73 2.73l.15.1a2 2 0 0 1 1 1.72v.51a2 2 0 0 1-1 1.74l-.15.09a2 2 0 0 0-.73 2.73l.22.38a2 2 0 0 0 2.73.73l.15-.08a2 2 0 0 1 2 0l.43.25a2 2 0 0 1 1 1.73V20a2 2 0 0 0 2 2h.44a2 2 0 0 0 2-2v-.18a2 2 0 0 1 1-1.73l.43-.25a2 2 0 0 1 2 0l.15.08a2 2 0 0 0 2.73-.73l.22-.39a2 2 0 0 0-.73-2.73l-.15-.08a2 2 0 0 1-1-1.74v-.5a2 2 0 0 1 1-1.74l.15-.09a2 2 0 0 0 .73-2.73l-.22-.38a2 2 0 0 0-2.73-.73l-.15.08a2 2 0 0 1-2 0l-.43-.25a2 2 0 0 1-1-1.73V4a2 2 0 0 0-2-2z" />
|
||||||
|
<circle cx="12" cy="12" r="3" />
|
||||||
|
</svg>
|
||||||
|
{!collapsed && "Settings"}
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
Reference in New Issue
Block a user