s
9
.env.example
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# Navidrome Server Configuration
|
||||||
|
NEXT_PUBLIC_NAVIDROME_URL=http://localhost:4533
|
||||||
|
NEXT_PUBLIC_NAVIDROME_USERNAME=your_username
|
||||||
|
NEXT_PUBLIC_NAVIDROME_PASSWORD=your_password
|
||||||
|
|
||||||
|
# Example for external server:
|
||||||
|
# NEXT_PUBLIC_NAVIDROME_URL=https://your-navidrome-server.com
|
||||||
|
# NEXT_PUBLIC_NAVIDROME_USERNAME=your_username
|
||||||
|
# NEXT_PUBLIC_NAVIDROME_PASSWORD=your_password
|
||||||
6
.eslintrc.json
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"extends": ["next/core-web-vitals", "next/typescript"],
|
||||||
|
"rules": {
|
||||||
|
"@typescript-eslint/no-unused-vars": "off"
|
||||||
|
}
|
||||||
|
}
|
||||||
5
.firebaserc
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"projects": {
|
||||||
|
"default": "offbrandspotifydb"
|
||||||
|
}
|
||||||
|
}
|
||||||
34
.github/workflows/jest.yml
vendored
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
name: Run Jest Tests
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
paths:
|
||||||
|
- 'apps/data/albums.ts'
|
||||||
|
- 'apps/data/artists.ts'
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
paths:
|
||||||
|
- 'apps/data/albums.ts'
|
||||||
|
- 'apps/data/artists.ts'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Set up Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: '22'
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm install
|
||||||
|
|
||||||
|
- name: Run Jest tests
|
||||||
|
run: npm test
|
||||||
23
.github/workflows/lint.yml
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
name: Lint
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- '**'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
lint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Set up Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm install
|
||||||
|
|
||||||
|
- name: Run Next.js lint
|
||||||
|
run: npm run lint
|
||||||
75
.gitignore
vendored
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
firebase-debug.log*
|
||||||
|
firebase-debug.*.log*
|
||||||
|
|
||||||
|
# Firebase cache
|
||||||
|
.firebase/
|
||||||
|
|
||||||
|
# Firebase config
|
||||||
|
|
||||||
|
# Uncomment this if you'd like others to create their own Firebase project.
|
||||||
|
# For a team working on the same Firebase project(s), it is recommended to leave
|
||||||
|
# it commented so all members can deploy to the same project(s) in .firebaserc.
|
||||||
|
# .firebaserc
|
||||||
|
|
||||||
|
# Runtime data
|
||||||
|
pids
|
||||||
|
*.pid
|
||||||
|
*.seed
|
||||||
|
*.pid.lock
|
||||||
|
|
||||||
|
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||||
|
lib-cov
|
||||||
|
|
||||||
|
# Coverage directory used by tools like istanbul
|
||||||
|
coverage
|
||||||
|
|
||||||
|
# nyc test coverage
|
||||||
|
.nyc_output
|
||||||
|
|
||||||
|
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
|
||||||
|
.grunt
|
||||||
|
|
||||||
|
# Bower dependency directory (https://bower.io/)
|
||||||
|
bower_components
|
||||||
|
|
||||||
|
# node-waf configuration
|
||||||
|
.lock-wscript
|
||||||
|
|
||||||
|
# Compiled binary addons (http://nodejs.org/api/addons.html)
|
||||||
|
build/Release
|
||||||
|
|
||||||
|
# Dependency directories
|
||||||
|
node_modules/
|
||||||
|
|
||||||
|
# Optional npm cache directory
|
||||||
|
.npm
|
||||||
|
|
||||||
|
# Optional eslint cache
|
||||||
|
.eslintcache
|
||||||
|
|
||||||
|
# Optional REPL history
|
||||||
|
.node_repl_history
|
||||||
|
|
||||||
|
# Output of 'npm pack'
|
||||||
|
*.tgz
|
||||||
|
|
||||||
|
# Yarn Integrity file
|
||||||
|
.yarn-integrity
|
||||||
|
|
||||||
|
# dotenv environment variables file
|
||||||
|
.env
|
||||||
|
*.tsbuildinfo
|
||||||
|
next-env.d.ts
|
||||||
|
|
||||||
|
# database
|
||||||
|
still-database/
|
||||||
|
|
||||||
|
.next/
|
||||||
|
certificates
|
||||||
|
.vercel
|
||||||
BIN
4xnored.png
Normal file
|
After Width: | Height: | Size: 72 KiB |
145
MIGRATION_COMPLETE.md
Normal file
@@ -0,0 +1,145 @@
|
|||||||
|
# Migration Summary: Firebase → Navidrome/Subsonic
|
||||||
|
|
||||||
|
## ✅ Completed Migration Tasks
|
||||||
|
|
||||||
|
### 🗑️ Removed Legacy Systems
|
||||||
|
- [x] **Firebase Dependencies**: Removed firebase, react-firebase-hooks packages
|
||||||
|
- [x] **Static Data Files**: Moved `app/data/` (albums.ts, artists.ts, playlists.ts) to backup
|
||||||
|
- [x] **Firebase Config**: Moved `app/firebase/` directory to backup
|
||||||
|
- [x] **Authentication System**: Removed Firebase Auth integration
|
||||||
|
- [x] **Database Connections**: Removed Firestore database calls
|
||||||
|
|
||||||
|
### 🚀 Implemented Navidrome Integration
|
||||||
|
- [x] **Navidrome API Client** (`lib/navidrome.ts`)
|
||||||
|
- Subsonic API authentication with token-based security
|
||||||
|
- All major endpoints: ping, getArtists, getAlbums, getAlbum, search3, etc.
|
||||||
|
- Stream URL generation for audio playback
|
||||||
|
- Cover art URL generation with size parameters
|
||||||
|
- Star/unstar functionality for favorites
|
||||||
|
- Scrobbling support for play tracking
|
||||||
|
|
||||||
|
- [x] **React Context Provider** (`app/components/NavidromeContext.tsx`)
|
||||||
|
- Global state management for music library data
|
||||||
|
- Loading states for UI feedback
|
||||||
|
- Error handling and connection testing
|
||||||
|
- Data fetching with automatic refresh
|
||||||
|
- CRUD operations for playlists
|
||||||
|
|
||||||
|
### 🎵 Updated Audio System
|
||||||
|
- [x] **AudioPlayerContext** - Completely rewritten for Navidrome
|
||||||
|
- Real audio streaming instead of static file URLs
|
||||||
|
- Queue management with Navidrome song objects
|
||||||
|
- Automatic scrobbling when tracks play
|
||||||
|
- Track conversion from Navidrome Song to playable Track format
|
||||||
|
|
||||||
|
- [x] **AudioPlayer Component**
|
||||||
|
- Updated to handle Navidrome streaming URLs
|
||||||
|
- Dynamic cover art from Navidrome getCoverArt API
|
||||||
|
- Proper track metadata display (artist, album, duration)
|
||||||
|
|
||||||
|
### 🎨 Updated UI Components
|
||||||
|
- [x] **AlbumArtwork Component**
|
||||||
|
- Uses Navidrome Album interface
|
||||||
|
- Dynamic cover art with getCoverArt API
|
||||||
|
- Context menu integration with Navidrome playlists
|
||||||
|
- Proper album metadata display (year, genre, song count)
|
||||||
|
|
||||||
|
- [x] **ArtistIcon Component**
|
||||||
|
- Uses Navidrome Artist interface
|
||||||
|
- Artist cover art support
|
||||||
|
- Album count display
|
||||||
|
- Star/unstar functionality in context menu
|
||||||
|
|
||||||
|
### 📄 Updated Pages
|
||||||
|
- [x] **Main Page** (`app/page.tsx`)
|
||||||
|
- Uses NavidromeContext for album data
|
||||||
|
- Loading states with skeleton UI
|
||||||
|
- Error handling for connection issues
|
||||||
|
- Recent and newest album sections
|
||||||
|
|
||||||
|
- [x] **Album Detail Page** (`app/album/[id]/page.tsx`)
|
||||||
|
- Fetches album and songs from Navidrome
|
||||||
|
- Real-time song playback with streaming
|
||||||
|
- Star/unstar album functionality
|
||||||
|
- Proper track listing with metadata
|
||||||
|
|
||||||
|
- [x] **Artist Page** (`app/artist/[artist]/page.tsx`)
|
||||||
|
- Artist details from Navidrome API
|
||||||
|
- Dynamic album grid for artist
|
||||||
|
- Star/unstar artist functionality
|
||||||
|
- Modern gradient header design
|
||||||
|
|
||||||
|
- [x] **Library Pages**
|
||||||
|
- `app/library/albums/page.tsx` - Shows all albums in grid layout
|
||||||
|
- `app/library/artists/page.tsx` - Shows all artists in grid layout
|
||||||
|
- `app/library/playlists/page.tsx` - Playlist management with CRUD operations
|
||||||
|
|
||||||
|
### 🔧 Configuration & Documentation
|
||||||
|
- [x] **Environment Configuration**
|
||||||
|
- `.env.example` with Navidrome connection settings
|
||||||
|
- Removed Firebase environment variables from package.json
|
||||||
|
|
||||||
|
- [x] **Documentation**
|
||||||
|
- `NAVIDROME_MIGRATION.md` - Detailed migration guide
|
||||||
|
- Updated `README.md` with new setup instructions
|
||||||
|
- Feature documentation and troubleshooting
|
||||||
|
|
||||||
|
- [x] **Type Safety**
|
||||||
|
- TypeScript interfaces matching Subsonic API responses
|
||||||
|
- Proper error handling throughout the application
|
||||||
|
- Type-safe component props and context values
|
||||||
|
|
||||||
|
### 🧪 Testing
|
||||||
|
- [x] **Test Suite** (`__tests__/navidrome.test.ts`)
|
||||||
|
- API client functionality tests
|
||||||
|
- TypeScript interface validation
|
||||||
|
- URL generation testing
|
||||||
|
- Configuration validation
|
||||||
|
|
||||||
|
## 🎯 Key Benefits Achieved
|
||||||
|
|
||||||
|
### **Real Music Streaming**
|
||||||
|
- Replaced static MP3 URLs with dynamic Navidrome streaming
|
||||||
|
- Support for multiple audio formats and bitrates
|
||||||
|
- Proper audio metadata from music files
|
||||||
|
|
||||||
|
### **Dynamic Library**
|
||||||
|
- No more manual JSON file management
|
||||||
|
- Auto-discovery of new music added to Navidrome
|
||||||
|
- Real-time library updates
|
||||||
|
|
||||||
|
### **Enhanced Features**
|
||||||
|
- Scrobbling for play tracking
|
||||||
|
- Star/favorite functionality
|
||||||
|
- Playlist management (create, edit, delete)
|
||||||
|
- Search across entire music library
|
||||||
|
- High-quality album artwork
|
||||||
|
|
||||||
|
### **Better Architecture**
|
||||||
|
- Removed Firebase dependency completely
|
||||||
|
- Self-hosted music solution
|
||||||
|
- Standards-based Subsonic API integration
|
||||||
|
- Type-safe development with proper interfaces
|
||||||
|
|
||||||
|
## 🔄 Migration Path
|
||||||
|
|
||||||
|
1. **Backup**: Old Firebase and static data moved to `-old` directories
|
||||||
|
2. **Dependencies**: Firebase packages removed, crypto built-in used
|
||||||
|
3. **Environment**: New `.env.local` needed with Navidrome credentials
|
||||||
|
4. **Data Flow**: `Static JSON → Firebase → Navidrome API`
|
||||||
|
5. **Authentication**: `Firebase Auth → Navidrome Server Authentication`
|
||||||
|
6. **Streaming**: `Static Files → Navidrome Stream API`
|
||||||
|
|
||||||
|
## 🚦 Ready for Production
|
||||||
|
|
||||||
|
The application is now fully migrated and ready for use with any Navidrome server. All core functionality has been preserved and enhanced:
|
||||||
|
|
||||||
|
- ✅ Browse music library (albums, artists, songs)
|
||||||
|
- ✅ Audio playback with queue management
|
||||||
|
- ✅ Search functionality
|
||||||
|
- ✅ Playlist management
|
||||||
|
- ✅ Favorites/starring
|
||||||
|
- ✅ Responsive design
|
||||||
|
- ✅ Error handling and loading states
|
||||||
|
|
||||||
|
**Next Steps**: Set up Navidrome server and configure connection in `.env.local`
|
||||||
151
NAVIDROME_MIGRATION.md
Normal file
@@ -0,0 +1,151 @@
|
|||||||
|
# Navidrome Integration Migration
|
||||||
|
|
||||||
|
This project has been migrated from a Firebase-based system with static data to use **Navidrome/Subsonic** as the backend music server.
|
||||||
|
|
||||||
|
## What Changed
|
||||||
|
|
||||||
|
### Removed:
|
||||||
|
- Firebase authentication and database
|
||||||
|
- Static album/artist data files
|
||||||
|
- Custom database URLs and tracklist JSON files
|
||||||
|
|
||||||
|
### Added:
|
||||||
|
- Navidrome/Subsonic API integration
|
||||||
|
- Real-time music streaming
|
||||||
|
- Dynamic music library loading
|
||||||
|
- Album cover art from Navidrome
|
||||||
|
- Playlist management through Navidrome
|
||||||
|
- Star/favorite functionality
|
||||||
|
- Scrobbling support
|
||||||
|
|
||||||
|
## Setup Instructions
|
||||||
|
|
||||||
|
### 1. Install Navidrome
|
||||||
|
|
||||||
|
First, you need to set up a Navidrome server. You can:
|
||||||
|
|
||||||
|
- **Self-host**: Follow the [Navidrome installation guide](https://www.navidrome.org/docs/installation/)
|
||||||
|
- **Docker**: Use the official Docker image
|
||||||
|
- **Pre-built binaries**: Download from GitHub releases
|
||||||
|
|
||||||
|
### 2. Configure Environment Variables
|
||||||
|
|
||||||
|
Copy `.env.example` to `.env.local` and configure your Navidrome server:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cp .env.example .env.local
|
||||||
|
```
|
||||||
|
|
||||||
|
Edit `.env.local`:
|
||||||
|
```env
|
||||||
|
NEXT_PUBLIC_NAVIDROME_URL=http://localhost:4533
|
||||||
|
NEXT_PUBLIC_NAVIDROME_USERNAME=your_username
|
||||||
|
NEXT_PUBLIC_NAVIDROME_PASSWORD=your_password
|
||||||
|
```
|
||||||
|
|
||||||
|
For production, use your actual Navidrome server URL:
|
||||||
|
```env
|
||||||
|
NEXT_PUBLIC_NAVIDROME_URL=https://your-navidrome-server.com
|
||||||
|
NEXT_PUBLIC_NAVIDROME_USERNAME=your_username
|
||||||
|
NEXT_PUBLIC_NAVIDROME_PASSWORD=your_password
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. Install Dependencies
|
||||||
|
|
||||||
|
Remove Firebase dependencies and install:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm install
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. Run the Application
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm dev
|
||||||
|
```
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
### Music Library
|
||||||
|
- **Albums**: Browse all albums in your Navidrome library
|
||||||
|
- **Artists**: Browse all artists with album counts
|
||||||
|
- **Songs**: Play individual tracks with streaming
|
||||||
|
- **Search**: Search across artists, albums, and songs
|
||||||
|
- **Playlists**: Create and manage playlists
|
||||||
|
|
||||||
|
### Audio Player
|
||||||
|
- **Streaming**: Direct streaming from Navidrome server
|
||||||
|
- **Queue Management**: Add albums/artists to queue
|
||||||
|
- **Scrobbling**: Track listening history
|
||||||
|
- **Controls**: Play, pause, skip, volume control
|
||||||
|
|
||||||
|
### User Features
|
||||||
|
- **Favorites**: Star/unstar albums, artists, and songs
|
||||||
|
- **Playlists**: Create, edit, and delete playlists
|
||||||
|
- **Recently Added**: See newest additions to your library
|
||||||
|
- **Album Artwork**: High quality cover art from Navidrome
|
||||||
|
|
||||||
|
## API Integration
|
||||||
|
|
||||||
|
The app uses the Subsonic API (compatible with Navidrome) with these endpoints:
|
||||||
|
|
||||||
|
- `ping` - Test server connection
|
||||||
|
- `getArtists` - Get all artists
|
||||||
|
- `getAlbums` - Get albums (newest, recent, etc.)
|
||||||
|
- `getAlbum` - Get album details and tracks
|
||||||
|
- `search3` - Search music library
|
||||||
|
- `getPlaylists` - Get user playlists
|
||||||
|
- `stream` - Stream audio files
|
||||||
|
- `getCoverArt` - Get album/artist artwork
|
||||||
|
- `star/unstar` - Favorite items
|
||||||
|
- `scrobble` - Track listening
|
||||||
|
|
||||||
|
## File Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
lib/
|
||||||
|
navidrome.ts # Navidrome API client
|
||||||
|
app/
|
||||||
|
components/
|
||||||
|
NavidromeContext.tsx # React context for Navidrome data
|
||||||
|
AudioPlayerContext.tsx # Updated for Navidrome streaming
|
||||||
|
album-artwork.tsx # Updated for Navidrome albums
|
||||||
|
artist-icon.tsx # Updated for Navidrome artists
|
||||||
|
AudioPlayer.tsx # Updated for streaming
|
||||||
|
```
|
||||||
|
|
||||||
|
## Migration Notes
|
||||||
|
|
||||||
|
- **Authentication**: Removed Firebase auth (Navidrome handles users)
|
||||||
|
- **Data Source**: Now uses live music library instead of static JSON
|
||||||
|
- **Streaming**: Direct audio streaming instead of static file URLs
|
||||||
|
- **Cover Art**: Dynamic cover art from Navidrome instead of static images
|
||||||
|
- **Playlists**: Managed through Navidrome instead of static data
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Connection Issues
|
||||||
|
1. Verify Navidrome server is running
|
||||||
|
2. Check URL, username, and password in `.env.local`
|
||||||
|
3. Ensure CORS is properly configured in Navidrome
|
||||||
|
4. Check network connectivity
|
||||||
|
|
||||||
|
### Audio Issues
|
||||||
|
1. Verify audio files are properly imported in Navidrome
|
||||||
|
2. Check browser audio permissions
|
||||||
|
3. Ensure audio codecs are supported
|
||||||
|
|
||||||
|
### Performance
|
||||||
|
1. Navidrome server performance affects loading times
|
||||||
|
2. Consider server location for streaming quality
|
||||||
|
3. Check network bandwidth for audio streaming
|
||||||
|
|
||||||
|
## Development
|
||||||
|
|
||||||
|
The app now uses TypeScript interfaces that match the Subsonic API responses. All components have been updated to work with the new data structure and real-time streaming.
|
||||||
|
|
||||||
|
Key changes:
|
||||||
|
- Album interface now includes Navidrome-specific fields
|
||||||
|
- Artist interface includes album counts and cover art
|
||||||
|
- Song interface includes streaming URLs and metadata
|
||||||
|
- Playlist interface matches Navidrome playlist structure
|
||||||
110
README.md
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|

|
||||||
|
# stillnavidrome (project still)
|
||||||
|
> still water, now with navidrome
|
||||||
|
|
||||||
|
> project based on [shadcn/ui](https://github.com/shadcn-ui/ui)'s music template
|
||||||
|
|
||||||
|
This is a modern music streaming web application built with [Next.js](https://nextjs.org/) and [shadcn/ui](https://ui.shadcn.com/), now powered by **Navidrome** for a complete self-hosted music streaming experience.
|
||||||
|
|
||||||
|
**✨ New**: Migrated from Firebase + static data to **Navidrome/Subsonic** integration for real music streaming!
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
- 🎵 **Real Music Streaming** via Navidrome/Subsonic API
|
||||||
|
- 📱 **Modern UI** with shadcn/ui components
|
||||||
|
- 🎨 **Dynamic Album Artwork** from your music library
|
||||||
|
- ⭐ **Favorites** - Star albums, artists, and songs
|
||||||
|
- 📋 **Playlist Management** - Create and manage playlists
|
||||||
|
- 🔍 **Search** - Find music across your entire library
|
||||||
|
- 🎧 **Audio Player** with queue management
|
||||||
|
- 📊 **Scrobbling** - Track your listening history
|
||||||
|
|
||||||
|
### Preview
|
||||||
|

|
||||||
|
|
||||||
|
## Quick Start
|
||||||
|
|
||||||
|
### Prerequisites
|
||||||
|
- [Navidrome](https://www.navidrome.org/) server running
|
||||||
|
- Node.js 18+ and pnpm
|
||||||
|
|
||||||
|
### Setup
|
||||||
|
|
||||||
|
1. **Clone and install**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/sillyangel/project-still.git
|
||||||
|
cd project-still/
|
||||||
|
pnpm install
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Configure Navidrome connection**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cp .env.example .env.local
|
||||||
|
```
|
||||||
|
|
||||||
|
Edit `.env.local` with your Navidrome server details:
|
||||||
|
|
||||||
|
```env
|
||||||
|
NEXT_PUBLIC_NAVIDROME_URL=http://localhost:4533
|
||||||
|
NEXT_PUBLIC_NAVIDROME_USERNAME=your_username
|
||||||
|
NEXT_PUBLIC_NAVIDROME_PASSWORD=your_password
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Run the development server**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm dev
|
||||||
|
```
|
||||||
|
|
||||||
|
Open [http://localhost:40625](http://localhost:40625) in your browser.
|
||||||
|
|
||||||
|
## Migration from Firebase
|
||||||
|
|
||||||
|
This project was migrated from Firebase to Navidrome. See [NAVIDROME_MIGRATION.md](./NAVIDROME_MIGRATION.md) for detailed migration notes and troubleshooting.
|
||||||
|
|
||||||
|
## Tech Stack
|
||||||
|
|
||||||
|
- **Frontend**: Next.js 15, React 19, TypeScript
|
||||||
|
- **UI**: shadcn/ui, Tailwind CSS, Radix UI
|
||||||
|
- **Backend**: Navidrome (Subsonic API compatible)
|
||||||
|
- **Audio**: Web Audio API with streaming
|
||||||
|
- **State**: React Context for global state management
|
||||||
|
|
||||||
|
## Project Structure
|
||||||
|
```
|
||||||
|
lib/
|
||||||
|
navidrome.ts # Navidrome API client
|
||||||
|
app/
|
||||||
|
components/
|
||||||
|
NavidromeContext.tsx # Data provider for Navidrome
|
||||||
|
AudioPlayerContext.tsx # Audio player state management
|
||||||
|
album-artwork.tsx # Album display component
|
||||||
|
artist-icon.tsx # Artist display component
|
||||||
|
AudioPlayer.tsx # Main audio player
|
||||||
|
library/ # Library pages
|
||||||
|
albums/ # Albums view
|
||||||
|
artists/ # Artists view
|
||||||
|
playlists/ # Playlists view
|
||||||
|
album/[id]/ # Album detail page
|
||||||
|
artist/[artist]/ # Artist detail page
|
||||||
|
```
|
||||||
|
|
||||||
|
## Contributing
|
||||||
|
|
||||||
|
1. Fork the repository
|
||||||
|
2. Create a feature branch
|
||||||
|
3. Make your changes
|
||||||
|
4. Test with your Navidrome server
|
||||||
|
5. Submit a pull request
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
This project is licensed under the MIT License.
|
||||||
|
|
||||||
|
## Acknowledgments
|
||||||
|
|
||||||
|
- [shadcn/ui](https://ui.shadcn.com/) for the beautiful UI components
|
||||||
|
- [Navidrome](https://www.navidrome.org/) for the amazing music server
|
||||||
|
- [Subsonic API](http://www.subsonic.org/pages/api.jsp) for the API specification
|
||||||
152
app/album/[id]/page.tsx
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
'use client';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { useParams } from 'next/navigation';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { Album, Song } from '@/lib/navidrome';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import { Play, Heart } from 'lucide-react';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import Link from 'next/link';
|
||||||
|
import { PlusIcon } from "@radix-ui/react-icons";
|
||||||
|
import { useAudioPlayer } from '@/app/components/AudioPlayerContext'
|
||||||
|
import Loading from "@/app/components/loading";
|
||||||
|
import { Separator } from '@/components/ui/separator';
|
||||||
|
import { getNavidromeAPI } from '@/lib/navidrome';
|
||||||
|
|
||||||
|
export default function AlbumPage() {
|
||||||
|
const { id } = useParams();
|
||||||
|
const [album, setAlbum] = useState<Album | null>(null);
|
||||||
|
const [tracklist, setTracklist] = useState<Song[]>([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const [isStarred, setIsStarred] = useState(false);
|
||||||
|
const { getAlbum, starItem, unstarItem } = useNavidrome();
|
||||||
|
const { playTrack, addAlbumToQueue, playAlbum, playAlbumFromTrack } = useAudioPlayer();
|
||||||
|
const api = getNavidromeAPI();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchAlbum = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
console.log(`Fetching album with id: ${id}`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const albumData = await getAlbum(id as string);
|
||||||
|
setAlbum(albumData.album);
|
||||||
|
setTracklist(albumData.songs);
|
||||||
|
setIsStarred(!!albumData.album.starred);
|
||||||
|
console.log(`Album found: ${albumData.album.name}`);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch album:', error);
|
||||||
|
}
|
||||||
|
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (id) {
|
||||||
|
fetchAlbum();
|
||||||
|
}
|
||||||
|
}, [id, getAlbum]);
|
||||||
|
|
||||||
|
const handleStar = async () => {
|
||||||
|
if (!album) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (isStarred) {
|
||||||
|
await unstarItem(album.id, 'album');
|
||||||
|
setIsStarred(false);
|
||||||
|
} else {
|
||||||
|
await starItem(album.id, 'album');
|
||||||
|
setIsStarred(true);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to star/unstar album:', error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!album) {
|
||||||
|
return <p>Album not found</p>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlePlayClick = async (song: Song): Promise<void> => {
|
||||||
|
if (!album) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await playAlbumFromTrack(album.id, song.id);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to play album from track:', error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDuration = (duration: number): string => {
|
||||||
|
const minutes = Math.floor(duration / 60);
|
||||||
|
const seconds = duration % 60;
|
||||||
|
return `${minutes}:${seconds.toString().padStart(2, '0')}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Get cover art URL with proper fallback
|
||||||
|
const coverArtUrl = album.coverArt
|
||||||
|
? api.getCoverArtUrl(album.coverArt, 300)
|
||||||
|
: '/default-user.jpg';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<div className="space-y-4">
|
||||||
|
<div className="flex items-start gap-6">
|
||||||
|
<Image
|
||||||
|
src={coverArtUrl}
|
||||||
|
alt={album.name}
|
||||||
|
width={300}
|
||||||
|
height={300}
|
||||||
|
className="rounded-md"
|
||||||
|
/>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<div className="flex items-center space-x-4">
|
||||||
|
<p className="text-3xl font-semibold tracking-tight">{album.name}</p>
|
||||||
|
<Button onClick={handleStar} variant="ghost">
|
||||||
|
<Heart className={isStarred ? 'text-primary' : 'text-gray-500'} fill={isStarred ? 'var(--primary)' : ""}/>
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<Link href={`/artist/${album.artistId}`}>
|
||||||
|
<p className="text-xl text-primary mt-0 mb-4 underline">{album.artist}</p>
|
||||||
|
</Link>
|
||||||
|
<Button className="px-5" onClick={() => playAlbum(album.id)}>
|
||||||
|
<Play />
|
||||||
|
Play Album
|
||||||
|
</Button>
|
||||||
|
<div className="text-sm text-muted-foreground">
|
||||||
|
<p>{album.songCount} songs • {album.year} • {album.genre}</p>
|
||||||
|
<p>Duration: {formatDuration(album.duration)}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Separator />
|
||||||
|
{tracklist.map((song, index) => (
|
||||||
|
<div key={song.id} className="py-2 flex justify-between items-center hover:bg-hover rounded-lg cursor-pointer" onClick={() => handlePlayClick(song)}>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<div className="mr-2 w-6 text-right">{song.track || index + 1}</div>
|
||||||
|
<div>
|
||||||
|
<p className="font-semibold text-lg flex items-center">
|
||||||
|
{song.title}
|
||||||
|
</p>
|
||||||
|
<p className="text-sm font-normal flex items-center">
|
||||||
|
<p className="text-gray-400">{song.artist}</p>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center space-x-2">
|
||||||
|
<p className="text-sm mr-4">{formatDuration(song.duration)}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<br/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
123
app/artist/[artist]/page.tsx
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
'use client';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { useParams } from 'next/navigation';
|
||||||
|
import { Album, Artist } from '@/lib/navidrome';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import { AlbumArtwork } from '@/app/components/album-artwork';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { Heart } from 'lucide-react';
|
||||||
|
import { ScrollArea, ScrollBar } from '@/components/ui/scroll-area';
|
||||||
|
import Loading from '@/app/components/loading';
|
||||||
|
import { getNavidromeAPI } from '@/lib/navidrome';
|
||||||
|
|
||||||
|
export default function ArtistPage() {
|
||||||
|
const { artist: artistId } = useParams();
|
||||||
|
const [isStarred, setIsStarred] = useState(false);
|
||||||
|
const [artistAlbums, setArtistAlbums] = useState<Album[]>([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const [artist, setArtist] = useState<Artist | null>(null);
|
||||||
|
const { getArtist, starItem, unstarItem } = useNavidrome();
|
||||||
|
const api = getNavidromeAPI();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchArtistData = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
try {
|
||||||
|
if (artistId) {
|
||||||
|
const artistData = await getArtist(artistId as string);
|
||||||
|
setArtist(artistData.artist);
|
||||||
|
setArtistAlbums(artistData.albums);
|
||||||
|
setIsStarred(!!artistData.artist.starred);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch artist data:', error);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
fetchArtistData();
|
||||||
|
}, [artistId, getArtist]);
|
||||||
|
|
||||||
|
const handleStar = async () => {
|
||||||
|
if (!artist) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (isStarred) {
|
||||||
|
await unstarItem(artist.id, 'artist');
|
||||||
|
setIsStarred(false);
|
||||||
|
} else {
|
||||||
|
await starItem(artist.id, 'artist');
|
||||||
|
setIsStarred(true);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to star/unstar artist:', error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!artist || artistAlbums.length === 0) {
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8 flex items-center justify-center">
|
||||||
|
<p>No albums found for this artist</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get artist image URL with proper fallback
|
||||||
|
const artistImageUrl = artist.coverArt
|
||||||
|
? api.getCoverArtUrl(artist.coverArt, 300)
|
||||||
|
: '/default-user.jpg';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<div className="space-y-6">
|
||||||
|
{/* Artist Header */}
|
||||||
|
<div className="relative bg-gradient-to-r from-blue-900 to-purple-900 rounded-lg p-8">
|
||||||
|
<div className="flex items-center space-x-6">
|
||||||
|
<div className="relative">
|
||||||
|
<Image
|
||||||
|
src={artistImageUrl}
|
||||||
|
alt={artist.name}
|
||||||
|
width={120}
|
||||||
|
height={120}
|
||||||
|
className="rounded-full border-4 border-white shadow-lg"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex-1">
|
||||||
|
<h1 className="text-4xl font-bold text-white mb-2">{artist.name}</h1>
|
||||||
|
<p className="text-white/80 mb-4">{artist.albumCount} albums</p>
|
||||||
|
<Button onClick={handleStar} variant="secondary" className="mr-4">
|
||||||
|
<Heart className={isStarred ? 'text-red-500' : 'text-gray-500'} fill={isStarred ? 'red' : 'none'}/>
|
||||||
|
{isStarred ? 'Starred' : 'Star Artist'}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Albums Section */}
|
||||||
|
<div className="space-y-4">
|
||||||
|
<h2 className="text-2xl font-semibold tracking-tight">Albums</h2>
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5 2xl:grid-cols-6 gap-4 pb-4">
|
||||||
|
{artistAlbums.map((album) => (
|
||||||
|
<AlbumArtwork
|
||||||
|
key={album.id}
|
||||||
|
album={album}
|
||||||
|
className="w-full"
|
||||||
|
aspectRatio="square"
|
||||||
|
width={200}
|
||||||
|
height={200}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
87
app/browse/page.tsx
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { ScrollArea, ScrollBar } from '@/components/ui/scroll-area';
|
||||||
|
import { Separator } from '@/components/ui/separator';
|
||||||
|
import { Tabs, TabsContent } from '@/components/ui/tabs';
|
||||||
|
import { AlbumArtwork } from '@/app/components/album-artwork';
|
||||||
|
import { ArtistIcon } from '@/app/components/artist-icon';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import Loading from '@/app/components/loading';
|
||||||
|
|
||||||
|
export default function BrowsePage() {
|
||||||
|
const { albums, artists, isLoading } = useNavidrome();
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<>
|
||||||
|
<Tabs defaultValue="music" className="h-full flex flex-col space-y-6">
|
||||||
|
<TabsContent value="music" className="border-none p-0 outline-none flex flex-col flex-grow">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Artists
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
the people who make the music
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative flex-grow">
|
||||||
|
<div className="relative">
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="flex space-x-4 pb-4">
|
||||||
|
{artists.map((artist) => (
|
||||||
|
<ArtistIcon
|
||||||
|
key={artist.id}
|
||||||
|
artist={artist}
|
||||||
|
className="flex-shrink-0"
|
||||||
|
size={150}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Browse
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
Browse the full collection of music available.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative flex-grow">
|
||||||
|
<ScrollArea className="h-full">
|
||||||
|
<div className="h-full overflow-y-auto">
|
||||||
|
<div className="flex flex-wrap gap-4 p-4">
|
||||||
|
{albums.map((album) => (
|
||||||
|
<AlbumArtwork
|
||||||
|
key={album.id}
|
||||||
|
album={album}
|
||||||
|
className="w-[230px]"
|
||||||
|
aspectRatio="square"
|
||||||
|
width={230}
|
||||||
|
height={230}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
</>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
202
app/components/AudioPlayer.tsx
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
'use client';
|
||||||
|
import React, { useEffect, useRef, useState } from 'react';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { useAudioPlayer } from '@/app/components/AudioPlayerContext';
|
||||||
|
import { FaPlay, FaPause, FaVolumeHigh, FaForward, FaBackward } from "react-icons/fa6";
|
||||||
|
import ColorThief from '@neutrixs/colorthief';
|
||||||
|
import { Progress } from '@/components/ui/progress';
|
||||||
|
import { useToast } from '@/hooks/use-toast';
|
||||||
|
|
||||||
|
export const AudioPlayer: React.FC = () => {
|
||||||
|
const { currentTrack, playPreviousTrack, addToQueue, playNextTrack, clearQueue } = useAudioPlayer();
|
||||||
|
const audioRef = useRef<HTMLAudioElement>(null);
|
||||||
|
const [progress, setProgress] = useState(0);
|
||||||
|
const [isPlaying, setIsPlaying] = useState(false);
|
||||||
|
const [showVolumeSlider, setShowVolumeSlider] = useState(false);
|
||||||
|
const [volume, setVolume] = useState(1);
|
||||||
|
const [isClient, setIsClient] = useState(false);
|
||||||
|
const audioCurrent = audioRef.current;
|
||||||
|
const { toast } = useToast();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setIsClient(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Save position when component unmounts or track changes
|
||||||
|
useEffect(() => {
|
||||||
|
return () => {
|
||||||
|
const audioCurrent = audioRef.current;
|
||||||
|
if (audioCurrent && currentTrack && audioCurrent.currentTime > 10) {
|
||||||
|
localStorage.setItem(`navidrome-track-time-${currentTrack.id}`, audioCurrent.currentTime.toString());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, [currentTrack?.id]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const audioCurrent = audioRef.current;
|
||||||
|
|
||||||
|
if (currentTrack && audioCurrent && audioCurrent.src !== currentTrack.url) {
|
||||||
|
audioCurrent.src = currentTrack.url;
|
||||||
|
|
||||||
|
// Check for saved timestamp (only restore if more than 10 seconds in)
|
||||||
|
const savedTime = localStorage.getItem(`navidrome-track-time-${currentTrack.id}`);
|
||||||
|
if (savedTime) {
|
||||||
|
const time = parseFloat(savedTime);
|
||||||
|
// Only restore if we were at least 10 seconds in and not near the end
|
||||||
|
if (time > 10 && time < (currentTrack.duration - 30)) {
|
||||||
|
const restorePosition = () => {
|
||||||
|
if (audioCurrent.readyState >= 2) { // HAVE_CURRENT_DATA
|
||||||
|
audioCurrent.currentTime = time;
|
||||||
|
audioCurrent.removeEventListener('loadeddata', restorePosition);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (audioCurrent.readyState >= 2) {
|
||||||
|
audioCurrent.currentTime = time;
|
||||||
|
} else {
|
||||||
|
audioCurrent.addEventListener('loadeddata', restorePosition);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
audioCurrent.play();
|
||||||
|
setIsPlaying(true);
|
||||||
|
}
|
||||||
|
}, [currentTrack?.id, currentTrack?.url]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const audioCurrent = audioRef.current;
|
||||||
|
let lastSavedTime = 0;
|
||||||
|
|
||||||
|
const updateProgress = () => {
|
||||||
|
if (audioCurrent && currentTrack) {
|
||||||
|
setProgress((audioCurrent.currentTime / audioCurrent.duration) * 100);
|
||||||
|
|
||||||
|
// Save current time every 10 seconds, but only if we've moved forward significantly
|
||||||
|
const currentTime = audioCurrent.currentTime;
|
||||||
|
if (Math.abs(currentTime - lastSavedTime) >= 10 && currentTime > 10) {
|
||||||
|
localStorage.setItem(`navidrome-track-time-${currentTrack.id}`, currentTime.toString());
|
||||||
|
lastSavedTime = currentTime;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleTrackEnd = () => {
|
||||||
|
if (currentTrack) {
|
||||||
|
// Clear saved time when track ends
|
||||||
|
localStorage.removeItem(`navidrome-track-time-${currentTrack.id}`);
|
||||||
|
}
|
||||||
|
playNextTrack();
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSeeked = () => {
|
||||||
|
if (audioCurrent && currentTrack) {
|
||||||
|
// Save immediately when user seeks
|
||||||
|
localStorage.setItem(`navidrome-track-time-${currentTrack.id}`, audioCurrent.currentTime.toString());
|
||||||
|
lastSavedTime = audioCurrent.currentTime;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (audioCurrent) {
|
||||||
|
audioCurrent.addEventListener('timeupdate', updateProgress);
|
||||||
|
audioCurrent.addEventListener('ended', handleTrackEnd);
|
||||||
|
audioCurrent.addEventListener('seeked', handleSeeked);
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
if (audioCurrent) {
|
||||||
|
audioCurrent.removeEventListener('timeupdate', updateProgress);
|
||||||
|
audioCurrent.removeEventListener('ended', handleTrackEnd);
|
||||||
|
audioCurrent.removeEventListener('seeked', handleSeeked);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, [playNextTrack, currentTrack]);
|
||||||
|
|
||||||
|
const handleProgressClick = (e: React.MouseEvent<HTMLDivElement, MouseEvent>) => {
|
||||||
|
if (audioCurrent && currentTrack) {
|
||||||
|
const rect = e.currentTarget.getBoundingClientRect();
|
||||||
|
const clickX = e.clientX - rect.left;
|
||||||
|
const newTime = (clickX / rect.width) * audioCurrent.duration;
|
||||||
|
audioCurrent.currentTime = newTime;
|
||||||
|
|
||||||
|
// Save the new position immediately
|
||||||
|
localStorage.setItem(`navidrome-track-time-${currentTrack.id}`, newTime.toString());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const togglePlayPause = () => {
|
||||||
|
if (audioCurrent) {
|
||||||
|
if (isPlaying) {
|
||||||
|
audioCurrent.pause();
|
||||||
|
} else {
|
||||||
|
audioCurrent.play();
|
||||||
|
}
|
||||||
|
setIsPlaying(!isPlaying);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const handleVolumeChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const newVolume = parseFloat(e.target.value);
|
||||||
|
setVolume(newVolume);
|
||||||
|
if (audioCurrent) {
|
||||||
|
audioCurrent.volume = newVolume;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function formatTime(seconds: number): string {
|
||||||
|
if (isNaN(seconds) || seconds < 0) {
|
||||||
|
return "0:00";
|
||||||
|
}
|
||||||
|
const minutes = Math.floor(seconds / 60);
|
||||||
|
const secs = Math.floor(seconds % 60).toString().padStart(2, "0");
|
||||||
|
return `${minutes}:${secs}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isClient) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="bg-background w-full text-white p-4 border-t border-t-1">
|
||||||
|
{currentTrack ? (
|
||||||
|
<div className="flex items-center">
|
||||||
|
<Image
|
||||||
|
src={currentTrack.coverArt || '/default-user.jpg'}
|
||||||
|
alt={currentTrack.name}
|
||||||
|
width={64}
|
||||||
|
height={64}
|
||||||
|
className="w-16 h-16 mr-4 rounded-md"
|
||||||
|
/>
|
||||||
|
<div className="flex-1 w-auto mr-4">
|
||||||
|
<p className="mb-0 font-semibold">{currentTrack.name}</p>
|
||||||
|
<p className='text-sm mt-0 text-gray-400'>{currentTrack.artist}</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col items-center mr-6">
|
||||||
|
<div className="flex items-center space-x-2 mb-2">
|
||||||
|
<button className="p-2 hover:bg-gray-700 rounded-full transition-colors" onClick={playPreviousTrack}>
|
||||||
|
<FaBackward className="w-4 h-4" />
|
||||||
|
</button>
|
||||||
|
<button className='p-3 hover:bg-gray-700 rounded-full transition-colors' onClick={togglePlayPause}>
|
||||||
|
{isPlaying ? <FaPause className="w-5 h-5" /> : <FaPlay className="w-5 h-5" />}
|
||||||
|
</button>
|
||||||
|
<button className='p-2 hover:bg-gray-700 rounded-full transition-colors' onClick={playNextTrack}>
|
||||||
|
<FaForward className="w-4 h-4" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center space-x-2 w-full">
|
||||||
|
<span className="text-xs text-gray-400 w-10 text-right">
|
||||||
|
{formatTime(audioCurrent?.currentTime ?? 0)}
|
||||||
|
</span>
|
||||||
|
<Progress value={progress} className="flex-1 cursor-pointer" onClick={handleProgressClick}/>
|
||||||
|
<span className="text-xs text-gray-400 w-10">
|
||||||
|
{formatTime(audioCurrent?.duration ?? 0)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<p>No track playing</p>
|
||||||
|
)}
|
||||||
|
<audio ref={audioRef} hidden />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
311
app/components/AudioPlayerContext.tsx
Normal file
@@ -0,0 +1,311 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import React, { createContext, useContext, useState, useEffect, useMemo, useCallback } from 'react';
|
||||||
|
import { Song, Album, Artist } from '@/lib/navidrome';
|
||||||
|
import { getNavidromeAPI } from '@/lib/navidrome';
|
||||||
|
import { useToast } from "@/hooks/use-toast";
|
||||||
|
|
||||||
|
interface Track {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
url: string;
|
||||||
|
artist: string;
|
||||||
|
album: string;
|
||||||
|
duration: number;
|
||||||
|
coverArt?: string;
|
||||||
|
albumId: string;
|
||||||
|
artistId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AudioPlayerContextProps {
|
||||||
|
currentTrack: Track | null;
|
||||||
|
playTrack: (track: Track) => void;
|
||||||
|
queue: Track[];
|
||||||
|
addToQueue: (track: Track) => void;
|
||||||
|
playNextTrack: () => void;
|
||||||
|
clearQueue: () => void;
|
||||||
|
addAlbumToQueue: (albumId: string) => Promise<void>;
|
||||||
|
playAlbum: (albumId: string) => Promise<void>;
|
||||||
|
playAlbumFromTrack: (albumId: string, startingSongId: string) => Promise<void>;
|
||||||
|
removeTrackFromQueue: (index: number) => void;
|
||||||
|
skipToTrackInQueue: (index: number) => void;
|
||||||
|
addArtistToQueue: (artistId: string) => Promise<void>;
|
||||||
|
playPreviousTrack: () => void;
|
||||||
|
isLoading: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const AudioPlayerContext = createContext<AudioPlayerContextProps | undefined>(undefined);
|
||||||
|
|
||||||
|
export const AudioPlayerProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
||||||
|
const [currentTrack, setCurrentTrack] = useState<Track | null>(null);
|
||||||
|
const [queue, setQueue] = useState<Track[]>([]);
|
||||||
|
const [playedTracks, setPlayedTracks] = useState<Track[]>([]);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const { toast } = useToast();
|
||||||
|
const api = useMemo(() => getNavidromeAPI(), []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const savedQueue = localStorage.getItem('navidrome-audioQueue');
|
||||||
|
if (savedQueue) {
|
||||||
|
try {
|
||||||
|
setQueue(JSON.parse(savedQueue));
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to parse saved queue:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
localStorage.setItem('navidrome-audioQueue', JSON.stringify(queue));
|
||||||
|
}, [queue]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const savedCurrentTrack = localStorage.getItem('navidrome-currentTrack');
|
||||||
|
if (savedCurrentTrack) {
|
||||||
|
try {
|
||||||
|
setCurrentTrack(JSON.parse(savedCurrentTrack));
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to parse saved current track:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (currentTrack) {
|
||||||
|
localStorage.setItem('navidrome-currentTrack', JSON.stringify(currentTrack));
|
||||||
|
} else {
|
||||||
|
localStorage.removeItem('navidrome-currentTrack');
|
||||||
|
}
|
||||||
|
}, [currentTrack]);
|
||||||
|
|
||||||
|
const songToTrack = useMemo(() => (song: Song): Track => {
|
||||||
|
return {
|
||||||
|
id: song.id,
|
||||||
|
name: song.title,
|
||||||
|
url: api.getStreamUrl(song.id),
|
||||||
|
artist: song.artist,
|
||||||
|
album: song.album,
|
||||||
|
duration: song.duration,
|
||||||
|
coverArt: song.coverArt ? api.getCoverArtUrl(song.coverArt, 300) : undefined,
|
||||||
|
albumId: song.albumId,
|
||||||
|
artistId: song.artistId
|
||||||
|
};
|
||||||
|
}, [api]);
|
||||||
|
|
||||||
|
const playTrack = useCallback((track: Track) => {
|
||||||
|
if (currentTrack) {
|
||||||
|
setPlayedTracks((prev) => [...prev, currentTrack]);
|
||||||
|
}
|
||||||
|
setCurrentTrack(track);
|
||||||
|
|
||||||
|
// Scrobble the track
|
||||||
|
api.scrobble(track.id).catch(error => {
|
||||||
|
console.error('Failed to scrobble track:', error);
|
||||||
|
});
|
||||||
|
}, [currentTrack, api]);
|
||||||
|
|
||||||
|
const addToQueue = useCallback((track: Track) => {
|
||||||
|
setQueue((prevQueue) => [...prevQueue, track]);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const clearQueue = useCallback(() => {
|
||||||
|
setQueue([]);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const removeTrackFromQueue = useCallback((index: number) => {
|
||||||
|
setQueue((prevQueue) => prevQueue.filter((_, i) => i !== index));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const playNextTrack = () => {
|
||||||
|
if (queue.length > 0) {
|
||||||
|
const nextTrack = queue[0];
|
||||||
|
setQueue((prevQueue) => prevQueue.slice(1));
|
||||||
|
playTrack(nextTrack);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const playPreviousTrack = () => {
|
||||||
|
if (playedTracks.length > 0) {
|
||||||
|
const previousTrack = playedTracks[playedTracks.length - 1];
|
||||||
|
setPlayedTracks((prevPlayedTracks) => prevPlayedTracks.slice(0, -1));
|
||||||
|
|
||||||
|
// Add current track back to beginning of queue
|
||||||
|
if (currentTrack) {
|
||||||
|
setQueue((prevQueue) => [currentTrack, ...prevQueue]);
|
||||||
|
}
|
||||||
|
|
||||||
|
setCurrentTrack(previousTrack);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const addAlbumToQueue = async (albumId: string) => {
|
||||||
|
setIsLoading(true);
|
||||||
|
try {
|
||||||
|
const { album, songs } = await api.getAlbum(albumId);
|
||||||
|
const tracks = songs.map(songToTrack);
|
||||||
|
setQueue((prevQueue) => [...prevQueue, ...tracks]);
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: "Album Added",
|
||||||
|
description: `Added "${album.name}" to queue`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to add album to queue:', error);
|
||||||
|
toast({
|
||||||
|
variant: "destructive",
|
||||||
|
title: "Error",
|
||||||
|
description: "Failed to add album to queue",
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const addArtistToQueue = async (artistId: string) => {
|
||||||
|
setIsLoading(true);
|
||||||
|
try {
|
||||||
|
const { artist, albums } = await api.getArtist(artistId);
|
||||||
|
|
||||||
|
// Add all albums from this artist to queue
|
||||||
|
for (const album of albums) {
|
||||||
|
const { songs } = await api.getAlbum(album.id);
|
||||||
|
const tracks = songs.map(songToTrack);
|
||||||
|
setQueue((prevQueue) => [...prevQueue, ...tracks]);
|
||||||
|
}
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: "Artist Added",
|
||||||
|
description: `Added all albums by "${artist.name}" to queue`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to add artist to queue:', error);
|
||||||
|
toast({
|
||||||
|
variant: "destructive",
|
||||||
|
title: "Error",
|
||||||
|
description: "Failed to add artist to queue",
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const playAlbum = async (albumId: string) => {
|
||||||
|
setIsLoading(true);
|
||||||
|
try {
|
||||||
|
const { album, songs } = await api.getAlbum(albumId);
|
||||||
|
const tracks = songs.map(songToTrack);
|
||||||
|
|
||||||
|
// Clear the queue and set the new tracks
|
||||||
|
setQueue(tracks.slice(1)); // All tracks except the first one
|
||||||
|
|
||||||
|
// Play the first track immediately
|
||||||
|
if (tracks.length > 0) {
|
||||||
|
playTrack(tracks[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: "Playing Album",
|
||||||
|
description: `Now playing "${album.name}"`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to play album:', error);
|
||||||
|
toast({
|
||||||
|
variant: "destructive",
|
||||||
|
title: "Error",
|
||||||
|
description: "Failed to play album",
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const playAlbumFromTrack = async (albumId: string, startingSongId: string) => {
|
||||||
|
setIsLoading(true);
|
||||||
|
try {
|
||||||
|
const { album, songs } = await api.getAlbum(albumId);
|
||||||
|
const tracks = songs.map(songToTrack);
|
||||||
|
|
||||||
|
// Find the starting track index
|
||||||
|
const startingIndex = tracks.findIndex(track => track.id === startingSongId);
|
||||||
|
|
||||||
|
if (startingIndex === -1) {
|
||||||
|
throw new Error('Starting song not found in album');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clear the queue and set the remaining tracks after the starting track
|
||||||
|
setQueue(tracks.slice(startingIndex + 1));
|
||||||
|
|
||||||
|
// Play the starting track immediately
|
||||||
|
playTrack(tracks[startingIndex]);
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: "Playing Album",
|
||||||
|
description: `Playing "${album.name}" from "${tracks[startingIndex].name}"`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to play album from track:', error);
|
||||||
|
toast({
|
||||||
|
variant: "destructive",
|
||||||
|
title: "Error",
|
||||||
|
description: "Failed to play album from selected track",
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const skipToTrackInQueue = useCallback((index: number) => {
|
||||||
|
if (index >= 0 && index < queue.length) {
|
||||||
|
const targetTrack = queue[index];
|
||||||
|
// Remove all tracks before the target track (including the target track)
|
||||||
|
setQueue((prevQueue) => prevQueue.slice(index + 1));
|
||||||
|
// Play the target track
|
||||||
|
playTrack(targetTrack);
|
||||||
|
}
|
||||||
|
}, [queue, playTrack]);
|
||||||
|
|
||||||
|
const contextValue = useMemo(() => ({
|
||||||
|
currentTrack,
|
||||||
|
playTrack,
|
||||||
|
queue,
|
||||||
|
addToQueue,
|
||||||
|
playNextTrack,
|
||||||
|
clearQueue,
|
||||||
|
addAlbumToQueue,
|
||||||
|
removeTrackFromQueue,
|
||||||
|
addArtistToQueue,
|
||||||
|
playPreviousTrack,
|
||||||
|
isLoading,
|
||||||
|
playAlbum,
|
||||||
|
playAlbumFromTrack,
|
||||||
|
skipToTrackInQueue
|
||||||
|
}), [
|
||||||
|
currentTrack,
|
||||||
|
queue,
|
||||||
|
isLoading,
|
||||||
|
playTrack,
|
||||||
|
addToQueue,
|
||||||
|
playNextTrack,
|
||||||
|
clearQueue,
|
||||||
|
addAlbumToQueue,
|
||||||
|
removeTrackFromQueue,
|
||||||
|
addArtistToQueue,
|
||||||
|
playPreviousTrack,
|
||||||
|
playAlbum,
|
||||||
|
playAlbumFromTrack,
|
||||||
|
skipToTrackInQueue
|
||||||
|
]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<AudioPlayerContext.Provider value={contextValue}>
|
||||||
|
{children}
|
||||||
|
</AudioPlayerContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useAudioPlayer = () => {
|
||||||
|
const context = useContext(AudioPlayerContext);
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('useAudioPlayer must be used within an AudioPlayerProvider');
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
295
app/components/NavidromeContext.tsx
Normal file
@@ -0,0 +1,295 @@
|
|||||||
|
'use client';
|
||||||
|
import React, { createContext, useContext, useEffect, useState, ReactNode } from 'react';
|
||||||
|
import { getNavidromeAPI, Album, Artist, Song, Playlist } from '@/lib/navidrome';
|
||||||
|
|
||||||
|
interface NavidromeContextType {
|
||||||
|
// Data
|
||||||
|
albums: Album[];
|
||||||
|
artists: Artist[];
|
||||||
|
playlists: Playlist[];
|
||||||
|
|
||||||
|
// Loading states
|
||||||
|
isLoading: boolean;
|
||||||
|
albumsLoading: boolean;
|
||||||
|
artistsLoading: boolean;
|
||||||
|
playlistsLoading: boolean;
|
||||||
|
|
||||||
|
// Error states
|
||||||
|
error: string | null;
|
||||||
|
|
||||||
|
// Methods
|
||||||
|
searchMusic: (query: string) => Promise<{ artists: Artist[]; albums: Album[]; songs: Song[] }>;
|
||||||
|
getAlbum: (albumId: string) => Promise<{ album: Album; songs: Song[] }>;
|
||||||
|
getArtist: (artistId: string) => Promise<{ artist: Artist; albums: Album[] }>;
|
||||||
|
getPlaylist: (playlistId: string) => Promise<{ playlist: Playlist; songs: Song[] }>;
|
||||||
|
getAllSongs: () => Promise<Song[]>;
|
||||||
|
refreshData: () => Promise<void>;
|
||||||
|
createPlaylist: (name: string, songIds?: string[]) => Promise<Playlist>;
|
||||||
|
updatePlaylist: (playlistId: string, name?: string, comment?: string, songIds?: string[]) => Promise<void>;
|
||||||
|
deletePlaylist: (playlistId: string) => Promise<void>;
|
||||||
|
starItem: (id: string, type: 'song' | 'album' | 'artist') => Promise<void>;
|
||||||
|
unstarItem: (id: string, type: 'song' | 'album' | 'artist') => Promise<void>;
|
||||||
|
scrobble: (songId: string) => Promise<void>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const NavidromeContext = createContext<NavidromeContextType | undefined>(undefined);
|
||||||
|
|
||||||
|
interface NavidromeProviderProps {
|
||||||
|
children: ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const NavidromeProvider: React.FC<NavidromeProviderProps> = ({ children }) => {
|
||||||
|
const [albums, setAlbums] = useState<Album[]>([]);
|
||||||
|
const [artists, setArtists] = useState<Artist[]>([]);
|
||||||
|
const [playlists, setPlaylists] = useState<Playlist[]>([]);
|
||||||
|
|
||||||
|
const [albumsLoading, setAlbumsLoading] = useState(false);
|
||||||
|
const [artistsLoading, setArtistsLoading] = useState(false);
|
||||||
|
const [playlistsLoading, setPlaylistsLoading] = useState(false);
|
||||||
|
|
||||||
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const isLoading = albumsLoading || artistsLoading || playlistsLoading;
|
||||||
|
|
||||||
|
const api = getNavidromeAPI();
|
||||||
|
|
||||||
|
const loadAlbums = async () => {
|
||||||
|
setAlbumsLoading(true);
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
const recentAlbums = await api.getAlbums('recent', 50);
|
||||||
|
const newestAlbums = await api.getAlbums('newest', 50);
|
||||||
|
|
||||||
|
// Combine and deduplicate albums
|
||||||
|
const allAlbums = [...recentAlbums, ...newestAlbums];
|
||||||
|
const uniqueAlbums = allAlbums.filter((album, index, self) =>
|
||||||
|
index === self.findIndex(a => a.id === album.id)
|
||||||
|
);
|
||||||
|
|
||||||
|
setAlbums(uniqueAlbums);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to load albums:', err);
|
||||||
|
setError('Failed to load albums');
|
||||||
|
} finally {
|
||||||
|
setAlbumsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadArtists = async () => {
|
||||||
|
setArtistsLoading(true);
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
const artistList = await api.getArtists();
|
||||||
|
setArtists(artistList);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to load artists:', err);
|
||||||
|
setError('Failed to load artists');
|
||||||
|
} finally {
|
||||||
|
setArtistsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadPlaylists = async () => {
|
||||||
|
setPlaylistsLoading(true);
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
const playlistList = await api.getPlaylists();
|
||||||
|
setPlaylists(playlistList);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to load playlists:', err);
|
||||||
|
setError('Failed to load playlists');
|
||||||
|
} finally {
|
||||||
|
setPlaylistsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const refreshData = async () => {
|
||||||
|
await Promise.all([loadAlbums(), loadArtists(), loadPlaylists()]);
|
||||||
|
};
|
||||||
|
|
||||||
|
const searchMusic = async (query: string) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
return await api.search(query);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Search failed:', err);
|
||||||
|
setError('Search failed');
|
||||||
|
return { artists: [], albums: [], songs: [] };
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getAlbum = async (albumId: string) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
return await api.getAlbum(albumId);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to get album:', err);
|
||||||
|
setError('Failed to get album');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getArtist = async (artistId: string) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
return await api.getArtist(artistId);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to get artist:', err);
|
||||||
|
setError('Failed to get artist');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getPlaylist = async (playlistId: string) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
return await api.getPlaylist(playlistId);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to get playlist:', err);
|
||||||
|
setError('Failed to get playlist');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getAllSongs = async () => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
return await api.getAllSongs();
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to get all songs:', err);
|
||||||
|
setError('Failed to get all songs');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const createPlaylist = async (name: string, songIds?: string[]) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
const playlist = await api.createPlaylist(name, songIds);
|
||||||
|
await loadPlaylists(); // Refresh playlists
|
||||||
|
return playlist;
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to create playlist:', err);
|
||||||
|
setError('Failed to create playlist');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const updatePlaylist = async (playlistId: string, name?: string, comment?: string, songIds?: string[]) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
await api.updatePlaylist(playlistId, name, comment, songIds);
|
||||||
|
await loadPlaylists(); // Refresh playlists
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to update playlist:', err);
|
||||||
|
setError('Failed to update playlist');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const deletePlaylist = async (playlistId: string) => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
await api.deletePlaylist(playlistId);
|
||||||
|
await loadPlaylists(); // Refresh playlists
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to delete playlist:', err);
|
||||||
|
setError('Failed to delete playlist');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const starItem = async (id: string, type: 'song' | 'album' | 'artist') => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
await api.star(id, type);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to star item:', err);
|
||||||
|
setError('Failed to star item');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const unstarItem = async (id: string, type: 'song' | 'album' | 'artist') => {
|
||||||
|
setError(null);
|
||||||
|
try {
|
||||||
|
await api.unstar(id, type);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to unstar item:', err);
|
||||||
|
setError('Failed to unstar item');
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const scrobble = async (songId: string) => {
|
||||||
|
try {
|
||||||
|
await api.scrobble(songId);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to scrobble:', err);
|
||||||
|
// Don't set error state for scrobbling failures as they're not critical
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// Test connection and load initial data
|
||||||
|
const initialize = async () => {
|
||||||
|
try {
|
||||||
|
const isConnected = await api.ping();
|
||||||
|
if (isConnected) {
|
||||||
|
await refreshData();
|
||||||
|
} else {
|
||||||
|
setError('Failed to connect to Navidrome server');
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to initialize Navidrome:', err);
|
||||||
|
setError('Failed to initialize Navidrome connection');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
initialize();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const value: NavidromeContextType = {
|
||||||
|
// Data
|
||||||
|
albums,
|
||||||
|
artists,
|
||||||
|
playlists,
|
||||||
|
|
||||||
|
// Loading states
|
||||||
|
isLoading,
|
||||||
|
albumsLoading,
|
||||||
|
artistsLoading,
|
||||||
|
playlistsLoading,
|
||||||
|
|
||||||
|
// Error state
|
||||||
|
error,
|
||||||
|
|
||||||
|
// Methods
|
||||||
|
searchMusic,
|
||||||
|
getAlbum,
|
||||||
|
getArtist,
|
||||||
|
getPlaylist,
|
||||||
|
getAllSongs,
|
||||||
|
refreshData,
|
||||||
|
createPlaylist,
|
||||||
|
updatePlaylist,
|
||||||
|
deletePlaylist,
|
||||||
|
starItem,
|
||||||
|
unstarItem,
|
||||||
|
scrobble
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<NavidromeContext.Provider value={value}>
|
||||||
|
{children}
|
||||||
|
</NavidromeContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useNavidrome = (): NavidromeContextType => {
|
||||||
|
const context = useContext(NavidromeContext);
|
||||||
|
if (context === undefined) {
|
||||||
|
throw new Error('useNavidrome must be used within a NavidromeProvider');
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
134
app/components/album-artwork.tsx
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import Image from "next/image"
|
||||||
|
import { PlusCircledIcon } from "@radix-ui/react-icons"
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import {
|
||||||
|
ContextMenu,
|
||||||
|
ContextMenuContent,
|
||||||
|
ContextMenuItem,
|
||||||
|
ContextMenuSeparator,
|
||||||
|
ContextMenuSub,
|
||||||
|
ContextMenuSubContent,
|
||||||
|
ContextMenuSubTrigger,
|
||||||
|
ContextMenuTrigger,
|
||||||
|
} from "../../components/ui/context-menu"
|
||||||
|
|
||||||
|
import { Album } from "@/lib/navidrome"
|
||||||
|
import { useNavidrome } from "./NavidromeContext"
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useAudioPlayer } from "@/app/components/AudioPlayerContext";
|
||||||
|
import { getNavidromeAPI } from "@/lib/navidrome";
|
||||||
|
|
||||||
|
interface AlbumArtworkProps extends React.HTMLAttributes<HTMLDivElement> {
|
||||||
|
album: Album
|
||||||
|
aspectRatio?: "portrait" | "square"
|
||||||
|
width?: number
|
||||||
|
height?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AlbumArtwork({
|
||||||
|
album,
|
||||||
|
aspectRatio = "portrait",
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: AlbumArtworkProps) {
|
||||||
|
const router = useRouter();
|
||||||
|
const { addAlbumToQueue } = useAudioPlayer();
|
||||||
|
const { playlists, starItem, unstarItem } = useNavidrome();
|
||||||
|
const api = getNavidromeAPI();
|
||||||
|
|
||||||
|
const handleClick = () => {
|
||||||
|
router.push(`/album/${album.id}`);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddToQueue = () => {
|
||||||
|
addAlbumToQueue(album.id);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleStar = () => {
|
||||||
|
if (album.starred) {
|
||||||
|
unstarItem(album.id, 'album');
|
||||||
|
} else {
|
||||||
|
starItem(album.id, 'album');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Get cover art URL with proper fallback
|
||||||
|
const coverArtUrl = album.coverArt
|
||||||
|
? api.getCoverArtUrl(album.coverArt, 300)
|
||||||
|
: '/default-user.jpg';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn("space-y-3", className)} {...props}>
|
||||||
|
<ContextMenu>
|
||||||
|
<ContextMenuTrigger>
|
||||||
|
<div onClick={handleClick} className="overflow-hidden rounded-md">
|
||||||
|
<Image
|
||||||
|
src={coverArtUrl}
|
||||||
|
alt={album.name}
|
||||||
|
width={width}
|
||||||
|
height={height}
|
||||||
|
|
||||||
|
className={cn(
|
||||||
|
"h-auto w-auto object-cover transition-all hover:scale-105",
|
||||||
|
aspectRatio === "portrait" ? "aspect-[3/4]" : "aspect-square"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</ContextMenuTrigger>
|
||||||
|
<ContextMenuContent className="w-40">
|
||||||
|
<ContextMenuItem onClick={handleStar}>
|
||||||
|
{album.starred ? 'Remove from Favorites' : 'Add to Favorites'}
|
||||||
|
</ContextMenuItem>
|
||||||
|
<ContextMenuSub>
|
||||||
|
<ContextMenuSubTrigger>Add to Playlist</ContextMenuSubTrigger>
|
||||||
|
<ContextMenuSubContent className="w-48">
|
||||||
|
<ContextMenuItem>
|
||||||
|
<PlusCircledIcon className="mr-2 h-4 w-4" />
|
||||||
|
New Playlist
|
||||||
|
</ContextMenuItem>
|
||||||
|
<ContextMenuSeparator />
|
||||||
|
{playlists.map((playlist) => (
|
||||||
|
<ContextMenuItem key={playlist.id}>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth="2"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path d="M21 15V6M18.5 18a2.5 2.5 0 1 0 0-5 2.5 2.5 0 0 0 0 5ZM12 12H3M16 6H3M12 18H3" />
|
||||||
|
</svg>
|
||||||
|
{playlist.name}
|
||||||
|
</ContextMenuItem>
|
||||||
|
))}
|
||||||
|
</ContextMenuSubContent>
|
||||||
|
</ContextMenuSub>
|
||||||
|
<ContextMenuSeparator />
|
||||||
|
<ContextMenuItem onClick={handleAddToQueue}>Add Album to Queue</ContextMenuItem>
|
||||||
|
<ContextMenuItem>Play Next</ContextMenuItem>
|
||||||
|
<ContextMenuItem>Play Later</ContextMenuItem>
|
||||||
|
<ContextMenuSeparator />
|
||||||
|
<ContextMenuItem onClick={handleStar}>
|
||||||
|
{album.starred ? '★ Starred' : '☆ Star'}
|
||||||
|
</ContextMenuItem>
|
||||||
|
<ContextMenuItem>Share</ContextMenuItem>
|
||||||
|
</ContextMenuContent>
|
||||||
|
</ContextMenu>
|
||||||
|
<div className="space-y-1 text-sm" >
|
||||||
|
<p className="font-medium leading-none" onClick={handleClick}>{album.name}</p>
|
||||||
|
<p className="text-xs text-muted-foreground underline">
|
||||||
|
<Link href={`/artist/${album.artistId}`}>{album.artist}</Link>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
125
app/components/artist-icon.tsx
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import Image from "next/image"
|
||||||
|
import { PlusCircledIcon } from "@radix-ui/react-icons"
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import {
|
||||||
|
ContextMenu,
|
||||||
|
ContextMenuContent,
|
||||||
|
ContextMenuItem,
|
||||||
|
ContextMenuSeparator,
|
||||||
|
ContextMenuSub,
|
||||||
|
ContextMenuSubContent,
|
||||||
|
ContextMenuSubTrigger,
|
||||||
|
ContextMenuTrigger,
|
||||||
|
} from "../../components/ui/context-menu"
|
||||||
|
|
||||||
|
import { Artist } from "@/lib/navidrome"
|
||||||
|
import { useNavidrome } from "./NavidromeContext"
|
||||||
|
import { useAudioPlayer } from "@/app/components/AudioPlayerContext";
|
||||||
|
import { getNavidromeAPI } from "@/lib/navidrome";
|
||||||
|
|
||||||
|
interface ArtistIconProps extends React.HTMLAttributes<HTMLDivElement> {
|
||||||
|
artist: Artist
|
||||||
|
size?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ArtistIcon({
|
||||||
|
artist,
|
||||||
|
size = 150,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: ArtistIconProps) {
|
||||||
|
const router = useRouter();
|
||||||
|
const { addArtistToQueue } = useAudioPlayer();
|
||||||
|
const { playlists, starItem, unstarItem } = useNavidrome();
|
||||||
|
const api = getNavidromeAPI();
|
||||||
|
|
||||||
|
const handleClick = () => {
|
||||||
|
router.push(`/artist/${artist.id}`);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddToQueue = () => {
|
||||||
|
addArtistToQueue(artist.id);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleStar = () => {
|
||||||
|
if (artist.starred) {
|
||||||
|
unstarItem(artist.id, 'artist');
|
||||||
|
} else {
|
||||||
|
starItem(artist.id, 'artist');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Get cover art URL with proper fallback
|
||||||
|
const artistImageUrl = artist.coverArt
|
||||||
|
? api.getCoverArtUrl(artist.coverArt, 200)
|
||||||
|
: '/default-user.jpg';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn("space-y-3", className)} {...props}>
|
||||||
|
<ContextMenu>
|
||||||
|
<ContextMenuTrigger>
|
||||||
|
<div className={cn("overflow-hidden")} onClick={handleClick}>
|
||||||
|
<Image
|
||||||
|
src={artistImageUrl}
|
||||||
|
alt={artist.name}
|
||||||
|
width={width}
|
||||||
|
height={height}
|
||||||
|
className={cn(
|
||||||
|
"transition-all hover:scale-105"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</ContextMenuTrigger>
|
||||||
|
<ContextMenuContent className="w-40">
|
||||||
|
<ContextMenuItem onClick={handleStar}>
|
||||||
|
{artist.starred ? 'Remove from Favorites' : 'Add to Favorites'}
|
||||||
|
</ContextMenuItem>
|
||||||
|
<ContextMenuSub>
|
||||||
|
<ContextMenuSubTrigger>Add to Playlist</ContextMenuSubTrigger>
|
||||||
|
<ContextMenuSubContent className="w-48">
|
||||||
|
<ContextMenuItem>
|
||||||
|
<PlusCircledIcon className="mr-2 h-4 w-4" />
|
||||||
|
New Playlist
|
||||||
|
</ContextMenuItem>
|
||||||
|
<ContextMenuSeparator />
|
||||||
|
{playlists.map((playlist) => (
|
||||||
|
<ContextMenuItem key={playlist.id}>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth="2"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path d="M21 15V6M18.5 18a2.5 2.5 0 1 0 0-5 2.5 2.5 0 0 0 0 5ZM12 12H3M16 6H3M12 18H3" />
|
||||||
|
</svg>
|
||||||
|
{playlist.name}
|
||||||
|
</ContextMenuItem>
|
||||||
|
))}
|
||||||
|
</ContextMenuSubContent>
|
||||||
|
</ContextMenuSub>
|
||||||
|
<ContextMenuSeparator />
|
||||||
|
<ContextMenuItem onClick={handleAddToQueue}>Add All Songs to Queue</ContextMenuItem>
|
||||||
|
<ContextMenuItem>Play Next</ContextMenuItem>
|
||||||
|
<ContextMenuItem>Play Later</ContextMenuItem>
|
||||||
|
<ContextMenuSeparator />
|
||||||
|
<ContextMenuItem onClick={handleStar}>
|
||||||
|
{artist.starred ? '★ Starred' : '☆ Star'}
|
||||||
|
</ContextMenuItem>
|
||||||
|
<ContextMenuItem>Share</ContextMenuItem>
|
||||||
|
</ContextMenuContent>
|
||||||
|
</ContextMenu>
|
||||||
|
<div className="space-y-1 text-sm" onClick={handleClick}>
|
||||||
|
<p className="font-medium leading-none text-center">{artist.name}</p>
|
||||||
|
<p className="text-xs text-muted-foreground text-center">{artist.albumCount} albums</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
46
app/components/feedbackpopup.tsx
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import React, { useState, useEffect } from 'react';
|
||||||
|
|
||||||
|
const FeedbackPopup: React.FC = () => {
|
||||||
|
const [showPopup, setShowPopup] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const isFirstVisit = localStorage.getItem('isFirstVisit');
|
||||||
|
if (!isFirstVisit) {
|
||||||
|
setShowPopup(true);
|
||||||
|
localStorage.setItem('isFirstVisit', 'true');
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleClosePopup = () => {
|
||||||
|
setShowPopup(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!showPopup) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="fixed inset-0 flex justify-center items-start bg-black bg-opacity-50 z-50">
|
||||||
|
<div className="bg-border p-6 rounded-lg mt-10 text-center">
|
||||||
|
<h2 className="text-xl font-bold mb-4">We value your feedback!</h2>
|
||||||
|
<p className="mb-4">Please take a moment to fill out our feedback form.</p>
|
||||||
|
<a
|
||||||
|
href="https://forms.gle/yHaXE4jEubsKsE6f6"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="text-blue-500 underline mb-4 block"
|
||||||
|
>
|
||||||
|
Give Feedback
|
||||||
|
</a>
|
||||||
|
<button
|
||||||
|
onClick={handleClosePopup}
|
||||||
|
className="bg-blue-500 text-white px-4 py-2 rounded"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FeedbackPopup;
|
||||||
64
app/components/ihateserverside.tsx
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
import { Menu } from "@/app/components/menu";
|
||||||
|
import { Sidebar } from "@/app/components/sidebar";
|
||||||
|
import { useNavidrome } from "@/app/components/NavidromeContext";
|
||||||
|
import { AudioPlayer } from "./AudioPlayer";
|
||||||
|
import { Toaster } from "@/components/ui/toaster"
|
||||||
|
|
||||||
|
interface IhateserversideProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Ihateserverside: React.FC<IhateserversideProps> = ({ children }) => {
|
||||||
|
const [isSidebarVisible, setIsSidebarVisible] = useState(true);
|
||||||
|
const [isStatusBarVisible, setIsStatusBarVisible] = useState(true);
|
||||||
|
const [isSidebarHidden, setIsSidebarHidden] = useState(false);
|
||||||
|
const { playlists } = useNavidrome();
|
||||||
|
|
||||||
|
const handleTransitionEnd = () => {
|
||||||
|
if (!isSidebarVisible) {
|
||||||
|
setIsSidebarHidden(true); // This will fully hide the sidebar after transition
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<div className="hidden md:flex md:flex-col md:h-screen">
|
||||||
|
{/* Top Menu */}
|
||||||
|
<div className="sticky top-0 z-10 bg-background border-b">
|
||||||
|
<Menu
|
||||||
|
toggleSidebar={() => setIsSidebarVisible(!isSidebarVisible)}
|
||||||
|
isSidebarVisible={isSidebarVisible}
|
||||||
|
toggleStatusBar={() => setIsStatusBarVisible(!isStatusBarVisible)}
|
||||||
|
isStatusBarVisible={isStatusBarVisible}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Main Content Area */}
|
||||||
|
<div className="flex-1 flex overflow-hidden">
|
||||||
|
{isSidebarVisible && (
|
||||||
|
<div className="w-64 flex-shrink-0">
|
||||||
|
<Sidebar
|
||||||
|
playlists={playlists}
|
||||||
|
className="h-full overflow-y-auto"
|
||||||
|
onTransitionEnd={handleTransitionEnd}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className={`flex-1 overflow-y-auto ${isStatusBarVisible ? 'pb-24' : ''}`}>
|
||||||
|
<div>{children}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Audio Player */}
|
||||||
|
{isStatusBarVisible && (
|
||||||
|
<div className="fixed bottom-0 left-0 right-0 z-50 bg-background">
|
||||||
|
<AudioPlayer />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<Toaster />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Ihateserverside;
|
||||||
18
app/components/loading.tsx
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
const Loading: React.FC = () => {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className="flex items-center justify-center min-h-screen">
|
||||||
|
<div className="text-center">
|
||||||
|
<div className="loader ease-linear rounded-full border-4 border-t-4 border-gray-200 h-12 w-12 mb-4"></div>
|
||||||
|
<p>Loading...</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Loading;
|
||||||
294
app/components/menu.tsx
Normal file
@@ -0,0 +1,294 @@
|
|||||||
|
import { useCallback } from "react";
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
import Image from "next/image";
|
||||||
|
import { Github, Mail } from "lucide-react"
|
||||||
|
import {
|
||||||
|
Menubar,
|
||||||
|
MenubarCheckboxItem,
|
||||||
|
MenubarContent,
|
||||||
|
MenubarLabel,
|
||||||
|
MenubarItem,
|
||||||
|
MenubarMenu,
|
||||||
|
MenubarSeparator,
|
||||||
|
MenubarShortcut,
|
||||||
|
MenubarSub,
|
||||||
|
MenubarSubContent,
|
||||||
|
MenubarSubTrigger,
|
||||||
|
MenubarTrigger,
|
||||||
|
} from "@/components/ui/menubar"
|
||||||
|
import { useState, useEffect } from "react"
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
|
import { Separator } from '@/components/ui/separator';
|
||||||
|
import { useNavidrome } from "./NavidromeContext";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
DialogTrigger,
|
||||||
|
} from "@/components/ui/dialog"
|
||||||
|
import { Input } from "@/components/ui/input"
|
||||||
|
import { Label } from "@/components/ui/label"
|
||||||
|
|
||||||
|
interface MenuProps {
|
||||||
|
toggleSidebar: () => void;
|
||||||
|
isSidebarVisible: boolean;
|
||||||
|
toggleStatusBar: () => void;
|
||||||
|
isStatusBarVisible: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Menu({ toggleSidebar, isSidebarVisible, toggleStatusBar, isStatusBarVisible }: MenuProps) {
|
||||||
|
const [isFullScreen, setIsFullScreen] = useState(false)
|
||||||
|
const router = useRouter();
|
||||||
|
const [open, setOpen] = useState(false);
|
||||||
|
const { isConnected } = useNavidrome();
|
||||||
|
|
||||||
|
// For this demo, we'll show connection status instead of user auth
|
||||||
|
const connectionStatus = isConnected ? "Connected to Navidrome" : "Not connected";
|
||||||
|
|
||||||
|
const handleFullScreen = useCallback(() => {
|
||||||
|
if (!isFullScreen) {
|
||||||
|
document.documentElement.requestFullscreen()
|
||||||
|
} else {
|
||||||
|
document.exitFullscreen()
|
||||||
|
}
|
||||||
|
setIsFullScreen(!isFullScreen)
|
||||||
|
}, [isFullScreen])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleKeyDown = (event: KeyboardEvent) => {
|
||||||
|
if ((event.metaKey || event.ctrlKey) && event.key === ',') {
|
||||||
|
event.preventDefault();
|
||||||
|
router.push('/settings');
|
||||||
|
}
|
||||||
|
if ((event.metaKey || event.ctrlKey) && event.key === 's') {
|
||||||
|
event.preventDefault();
|
||||||
|
toggleSidebar();
|
||||||
|
}
|
||||||
|
if ((event.metaKey || event.ctrlKey) && event.key === 'f') {
|
||||||
|
event.preventDefault();
|
||||||
|
handleFullScreen();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('keydown', handleKeyDown);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('keydown', handleKeyDown);
|
||||||
|
};
|
||||||
|
}, [router, toggleSidebar, handleFullScreen]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Menubar className="rounded-none border-b border-none px-2 lg:px-4">
|
||||||
|
<MenubarMenu>
|
||||||
|
<MenubarTrigger className="font-bold">offbrand spotify</MenubarTrigger>
|
||||||
|
<MenubarContent>
|
||||||
|
<MenubarItem onClick={() => setOpen(true)}>About Music</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem onClick={() => router.push('/settings')}>
|
||||||
|
Preferences <MenubarShortcut>⌘,</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>
|
||||||
|
Hide Music <MenubarShortcut>⌘H</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem>
|
||||||
|
Hide Others <MenubarShortcut>⇧⌘H</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarShortcut />
|
||||||
|
<MenubarItem>
|
||||||
|
Quit Music <MenubarShortcut>⌘Q</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
</MenubarContent>
|
||||||
|
</MenubarMenu>
|
||||||
|
<div className="border-r-4 w-0"><p className="invisible">j</p></div>
|
||||||
|
<MenubarMenu>
|
||||||
|
<MenubarTrigger className="relative">File</MenubarTrigger>
|
||||||
|
<MenubarContent>
|
||||||
|
<MenubarSub>
|
||||||
|
<MenubarSubTrigger>New</MenubarSubTrigger>
|
||||||
|
<MenubarSubContent className="w-[230px]">
|
||||||
|
<MenubarItem>
|
||||||
|
Playlist <MenubarShortcut>⌘N</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Playlist from Selection <MenubarShortcut>⇧⌘N</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem>
|
||||||
|
Smart Playlist <MenubarShortcut>⌥⌘N</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem>Playlist Folder</MenubarItem>
|
||||||
|
<MenubarItem disabled>Genius Playlist</MenubarItem>
|
||||||
|
</MenubarSubContent>
|
||||||
|
</MenubarSub>
|
||||||
|
<MenubarItem>
|
||||||
|
Open Stream URL <MenubarShortcut>⌘U</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem>
|
||||||
|
Close Window <MenubarShortcut>⌘W</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarSub>
|
||||||
|
<MenubarSubTrigger>Library</MenubarSubTrigger>
|
||||||
|
<MenubarSubContent>
|
||||||
|
<MenubarItem>Update Cloud Library</MenubarItem>
|
||||||
|
<MenubarItem>Update Genius</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>Organize Library</MenubarItem>
|
||||||
|
<MenubarItem>Export Library</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>Import Playlist</MenubarItem>
|
||||||
|
<MenubarItem disabled>Export Playlist</MenubarItem>
|
||||||
|
<MenubarItem>Show Duplicate Items</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>Get Album Artwork</MenubarItem>
|
||||||
|
<MenubarItem disabled>Get Track Names</MenubarItem>
|
||||||
|
</MenubarSubContent>
|
||||||
|
</MenubarSub>
|
||||||
|
<MenubarItem>
|
||||||
|
Import <MenubarShortcut>⌘O</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem disabled>Burn Playlist to Disc</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>
|
||||||
|
Show in Finder <MenubarShortcut>⇧⌘R</MenubarShortcut>{" "}
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem>Convert</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>Page Setup</MenubarItem>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Print <MenubarShortcut>⌘P</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
</MenubarContent>
|
||||||
|
</MenubarMenu>
|
||||||
|
<MenubarMenu>
|
||||||
|
<MenubarTrigger>Edit</MenubarTrigger>
|
||||||
|
<MenubarContent>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Undo <MenubarShortcut>⌘Z</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Redo <MenubarShortcut>⇧⌘Z</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Cut <MenubarShortcut>⌘X</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Copy <MenubarShortcut>⌘C</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Paste <MenubarShortcut>⌘V</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>
|
||||||
|
Select All <MenubarShortcut>⌘A</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem disabled>
|
||||||
|
Deselect All <MenubarShortcut>⇧⌘A</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem>
|
||||||
|
Smart Dictation{" "}
|
||||||
|
<MenubarShortcut>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth="2"
|
||||||
|
className="h-4 w-4"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path d="m12 8-9.04 9.06a2.82 2.82 0 1 0 3.98 3.98L16 12" />
|
||||||
|
<circle cx="17" cy="7" r="5" />
|
||||||
|
</svg>
|
||||||
|
</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem>
|
||||||
|
Emoji & Symbols{" "}
|
||||||
|
<MenubarShortcut>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth="2"
|
||||||
|
className="h-4 w-4"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<circle cx="12" cy="12" r="10" />
|
||||||
|
<path d="M2 12h20M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10z" />
|
||||||
|
</svg>
|
||||||
|
</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
</MenubarContent>
|
||||||
|
</MenubarMenu>
|
||||||
|
<MenubarMenu>
|
||||||
|
<MenubarTrigger>View</MenubarTrigger>
|
||||||
|
<MenubarContent>
|
||||||
|
<MenubarCheckboxItem disabled>Show Playing Next</MenubarCheckboxItem>
|
||||||
|
<MenubarCheckboxItem disabled>Show Lyrics</MenubarCheckboxItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem inset onClick={toggleStatusBar}>
|
||||||
|
{isStatusBarVisible ? "Hide Status Bar" : "Show Status Bar"}
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem inset onClick={toggleSidebar}>
|
||||||
|
{isSidebarVisible ? "Hide Sidebar" : "Show Sidebar"}
|
||||||
|
<MenubarShortcut>⌘S</MenubarShortcut>
|
||||||
|
</MenubarItem>
|
||||||
|
<MenubarItem inset onClick={handleFullScreen}>
|
||||||
|
{isFullScreen ? "Exit Full Screen" : "Enter Full Screen"}
|
||||||
|
</MenubarItem>
|
||||||
|
</MenubarContent>
|
||||||
|
</MenubarMenu>
|
||||||
|
<MenubarMenu>
|
||||||
|
<MenubarTrigger className="hidden md:block">Account</MenubarTrigger>
|
||||||
|
<MenubarContent forceMount>
|
||||||
|
<MenubarLabel>Server Status</MenubarLabel>
|
||||||
|
<MenubarItem>{connectionStatus}</MenubarItem>
|
||||||
|
<MenubarSeparator />
|
||||||
|
<MenubarItem onClick={() => router.push('/settings')}>
|
||||||
|
Settings
|
||||||
|
</MenubarItem>
|
||||||
|
</MenubarContent>
|
||||||
|
</MenubarMenu>
|
||||||
|
</Menubar>
|
||||||
|
|
||||||
|
<Dialog open={open} onOpenChange={setOpen}>
|
||||||
|
<DialogContent className="sm:max-w-[425px]">
|
||||||
|
<DialogHeader>
|
||||||
|
</DialogHeader>
|
||||||
|
<div className="grid gap-4 py-4">
|
||||||
|
<div>
|
||||||
|
<Image
|
||||||
|
src="/splash.png"
|
||||||
|
alt="music"
|
||||||
|
width={400}
|
||||||
|
height={400}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<Separator />
|
||||||
|
<p>
|
||||||
|
a music player that doesn't (yet) play music
|
||||||
|
</p>
|
||||||
|
<div className="flex space-x-4">
|
||||||
|
<a href="https://github.com/sillyangel/project-still" target="_blank" rel="noreferrer">
|
||||||
|
<Github />
|
||||||
|
</a>
|
||||||
|
<a href="mailto:angel@sillyangel.xyz">
|
||||||
|
<Mail />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
178
app/components/sidebar.tsx
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState, useEffect } from 'react';
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
import { usePathname } from 'next/navigation';
|
||||||
|
import { Button } from "../../components/ui/button";
|
||||||
|
import { ScrollArea } from "../../components/ui/scroll-area";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { Playlist } from "@/lib/navidrome";
|
||||||
|
|
||||||
|
interface SidebarProps extends React.HTMLAttributes<HTMLDivElement> {
|
||||||
|
playlists: Playlist[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Sidebar({ className, playlists }: SidebarProps) {
|
||||||
|
const isRoot = usePathname() === "/";
|
||||||
|
const isBrowse = usePathname() === "/browse";
|
||||||
|
const isAlbums = usePathname() === "/library/albums";
|
||||||
|
const isArtists = usePathname() === "/library/artists";
|
||||||
|
const isQueue = usePathname() === "/queue";
|
||||||
|
const isHistory = usePathname() === "/history";
|
||||||
|
const isSongs = usePathname() === "/library/songs"; const isPlaylists = usePathname() === "/library/playlists";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn("pb-6", className)}>
|
||||||
|
<div className="space-y-4 py-4">
|
||||||
|
<div className="px-3 py-2">
|
||||||
|
<p className="mb-2 px-4 text-lg font-semibold tracking-tight">
|
||||||
|
Discover
|
||||||
|
</p>
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Link href="/">
|
||||||
|
<Button variant={isRoot ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
>
|
||||||
|
<circle cx="12" cy="12" r="10" />
|
||||||
|
<polygon points="10 8 16 12 10 16 10 8" />
|
||||||
|
</svg>
|
||||||
|
Listen Now
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Link href="/browse">
|
||||||
|
<Button variant={isBrowse ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
>
|
||||||
|
<rect width="7" height="7" x="3" y="3" rx="1" />
|
||||||
|
<rect width="7" height="7" x="14" y="3" rx="1" />
|
||||||
|
<rect width="7" height="7" x="14" y="14" rx="1" />
|
||||||
|
<rect width="7" height="7" x="3" y="14" rx="1" />
|
||||||
|
</svg>
|
||||||
|
Browse
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Link href="/queue">
|
||||||
|
<Button variant={isQueue ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
>
|
||||||
|
<path d="M3 6h18M3 12h18M3 18h18" />
|
||||||
|
</svg>
|
||||||
|
Queue
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div className="px-3 py-2">
|
||||||
|
<p className="mb-2 px-4 text-lg font-semibold tracking-tight">
|
||||||
|
Library
|
||||||
|
</p>
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Link href="/library/playlists">
|
||||||
|
<Button variant={isPlaylists ? "secondary" : "ghost"} className="w-full justify-start mb-1">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
>
|
||||||
|
<path d="M21 15V6" />
|
||||||
|
<path d="M18.5 18a2.5 2.5 0 1 0 0-5 2.5 2.5 0 0 0 0 5Z" />
|
||||||
|
<path d="M12 12H3" />
|
||||||
|
<path d="M16 6H3" />
|
||||||
|
<path d="M12 18H3" />
|
||||||
|
</svg>
|
||||||
|
Playlists
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Link href="/library/songs">
|
||||||
|
<Button variant={isSongs ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg className="mr-2 h-4 w-4" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||||||
|
<circle cx="8" cy="18" r="4" />
|
||||||
|
<path d="M12 18V2l7 4" />
|
||||||
|
</svg>
|
||||||
|
Songs
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Link href="/library/artists">
|
||||||
|
<Button variant={isArtists ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg className="mr-2 h-4 w-4" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round" >
|
||||||
|
<path d="m12 8-9.04 9.06a2.82 2.82 0 1 0 3.98 3.98L16 12" />
|
||||||
|
<circle cx="17" cy="7" r="5" />
|
||||||
|
</svg>
|
||||||
|
Artists
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Link href="/library/albums">
|
||||||
|
<Button variant={isAlbums ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
>
|
||||||
|
<path d="m16 6 4 14" />
|
||||||
|
<path d="M12 6v14" />
|
||||||
|
<path d="M8 8v12" />
|
||||||
|
<path d="M4 4v16" />
|
||||||
|
</svg>
|
||||||
|
Albums
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Link href="/history">
|
||||||
|
<Button variant={isHistory ? "secondary" : "ghost"} className="w-full justify-start mb-2">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="2"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
className="mr-2 h-4 w-4"
|
||||||
|
>
|
||||||
|
<path d="M12 2C6.48 2 2 6.48 2 12c0 5.52 4.48 10 10 10 5.52 0 10-4.48 10-10 0-5.52-4.48-10-10-10Z" />
|
||||||
|
<path d="M12 8v4l4 2" />
|
||||||
|
</svg>
|
||||||
|
History
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
BIN
app/fonts/GeistMonoVF.woff
Normal file
BIN
app/fonts/GeistVF.woff
Normal file
120
app/globals.css
Normal file
@@ -0,0 +1,120 @@
|
|||||||
|
@tailwind base;
|
||||||
|
@tailwind components;
|
||||||
|
@tailwind utilities;
|
||||||
|
|
||||||
|
body {
|
||||||
|
font-family: Arial, Helvetica, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
@layer utilities {
|
||||||
|
.text-balance {
|
||||||
|
text-wrap: balance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@layer base {
|
||||||
|
:root {
|
||||||
|
--background: 0 0% 100%;
|
||||||
|
--foreground: 240 10% 3.9%;
|
||||||
|
--card: 0 0% 100%;
|
||||||
|
--card-foreground: 240 10% 3.9%;
|
||||||
|
--popover: 0 0% 100%;
|
||||||
|
--popover-foreground: 240 10% 3.9%;
|
||||||
|
--primary: 221.2 83.2% 53.3%;
|
||||||
|
--primary-foreground: 0 0% 98%;
|
||||||
|
--secondary: 240 4.8% 95.9%;
|
||||||
|
--secondary-foreground: 240 5.9% 10%;
|
||||||
|
--muted: 240 4.8% 95.9%;
|
||||||
|
--muted-foreground: 240 3.8% 46.1%;
|
||||||
|
--accent: 240 4.8% 95.9%;
|
||||||
|
--accent-foreground: 240 5.9% 10%;
|
||||||
|
--destructive: 0 84.2% 60.2%;
|
||||||
|
--destructive-foreground: 0 0% 98%;
|
||||||
|
--border: 240 5.9% 90%;
|
||||||
|
--input: 240 5.9% 90%;
|
||||||
|
--ring: 240 5.9% 10%;
|
||||||
|
--radius: 0.5rem;
|
||||||
|
--chart-1: 12 76% 61%;
|
||||||
|
--chart-2: 173 58% 39%;
|
||||||
|
--chart-3: 197 37% 24%;
|
||||||
|
--chart-4: 43 74% 66%;
|
||||||
|
--chart-5: 27 87% 67%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dark {
|
||||||
|
--background: 240 10% 3.9%;
|
||||||
|
--foreground: 0 0% 98%;
|
||||||
|
--card: 240 10% 3.9%;
|
||||||
|
--card-foreground: 0 0% 98%;
|
||||||
|
--popover: 240 10% 3.9%;
|
||||||
|
--popover-foreground: 0 0% 98%;
|
||||||
|
--primary: 217.2 91.2% 59.8%;
|
||||||
|
--primary-foreground: 222.2 47.4% 11.2%;
|
||||||
|
--secondary: 217.2 32.6% 17.5%;
|
||||||
|
--secondary-foreground: 0 0% 98%;
|
||||||
|
--muted: 217.2 32.6% 17.5%;
|
||||||
|
--muted-foreground: 215 20.2% 65.1%;
|
||||||
|
--accent: 217.2 32.6% 17.5%;
|
||||||
|
--accent-foreground: 0 0% 98%;
|
||||||
|
--destructive: 0 62.8% 30.6%;
|
||||||
|
--destructive-foreground: 0 0% 98%;
|
||||||
|
--border: 217.2 32.6% 17.5%;
|
||||||
|
--input: 217.2 32.6% 17.5%;
|
||||||
|
--ring: 224.3 76.3% 48%;
|
||||||
|
--chart-1: 220 70% 50%;
|
||||||
|
--chart-2: 160 60% 45%;
|
||||||
|
--chart-3: 30 80% 55%;
|
||||||
|
--chart-4: 280 65% 60%;
|
||||||
|
--chart-5: 340 75% 55%;
|
||||||
|
--hover: 240 27% 11%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
:focus-visible { outline-color: var(rgb(59 130 246)); }
|
||||||
|
::selection { background-color: var(rgb(59 130 246)); }
|
||||||
|
::marker { color: var(rgb(59 130 246)); }
|
||||||
|
|
||||||
|
|
||||||
|
::selection {
|
||||||
|
background: var(--primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
@layer base {
|
||||||
|
* {
|
||||||
|
@apply border-border;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
@apply bg-background text-foreground;
|
||||||
|
}
|
||||||
|
h1 {
|
||||||
|
@apply text-2xl;
|
||||||
|
@apply font-black;
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
@apply text-xl;
|
||||||
|
@apply font-black;
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
@apply text-lg;
|
||||||
|
@apply font-black;
|
||||||
|
}
|
||||||
|
h4 {
|
||||||
|
@apply text-base;
|
||||||
|
@apply font-black;
|
||||||
|
}
|
||||||
|
h5 {
|
||||||
|
@apply text-sm;
|
||||||
|
@apply font-black;
|
||||||
|
}
|
||||||
|
h6 {
|
||||||
|
@apply text-xs;
|
||||||
|
@apply font-black;
|
||||||
|
}
|
||||||
|
ul {
|
||||||
|
@apply list-disc;
|
||||||
|
@apply ml-9;
|
||||||
|
}
|
||||||
|
}
|
||||||
70
app/layout.tsx
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
import { SpeedInsights } from "@vercel/speed-insights/next";
|
||||||
|
import React from 'react';
|
||||||
|
import { Analytics } from "@vercel/analytics/react";
|
||||||
|
import localFont from "next/font/local";
|
||||||
|
import "./globals.css";
|
||||||
|
import { AudioPlayerProvider } from "./components/AudioPlayerContext";
|
||||||
|
import { NavidromeProvider } from "./components/NavidromeContext";
|
||||||
|
import { Metadata } from "next";
|
||||||
|
import type { Viewport } from 'next';
|
||||||
|
import Ihateserverside from './components/ihateserverside';
|
||||||
|
|
||||||
|
export const viewport: Viewport = {
|
||||||
|
themeColor: 'black',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const metadata: Metadata = {
|
||||||
|
title: {
|
||||||
|
template: 'offbrand spotify | %s',
|
||||||
|
default: 'offbrand spotify',
|
||||||
|
},
|
||||||
|
description: 'a very awesome music streaming service',
|
||||||
|
robots: {
|
||||||
|
index: true,
|
||||||
|
follow: true,
|
||||||
|
nocache: true,
|
||||||
|
googleBot: {
|
||||||
|
index: true,
|
||||||
|
follow: false,
|
||||||
|
noimageindex: true,
|
||||||
|
'max-video-preview': -1,
|
||||||
|
'max-image-preview': 'large',
|
||||||
|
'max-snippet': -1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const geistSans = localFont({
|
||||||
|
src: "./fonts/GeistVF.woff",
|
||||||
|
variable: "--font-geist-sans",
|
||||||
|
weight: "100 900",
|
||||||
|
});
|
||||||
|
const geistMono = localFont({
|
||||||
|
src: "./fonts/GeistMonoVF.woff",
|
||||||
|
variable: "--font-geist-mono",
|
||||||
|
weight: "100 900",
|
||||||
|
});
|
||||||
|
|
||||||
|
interface LayoutProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Layout({ children }: LayoutProps) {
|
||||||
|
|
||||||
|
|
||||||
|
return (
|
||||||
|
<html lang="en">
|
||||||
|
<body className={`${geistSans.variable} ${geistMono.variable} antialiase dark bg-background`}>
|
||||||
|
<NavidromeProvider>
|
||||||
|
<AudioPlayerProvider>
|
||||||
|
<SpeedInsights />
|
||||||
|
<Analytics />
|
||||||
|
<Ihateserverside>
|
||||||
|
{children}
|
||||||
|
</Ihateserverside>
|
||||||
|
</AudioPlayerProvider>
|
||||||
|
</NavidromeProvider>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
);
|
||||||
|
}
|
||||||
65
app/library/albums/page.tsx
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { ScrollArea, ScrollBar } from "@/components/ui/scroll-area";
|
||||||
|
import { useState, useEffect } from 'react';
|
||||||
|
import { Separator } from "@/components/ui/separator";
|
||||||
|
import { Tabs, TabsContent } from "@/components/ui/tabs";
|
||||||
|
import { AlbumArtwork } from '@/app/components/album-artwork';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import { Album } from '@/lib/navidrome';
|
||||||
|
import Loading from '@/app/components/loading';
|
||||||
|
|
||||||
|
export default function Albumpage() {
|
||||||
|
const { albums, isLoading } = useNavidrome();
|
||||||
|
const [sortedAlbums, setSortedAlbums] = useState<Album[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (albums.length > 0) {
|
||||||
|
// Sort albums alphabetically by name
|
||||||
|
const sorted = [...albums].sort((a, b) => a.name.localeCompare(b.name));
|
||||||
|
setSortedAlbums(sorted);
|
||||||
|
}
|
||||||
|
}, [albums]);
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<Tabs defaultValue="music" className="h-full space-y-6">
|
||||||
|
<TabsContent value="music" className="border-none p-0 outline-none">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Albums
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
All albums in your music library ({sortedAlbums.length} albums)
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative">
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5 2xl:grid-cols-6 gap-4 pb-4">
|
||||||
|
{sortedAlbums.map((album) => (
|
||||||
|
<AlbumArtwork
|
||||||
|
key={album.id}
|
||||||
|
album={album}
|
||||||
|
className="w-full"
|
||||||
|
aspectRatio="square"
|
||||||
|
width={200}
|
||||||
|
height={200}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
63
app/library/artists/page.tsx
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import { ScrollArea, ScrollBar } from "@/components/ui/scroll-area";
|
||||||
|
import { useState, useEffect } from 'react';
|
||||||
|
import { Separator } from "@/components/ui/separator";
|
||||||
|
import { Tabs, TabsContent } from "@/components/ui/tabs";
|
||||||
|
import { ArtistIcon } from '@/app/components/artist-icon';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import { Artist } from '@/lib/navidrome';
|
||||||
|
import Loading from '@/app/components/loading';
|
||||||
|
|
||||||
|
export default function ArtistPage() {
|
||||||
|
const { artists, isLoading } = useNavidrome();
|
||||||
|
const [sortedArtists, setSortedArtists] = useState<Artist[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (artists.length > 0) {
|
||||||
|
// Sort artists alphabetically by name
|
||||||
|
const sorted = [...artists].sort((a, b) => a.name.localeCompare(b.name));
|
||||||
|
setSortedArtists(sorted);
|
||||||
|
}
|
||||||
|
}, [artists]);
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<Tabs defaultValue="music" className="h-full space-y-6">
|
||||||
|
<TabsContent value="music" className="border-none p-0 outline-none">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Artists
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
All artists in your music library ({sortedArtists.length} artists)
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative">
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5 2xl:grid-cols-6 gap-4 pb-4">
|
||||||
|
{sortedArtists.map((artist) => (
|
||||||
|
<ArtistIcon
|
||||||
|
key={artist.id}
|
||||||
|
artist={artist}
|
||||||
|
className="flex justify-center"
|
||||||
|
size={150}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
95
app/library/playlists/page.tsx
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { ScrollArea, ScrollBar } from "@/components/ui/scroll-area";
|
||||||
|
import { Separator } from "@/components/ui/separator";
|
||||||
|
import { Tabs, TabsContent } from "@/components/ui/tabs";
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import Loading from '@/app/components/loading';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { PlusCircledIcon } from "@radix-ui/react-icons";
|
||||||
|
import Link from 'next/link';
|
||||||
|
|
||||||
|
const PlaylistsPage: React.FC = () => {
|
||||||
|
const { playlists, isLoading, createPlaylist } = useNavidrome();
|
||||||
|
|
||||||
|
const handleCreatePlaylist = async () => {
|
||||||
|
const name = prompt('Enter playlist name:');
|
||||||
|
if (name) {
|
||||||
|
try {
|
||||||
|
await createPlaylist(name);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to create playlist:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<Tabs defaultValue="music" className="h-full space-y-6">
|
||||||
|
<TabsContent value="music" className="border-none p-0 outline-none">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Playlists
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
Your custom playlists ({playlists.length} playlists)
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<Button onClick={handleCreatePlaylist}>
|
||||||
|
<PlusCircledIcon className="mr-2 h-4 w-4" />
|
||||||
|
New Playlist
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative">
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4 pb-4">
|
||||||
|
{playlists.map((playlist) => (
|
||||||
|
<Link key={playlist.id} href={`/playlist/${playlist.id}`}>
|
||||||
|
<div className="p-4 rounded-lg border border-border hover:bg-accent hover:text-accent-foreground transition-colors cursor-pointer">
|
||||||
|
<div className="flex items-center space-x-4">
|
||||||
|
<div className="w-12 h-12 bg-muted rounded-md flex items-center justify-center">
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth="2"
|
||||||
|
className="h-6 w-6"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path d="M21 15V6M18.5 18a2.5 2.5 0 1 0 0-5 2.5 2.5 0 0 0 0 5ZM12 12H3M16 6H3M12 18H3" />
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div className="flex-1 min-w-0">
|
||||||
|
<p className="font-medium leading-none truncate">{playlist.name}</p>
|
||||||
|
<p className="text-sm text-muted-foreground mt-1">
|
||||||
|
{playlist.songCount} songs
|
||||||
|
</p>
|
||||||
|
{playlist.comment && (
|
||||||
|
<p className="text-xs text-muted-foreground mt-1 truncate">
|
||||||
|
{playlist.comment}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Link>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default PlaylistsPage;
|
||||||
294
app/library/songs/page.tsx
Normal file
@@ -0,0 +1,294 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import { useAudioPlayer } from '@/app/components/AudioPlayerContext';
|
||||||
|
import { Song } from '@/lib/navidrome';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { Separator } from '@/components/ui/separator';
|
||||||
|
import { ScrollArea } from '@/components/ui/scroll-area';
|
||||||
|
import { Input } from '@/components/ui/input';
|
||||||
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
||||||
|
import { Search, Play, Plus, Clock, User, Disc } from 'lucide-react';
|
||||||
|
import Loading from '@/app/components/loading';
|
||||||
|
import { getNavidromeAPI } from '@/lib/navidrome';
|
||||||
|
|
||||||
|
type SortOption = 'title' | 'artist' | 'album' | 'year' | 'duration' | 'track';
|
||||||
|
type SortDirection = 'asc' | 'desc';
|
||||||
|
|
||||||
|
export default function SongsPage() {
|
||||||
|
const { getAllSongs } = useNavidrome();
|
||||||
|
const { playTrack, addToQueue, currentTrack } = useAudioPlayer();
|
||||||
|
const [songs, setSongs] = useState<Song[]>([]);
|
||||||
|
const [filteredSongs, setFilteredSongs] = useState<Song[]>([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const [searchQuery, setSearchQuery] = useState('');
|
||||||
|
const [sortBy, setSortBy] = useState<SortOption>('title');
|
||||||
|
const [sortDirection, setSortDirection] = useState<SortDirection>('asc');
|
||||||
|
const api = getNavidromeAPI();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchSongs = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
try {
|
||||||
|
const allSongs = await getAllSongs();
|
||||||
|
setSongs(allSongs);
|
||||||
|
setFilteredSongs(allSongs);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch songs:', error);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
fetchSongs();
|
||||||
|
}, [getAllSongs]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
let filtered = songs;
|
||||||
|
|
||||||
|
// Apply search filter
|
||||||
|
if (searchQuery.trim()) {
|
||||||
|
const query = searchQuery.toLowerCase();
|
||||||
|
filtered = songs.filter(song =>
|
||||||
|
song.title.toLowerCase().includes(query) ||
|
||||||
|
song.artist.toLowerCase().includes(query) ||
|
||||||
|
song.album.toLowerCase().includes(query)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply sorting
|
||||||
|
filtered = [...filtered].sort((a, b) => {
|
||||||
|
let aValue: string | number;
|
||||||
|
let bValue: string | number;
|
||||||
|
|
||||||
|
switch (sortBy) {
|
||||||
|
case 'title':
|
||||||
|
aValue = a.title.toLowerCase();
|
||||||
|
bValue = b.title.toLowerCase();
|
||||||
|
break;
|
||||||
|
case 'artist':
|
||||||
|
aValue = a.artist.toLowerCase();
|
||||||
|
bValue = b.artist.toLowerCase();
|
||||||
|
break;
|
||||||
|
case 'album':
|
||||||
|
aValue = a.album.toLowerCase();
|
||||||
|
bValue = b.album.toLowerCase();
|
||||||
|
break;
|
||||||
|
case 'year':
|
||||||
|
aValue = a.year || 0;
|
||||||
|
bValue = b.year || 0;
|
||||||
|
break;
|
||||||
|
case 'duration':
|
||||||
|
aValue = a.duration;
|
||||||
|
bValue = b.duration;
|
||||||
|
break;
|
||||||
|
case 'track':
|
||||||
|
aValue = a.track || 0;
|
||||||
|
bValue = b.track || 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
aValue = a.title.toLowerCase();
|
||||||
|
bValue = b.title.toLowerCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sortDirection === 'asc') {
|
||||||
|
return aValue < bValue ? -1 : aValue > bValue ? 1 : 0;
|
||||||
|
} else {
|
||||||
|
return aValue > bValue ? -1 : aValue < bValue ? 1 : 0;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
setFilteredSongs(filtered);
|
||||||
|
}, [songs, searchQuery, sortBy, sortDirection]);
|
||||||
|
|
||||||
|
const handlePlaySong = (song: Song) => {
|
||||||
|
const track = {
|
||||||
|
id: song.id,
|
||||||
|
name: song.title,
|
||||||
|
url: api.getStreamUrl(song.id),
|
||||||
|
artist: song.artist,
|
||||||
|
album: song.album,
|
||||||
|
duration: song.duration,
|
||||||
|
coverArt: song.coverArt ? api.getCoverArtUrl(song.coverArt, 300) : undefined,
|
||||||
|
albumId: song.albumId,
|
||||||
|
artistId: song.artistId
|
||||||
|
};
|
||||||
|
|
||||||
|
playTrack(track);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddToQueue = (song: Song) => {
|
||||||
|
const track = {
|
||||||
|
id: song.id,
|
||||||
|
name: song.title,
|
||||||
|
url: api.getStreamUrl(song.id),
|
||||||
|
artist: song.artist,
|
||||||
|
album: song.album,
|
||||||
|
duration: song.duration,
|
||||||
|
coverArt: song.coverArt ? api.getCoverArtUrl(song.coverArt, 300) : undefined,
|
||||||
|
albumId: song.albumId,
|
||||||
|
artistId: song.artistId
|
||||||
|
};
|
||||||
|
|
||||||
|
addToQueue(track);
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDuration = (seconds: number): string => {
|
||||||
|
const minutes = Math.floor(seconds / 60);
|
||||||
|
const remainingSeconds = seconds % 60;
|
||||||
|
return `${minutes}:${remainingSeconds.toString().padStart(2, '0')}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const isCurrentlyPlaying = (song: Song): boolean => {
|
||||||
|
return currentTrack?.id === song.id;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<div className="space-y-6">
|
||||||
|
{/* Header */}
|
||||||
|
<div className="space-y-2">
|
||||||
|
<h1 className="text-3xl font-semibold tracking-tight">Songs</h1>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{filteredSongs.length} of {songs.length} songs
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Search and Filters */}
|
||||||
|
<div className="flex flex-col sm:flex-row gap-4">
|
||||||
|
<div className="relative flex-1">
|
||||||
|
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 text-muted-foreground w-4 h-4" />
|
||||||
|
<Input
|
||||||
|
placeholder="Search songs, artists, or albums..."
|
||||||
|
value={searchQuery}
|
||||||
|
onChange={(e) => setSearchQuery(e.target.value)}
|
||||||
|
className="pl-10"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Select value={sortBy} onValueChange={(value: SortOption) => setSortBy(value)}>
|
||||||
|
<SelectTrigger className="w-[140px]">
|
||||||
|
<SelectValue />
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
<SelectItem value="title">Title</SelectItem>
|
||||||
|
<SelectItem value="artist">Artist</SelectItem>
|
||||||
|
<SelectItem value="album">Album</SelectItem>
|
||||||
|
<SelectItem value="year">Year</SelectItem>
|
||||||
|
<SelectItem value="duration">Duration</SelectItem>
|
||||||
|
<SelectItem value="track">Track #</SelectItem>
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc')}
|
||||||
|
>
|
||||||
|
{sortDirection === 'asc' ? '↑' : '↓'}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Separator />
|
||||||
|
|
||||||
|
{/* Songs List */}
|
||||||
|
<ScrollArea className="h-[calc(100vh-300px)]">
|
||||||
|
{filteredSongs.length === 0 ? (
|
||||||
|
<div className="text-center py-12">
|
||||||
|
<p className="text-muted-foreground">
|
||||||
|
{searchQuery ? 'No songs found matching your search.' : 'No songs available.'}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="space-y-1">
|
||||||
|
{filteredSongs.map((song, index) => (
|
||||||
|
<div
|
||||||
|
key={song.id}
|
||||||
|
className={`group flex items-center p-3 rounded-lg hover:bg-accent/50 cursor-pointer transition-colors ${
|
||||||
|
isCurrentlyPlaying(song) ? 'bg-accent/50 border-l-4 border-primary' : ''
|
||||||
|
}`}
|
||||||
|
onClick={() => handlePlaySong(song)}
|
||||||
|
>
|
||||||
|
{/* Track Number / Play Indicator */}
|
||||||
|
<div className="w-8 text-center text-sm text-muted-foreground mr-3">
|
||||||
|
{isCurrentlyPlaying(song) ? (
|
||||||
|
<div className="w-4 h-4 mx-auto">
|
||||||
|
<div className="w-full h-full bg-primary rounded-full animate-pulse" />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<span className="group-hover:hidden">{index + 1}</span>
|
||||||
|
)}
|
||||||
|
<Play className="w-4 h-4 mx-auto hidden group-hover:block" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Album Art */}
|
||||||
|
<div className="w-12 h-12 mr-4 flex-shrink-0">
|
||||||
|
<Image
|
||||||
|
src={song.coverArt ? api.getCoverArtUrl(song.coverArt, 100) : '/default-user.jpg'}
|
||||||
|
alt={song.album}
|
||||||
|
width={48}
|
||||||
|
height={48}
|
||||||
|
className="w-full h-full object-cover rounded-md"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Song Info */}
|
||||||
|
<div className="flex-1 min-w-0 mr-4">
|
||||||
|
<div className="flex items-center gap-2 mb-1">
|
||||||
|
<p className={`font-semibold truncate ${
|
||||||
|
isCurrentlyPlaying(song) ? 'text-primary' : ''
|
||||||
|
}`}>
|
||||||
|
{song.title}
|
||||||
|
</p>
|
||||||
|
{song.year && (
|
||||||
|
<span className="text-xs text-muted-foreground bg-muted px-1.5 py-0.5 rounded">
|
||||||
|
{song.year}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center text-sm text-muted-foreground space-x-4">
|
||||||
|
<div className="flex items-center gap-1">
|
||||||
|
<User className="w-3 h-3" />
|
||||||
|
<span className="truncate">{song.artist}</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-1">
|
||||||
|
<Disc className="w-3 h-3" />
|
||||||
|
<span className="truncate">{song.album}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Duration */}
|
||||||
|
<div className="flex items-center text-sm text-muted-foreground mr-4">
|
||||||
|
<Clock className="w-3 h-3 mr-1" />
|
||||||
|
{formatDuration(song.duration)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Actions */}
|
||||||
|
<div className="flex items-center space-x-2 opacity-0 group-hover:opacity-100 transition-opacity">
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
handleAddToQueue(song);
|
||||||
|
}}
|
||||||
|
className="h-8 w-8 p-0"
|
||||||
|
>
|
||||||
|
<Plus className="w-4 h-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
44
app/manifest.ts
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import type { MetadataRoute } from 'next'
|
||||||
|
|
||||||
|
export default function manifest(): MetadataRoute.Manifest {
|
||||||
|
return {
|
||||||
|
name: 'Offbrand Spotify',
|
||||||
|
short_name: 'Offbrand',
|
||||||
|
description: 'a very offbrand spotify clone',
|
||||||
|
start_url: '/',
|
||||||
|
categories: ["music", "entertainment"],
|
||||||
|
display_override: ['window-controls-overlay'],
|
||||||
|
display: 'standalone',
|
||||||
|
background_color: '#0f0f0f',
|
||||||
|
theme_color: '#0f0f0f',
|
||||||
|
icons: [
|
||||||
|
{
|
||||||
|
src: '/favicon.ico',
|
||||||
|
type: 'image/x-icon',
|
||||||
|
sizes: '16x16 32x32'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/icon-192.png',
|
||||||
|
type: 'image/png',
|
||||||
|
sizes: '192x192'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/icon-512.png',
|
||||||
|
type: 'image/png',
|
||||||
|
sizes: '512x512'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/icon-192-maskable.png',
|
||||||
|
type: 'image/png',
|
||||||
|
sizes: '192x192',
|
||||||
|
purpose: 'maskable'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: './icon-512-maskable.png',
|
||||||
|
type: 'image/png',
|
||||||
|
sizes: '512x512',
|
||||||
|
purpose: 'maskable'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
}
|
||||||
|
}
|
||||||
118
app/page.tsx
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { ScrollArea, ScrollBar } from '../components/ui/scroll-area';
|
||||||
|
import { Separator } from '../components/ui/separator';
|
||||||
|
import { Tabs, TabsContent } from '../components/ui/tabs';
|
||||||
|
import { AlbumArtwork } from './components/album-artwork';
|
||||||
|
import { useNavidrome } from './components/NavidromeContext';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { Album } from '@/lib/navidrome';
|
||||||
|
|
||||||
|
export default function MusicPage() {
|
||||||
|
const { albums, isLoading, error } = useNavidrome();
|
||||||
|
const [recentAlbums, setRecentAlbums] = useState<Album[]>([]);
|
||||||
|
const [newestAlbums, setNewestAlbums] = useState<Album[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (albums.length > 0) {
|
||||||
|
// Split albums into recent and newest for display
|
||||||
|
const recent = albums.slice(0, Math.ceil(albums.length / 2));
|
||||||
|
const newest = albums.slice(Math.ceil(albums.length / 2));
|
||||||
|
setRecentAlbums(recent);
|
||||||
|
setNewestAlbums(newest);
|
||||||
|
}
|
||||||
|
}, [albums]);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8 flex items-center justify-center">
|
||||||
|
<div className="text-center">
|
||||||
|
<p className="text-xl font-semibold text-destructive mb-2">Connection Error</p>
|
||||||
|
<p className="text-muted-foreground">{error}</p>
|
||||||
|
<p className="text-sm text-muted-foreground mt-2">
|
||||||
|
Please check your Navidrome server configuration.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<>
|
||||||
|
<Tabs defaultValue="music" className="h-full space-y-6">
|
||||||
|
<TabsContent value="music" className="border-none p-0 outline-none">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Recently Added
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
Latest additions to your music library.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative">
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="flex space-x-4 pb-4">
|
||||||
|
{isLoading ? (
|
||||||
|
// Loading skeletons
|
||||||
|
Array.from({ length: 6 }).map((_, i) => (
|
||||||
|
<div key={i} className="w-[300px] h-[300px] bg-muted animate-pulse rounded-md" />
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
recentAlbums.map((album) => (
|
||||||
|
<AlbumArtwork
|
||||||
|
key={album.id}
|
||||||
|
album={album}
|
||||||
|
className="w-[300px]"
|
||||||
|
aspectRatio="square"
|
||||||
|
width={300}
|
||||||
|
height={300}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
<div className="mt-6 space-y-1">
|
||||||
|
<p className="text-2xl font-semibold tracking-tight">
|
||||||
|
Your Library
|
||||||
|
</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
Albums from your music collection.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-4" />
|
||||||
|
<div className="relative">
|
||||||
|
<ScrollArea>
|
||||||
|
<div className="flex space-x-4 pb-4">
|
||||||
|
{isLoading ? (
|
||||||
|
// Loading skeletons
|
||||||
|
Array.from({ length: 10 }).map((_, i) => (
|
||||||
|
<div key={i} className="w-[150px] h-[150px] bg-muted animate-pulse rounded-md" />
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
newestAlbums.map((album) => (
|
||||||
|
<AlbumArtwork
|
||||||
|
key={album.id}
|
||||||
|
album={album}
|
||||||
|
className="w-[150px]"
|
||||||
|
aspectRatio="square"
|
||||||
|
width={150}
|
||||||
|
height={150}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<ScrollBar orientation="horizontal" />
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
</>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
157
app/playlist/[id]/page.tsx
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { useParams } from 'next/navigation';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { Playlist, Song } from '@/lib/navidrome';
|
||||||
|
import { useNavidrome } from '@/app/components/NavidromeContext';
|
||||||
|
import { useAudioPlayer } from '@/app/components/AudioPlayerContext';
|
||||||
|
import { Play, Heart, Plus } from 'lucide-react';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import Loading from "@/app/components/loading";
|
||||||
|
import { Separator } from '@/components/ui/separator';
|
||||||
|
|
||||||
|
export default function PlaylistPage() {
|
||||||
|
const { id } = useParams();
|
||||||
|
const [playlist, setPlaylist] = useState<Playlist | null>(null);
|
||||||
|
const [tracklist, setTracklist] = useState<Song[]>([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const { getPlaylist } = useNavidrome();
|
||||||
|
const { playTrack, addToQueue } = useAudioPlayer();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchPlaylist = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
console.log(`Fetching playlist with id: ${id}`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const playlistData = await getPlaylist(id as string);
|
||||||
|
setPlaylist(playlistData.playlist);
|
||||||
|
setTracklist(playlistData.songs);
|
||||||
|
console.log(`Playlist found: ${playlistData.playlist.name}`);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch playlist:', error);
|
||||||
|
}
|
||||||
|
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (id) {
|
||||||
|
fetchPlaylist();
|
||||||
|
}
|
||||||
|
}, [id, getPlaylist]);
|
||||||
|
|
||||||
|
const handlePlayClick = (song: Song) => {
|
||||||
|
const track = {
|
||||||
|
id: song.id,
|
||||||
|
name: song.title,
|
||||||
|
url: '', // Will be set by the context
|
||||||
|
artist: song.artist,
|
||||||
|
album: song.album,
|
||||||
|
duration: song.duration,
|
||||||
|
coverArt: song.coverArt,
|
||||||
|
albumId: song.albumId,
|
||||||
|
artistId: song.artistId
|
||||||
|
};
|
||||||
|
playTrack(track);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddToQueue = (song: Song) => {
|
||||||
|
const track = {
|
||||||
|
id: song.id,
|
||||||
|
name: song.title,
|
||||||
|
url: '', // Will be set by the context
|
||||||
|
artist: song.artist,
|
||||||
|
album: song.album,
|
||||||
|
duration: song.duration,
|
||||||
|
coverArt: song.coverArt,
|
||||||
|
albumId: song.albumId,
|
||||||
|
artistId: song.artistId
|
||||||
|
};
|
||||||
|
addToQueue(track);
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDuration = (seconds: number) => {
|
||||||
|
const minutes = Math.floor(seconds / 60);
|
||||||
|
const secs = Math.floor(seconds % 60).toString().padStart(2, '0');
|
||||||
|
return `${minutes}:${secs}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <Loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!playlist) {
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<div className="text-center">
|
||||||
|
<h2 className="text-2xl font-bold">Playlist not found</h2>
|
||||||
|
<p className="text-muted-foreground">The playlist you're looking for doesn't exist.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full px-4 py-6 lg:px-8">
|
||||||
|
<div className="space-y-4">
|
||||||
|
<div className="flex items-start gap-6">
|
||||||
|
<div className="w-[300px] h-[300px] bg-muted rounded-md flex items-center justify-center">
|
||||||
|
<Play className="h-16 w-16 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<div className="flex items-center space-x-4">
|
||||||
|
<p className="text-3xl font-semibold tracking-tight">{playlist.name}</p>
|
||||||
|
</div>
|
||||||
|
{playlist.comment && (
|
||||||
|
<p className="text-lg text-muted-foreground">{playlist.comment}</p>
|
||||||
|
)}
|
||||||
|
<div className="text-sm text-muted-foreground">
|
||||||
|
<p>{playlist.songCount} songs • {formatDuration(playlist.duration || 0)}</p>
|
||||||
|
{playlist.public !== undefined && (
|
||||||
|
<p>{playlist.public ? 'Public' : 'Private'} playlist</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Separator />
|
||||||
|
{tracklist.length > 0 ? (
|
||||||
|
tracklist.map((song, index) => (
|
||||||
|
<div key={song.id} className="py-2 flex justify-between items-center hover:bg-hover rounded-lg cursor-pointer" onClick={() => handlePlayClick(song)}>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<div className="mr-2 w-6 text-right">{index + 1}</div>
|
||||||
|
<div>
|
||||||
|
<p className="font-semibold text-lg flex items-center">
|
||||||
|
{song.title}
|
||||||
|
</p>
|
||||||
|
<p className="text-sm font-normal flex items-center">
|
||||||
|
<span className="text-gray-400">{song.artist}</span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center space-x-2">
|
||||||
|
<p className="text-sm mr-4">{formatDuration(song.duration)}</p>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
handleAddToQueue(song);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Plus className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
<div className="text-center py-8">
|
||||||
|
<p className="text-muted-foreground">This playlist is empty</p>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
102
app/queue/page.tsx
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
import Image from 'next/image';
|
||||||
|
import { useAudioPlayer } from '@/app/components/AudioPlayerContext';
|
||||||
|
|
||||||
|
const QueuePage: React.FC = () => {
|
||||||
|
const { queue, currentTrack, removeTrackFromQueue, clearQueue, skipToTrackInQueue } = useAudioPlayer();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="p-4">
|
||||||
|
<div className="flex items-center justify-between mb-6">
|
||||||
|
<div>
|
||||||
|
<h1 className="text-2xl font-semibold mb-1">Queue</h1>
|
||||||
|
<p className="text-sm text-muted-foreground">Click on a track to skip to it</p>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={clearQueue}
|
||||||
|
className="px-4 py-2 bg-destructive text-destructive-foreground rounded-md hover:bg-destructive/90"
|
||||||
|
disabled={queue.length === 0}
|
||||||
|
>
|
||||||
|
Clear queue
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Currently Playing */}
|
||||||
|
{currentTrack && (
|
||||||
|
<div className="mb-6">
|
||||||
|
<h2 className="text-lg font-semibold mb-3">Now Playing</h2>
|
||||||
|
<div className="p-4 bg-accent/50 rounded-lg border-l-4 border-primary">
|
||||||
|
<div className="flex items-center">
|
||||||
|
<Image
|
||||||
|
src={currentTrack.coverArt || '/default-user.jpg'}
|
||||||
|
alt={currentTrack.name}
|
||||||
|
width={60}
|
||||||
|
height={60}
|
||||||
|
className="rounded-md mr-4"
|
||||||
|
/>
|
||||||
|
<div className="flex-1">
|
||||||
|
<p className="font-semibold text-lg">{currentTrack.name}</p>
|
||||||
|
<p className="text-sm text-muted-foreground">{currentTrack.artist}</p>
|
||||||
|
<p className="text-xs text-muted-foreground">{currentTrack.album}</p>
|
||||||
|
</div>
|
||||||
|
<div className="text-sm text-muted-foreground">
|
||||||
|
{Math.floor(currentTrack.duration / 60)}:{(currentTrack.duration % 60).toString().padStart(2, '0')}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Queue */}
|
||||||
|
<div>
|
||||||
|
<h2 className="text-lg font-semibold mb-3">Up Next</h2>
|
||||||
|
{queue.length === 0 ? (
|
||||||
|
<p className="text-muted-foreground">No tracks in the queue</p>
|
||||||
|
) : (
|
||||||
|
<div className="space-y-2">
|
||||||
|
{queue.map((track, index) => (
|
||||||
|
<div
|
||||||
|
key={`${track.id}-${index}`}
|
||||||
|
className="flex items-center p-3 rounded-lg hover:bg-accent/50 cursor-pointer group"
|
||||||
|
onClick={() => skipToTrackInQueue(index)}
|
||||||
|
>
|
||||||
|
<div className="w-8 text-center text-sm text-muted-foreground mr-3">
|
||||||
|
{index + 1}
|
||||||
|
</div>
|
||||||
|
<Image
|
||||||
|
src={track.coverArt || '/default-user.jpg'}
|
||||||
|
alt={track.name}
|
||||||
|
width={50}
|
||||||
|
height={50}
|
||||||
|
className="rounded-md mr-4"
|
||||||
|
/>
|
||||||
|
<div className="flex-1 min-w-0">
|
||||||
|
<p className="font-semibold truncate">{track.name}</p>
|
||||||
|
<p className="text-sm text-muted-foreground truncate">{track.artist}</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center space-x-4">
|
||||||
|
<div className="text-sm text-muted-foreground">
|
||||||
|
{Math.floor(track.duration / 60)}:{(track.duration % 60).toString().padStart(2, '0')}
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
removeTrackFromQueue(index);
|
||||||
|
}}
|
||||||
|
className="opacity-0 group-hover:opacity-100 px-3 py-1 text-sm bg-destructive text-destructive-foreground rounded hover:bg-destructive/90 transition-all"
|
||||||
|
>
|
||||||
|
Remove
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default QueuePage;
|
||||||
22
app/settings/page.tsx
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Label } from '@/components/ui/label';
|
||||||
|
import { Select, SelectTrigger, SelectContent, SelectItem, SelectValue } from '@/components/ui/select';
|
||||||
|
|
||||||
|
const SettingsPage = () => {
|
||||||
|
return (
|
||||||
|
<div className="container mx-auto p-4">
|
||||||
|
<Label>Theme</Label>
|
||||||
|
<Select>
|
||||||
|
<SelectTrigger>
|
||||||
|
<SelectValue>Light</SelectValue>
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
<SelectItem value="light">Light</SelectItem>
|
||||||
|
<SelectItem value="dark">Dark</SelectItem>
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SettingsPage;
|
||||||
12
colors.txt
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
border is located at *
|
||||||
|
hsl(214.3deg 3.81% 25%)
|
||||||
|
|
||||||
|
background color in sticky top-0
|
||||||
|
hsl(0deg 0% 5.86%)
|
||||||
|
|
||||||
|
.text muted foreground
|
||||||
|
hsl(0deg 0% 58.47%)
|
||||||
|
|
||||||
|
change in body color
|
||||||
|
hsl(0 0% 100%)
|
||||||
|
|
||||||
21
components.json
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://ui.shadcn.com/schema.json",
|
||||||
|
"style": "new-york",
|
||||||
|
"rsc": true,
|
||||||
|
"tsx": true,
|
||||||
|
"tailwind": {
|
||||||
|
"config": "tailwind.config.ts",
|
||||||
|
"css": "app/globals.css",
|
||||||
|
"baseColor": "slate",
|
||||||
|
"cssVariables": true,
|
||||||
|
"prefix": ""
|
||||||
|
},
|
||||||
|
"aliases": {
|
||||||
|
"components": "@/components",
|
||||||
|
"utils": "@/lib/utils",
|
||||||
|
"ui": "@/components/ui",
|
||||||
|
"lib": "@/lib",
|
||||||
|
"hooks": "@/hooks"
|
||||||
|
},
|
||||||
|
"iconLibrary": "lucide"
|
||||||
|
}
|
||||||
59
components/ui/alert.tsx
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const alertVariants = cva(
|
||||||
|
"relative w-full rounded-lg border px-4 py-3 text-sm [&>svg+div]:translate-y-[-3px] [&>svg]:absolute [&>svg]:left-4 [&>svg]:top-4 [&>svg]:text-foreground [&>svg~*]:pl-7",
|
||||||
|
{
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
default: "bg-background text-foreground",
|
||||||
|
destructive:
|
||||||
|
"border-destructive/50 text-destructive dark:border-destructive [&>svg]:text-destructive",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
variant: "default",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const Alert = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement> & VariantProps<typeof alertVariants>
|
||||||
|
>(({ className, variant, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
role="alert"
|
||||||
|
className={cn(alertVariants({ variant }), className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Alert.displayName = "Alert"
|
||||||
|
|
||||||
|
const AlertTitle = React.forwardRef<
|
||||||
|
HTMLParagraphElement,
|
||||||
|
React.HTMLAttributes<HTMLHeadingElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<h5
|
||||||
|
ref={ref}
|
||||||
|
className={cn("mb-1 font-medium leading-none tracking-tight", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
AlertTitle.displayName = "AlertTitle"
|
||||||
|
|
||||||
|
const AlertDescription = React.forwardRef<
|
||||||
|
HTMLParagraphElement,
|
||||||
|
React.HTMLAttributes<HTMLParagraphElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm [&_p]:leading-relaxed", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
AlertDescription.displayName = "AlertDescription"
|
||||||
|
|
||||||
|
export { Alert, AlertTitle, AlertDescription }
|
||||||
50
components/ui/avatar.tsx
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as AvatarPrimitive from "@radix-ui/react-avatar"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Avatar = React.forwardRef<
|
||||||
|
React.ElementRef<typeof AvatarPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof AvatarPrimitive.Root>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<AvatarPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex h-10 w-10 shrink-0 overflow-hidden rounded-full",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Avatar.displayName = AvatarPrimitive.Root.displayName
|
||||||
|
|
||||||
|
const AvatarImage = React.forwardRef<
|
||||||
|
React.ElementRef<typeof AvatarPrimitive.Image>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof AvatarPrimitive.Image>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<AvatarPrimitive.Image
|
||||||
|
ref={ref}
|
||||||
|
className={cn("aspect-square h-full w-full", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
AvatarImage.displayName = AvatarPrimitive.Image.displayName
|
||||||
|
|
||||||
|
const AvatarFallback = React.forwardRef<
|
||||||
|
React.ElementRef<typeof AvatarPrimitive.Fallback>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof AvatarPrimitive.Fallback>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<AvatarPrimitive.Fallback
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex h-full w-full items-center justify-center rounded-full bg-muted",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
AvatarFallback.displayName = AvatarPrimitive.Fallback.displayName
|
||||||
|
|
||||||
|
export { Avatar, AvatarImage, AvatarFallback }
|
||||||
57
components/ui/button.tsx
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import { Slot } from "@radix-ui/react-slot"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const buttonVariants = cva(
|
||||||
|
"inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0",
|
||||||
|
{
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
default:
|
||||||
|
"bg-primary text-primary-foreground shadow hover:bg-primary/90",
|
||||||
|
destructive:
|
||||||
|
"bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90",
|
||||||
|
outline:
|
||||||
|
"border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground",
|
||||||
|
secondary:
|
||||||
|
"bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80",
|
||||||
|
ghost: "hover:bg-accent hover:text-accent-foreground",
|
||||||
|
link: "text-primary underline-offset-4 hover:underline",
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
default: "h-9 px-4 py-2",
|
||||||
|
sm: "h-8 rounded-md px-3 text-xs",
|
||||||
|
lg: "h-10 rounded-md px-8",
|
||||||
|
icon: "h-9 w-9",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
variant: "default",
|
||||||
|
size: "default",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
export interface ButtonProps
|
||||||
|
extends React.ButtonHTMLAttributes<HTMLButtonElement>,
|
||||||
|
VariantProps<typeof buttonVariants> {
|
||||||
|
asChild?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
|
||||||
|
({ className, variant, size, asChild = false, ...props }, ref) => {
|
||||||
|
const Comp = asChild ? Slot : "button"
|
||||||
|
return (
|
||||||
|
<Comp
|
||||||
|
className={cn(buttonVariants({ variant, size, className }))}
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
Button.displayName = "Button"
|
||||||
|
|
||||||
|
export { Button, buttonVariants }
|
||||||
76
components/ui/card.tsx
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Card = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"rounded-xl border bg-card text-card-foreground shadow",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Card.displayName = "Card"
|
||||||
|
|
||||||
|
const CardHeader = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("flex flex-col space-y-1.5 p-6", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardHeader.displayName = "CardHeader"
|
||||||
|
|
||||||
|
const CardTitle = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("font-semibold leading-none tracking-tight", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardTitle.displayName = "CardTitle"
|
||||||
|
|
||||||
|
const CardDescription = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm text-muted-foreground", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardDescription.displayName = "CardDescription"
|
||||||
|
|
||||||
|
const CardContent = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div ref={ref} className={cn("p-6 pt-0", className)} {...props} />
|
||||||
|
))
|
||||||
|
CardContent.displayName = "CardContent"
|
||||||
|
|
||||||
|
const CardFooter = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("flex items-center p-6 pt-0", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardFooter.displayName = "CardFooter"
|
||||||
|
|
||||||
|
export { Card, CardHeader, CardFooter, CardTitle, CardDescription, CardContent }
|
||||||
200
components/ui/context-menu.tsx
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as ContextMenuPrimitive from "@radix-ui/react-context-menu"
|
||||||
|
import { Check, ChevronRight, Circle } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const ContextMenu = ContextMenuPrimitive.Root
|
||||||
|
|
||||||
|
const ContextMenuTrigger = ContextMenuPrimitive.Trigger
|
||||||
|
|
||||||
|
const ContextMenuGroup = ContextMenuPrimitive.Group
|
||||||
|
|
||||||
|
const ContextMenuPortal = ContextMenuPrimitive.Portal
|
||||||
|
|
||||||
|
const ContextMenuSub = ContextMenuPrimitive.Sub
|
||||||
|
|
||||||
|
const ContextMenuRadioGroup = ContextMenuPrimitive.RadioGroup
|
||||||
|
|
||||||
|
const ContextMenuSubTrigger = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.SubTrigger>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.SubTrigger> & {
|
||||||
|
inset?: boolean
|
||||||
|
}
|
||||||
|
>(({ className, inset, children, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.SubTrigger
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground",
|
||||||
|
inset && "pl-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<ChevronRight className="ml-auto h-4 w-4" />
|
||||||
|
</ContextMenuPrimitive.SubTrigger>
|
||||||
|
))
|
||||||
|
ContextMenuSubTrigger.displayName = ContextMenuPrimitive.SubTrigger.displayName
|
||||||
|
|
||||||
|
const ContextMenuSubContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.SubContent>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.SubContent>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.SubContent
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-lg data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ContextMenuSubContent.displayName = ContextMenuPrimitive.SubContent.displayName
|
||||||
|
|
||||||
|
const ContextMenuContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.Content>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Content>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.Portal>
|
||||||
|
<ContextMenuPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-md data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
</ContextMenuPrimitive.Portal>
|
||||||
|
))
|
||||||
|
ContextMenuContent.displayName = ContextMenuPrimitive.Content.displayName
|
||||||
|
|
||||||
|
const ContextMenuItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.Item>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Item> & {
|
||||||
|
inset?: boolean
|
||||||
|
}
|
||||||
|
>(({ className, inset, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.Item
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
inset && "pl-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ContextMenuItem.displayName = ContextMenuPrimitive.Item.displayName
|
||||||
|
|
||||||
|
const ContextMenuCheckboxItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.CheckboxItem>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.CheckboxItem>
|
||||||
|
>(({ className, children, checked, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.CheckboxItem
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
checked={checked}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
||||||
|
<ContextMenuPrimitive.ItemIndicator>
|
||||||
|
<Check className="h-4 w-4" />
|
||||||
|
</ContextMenuPrimitive.ItemIndicator>
|
||||||
|
</span>
|
||||||
|
{children}
|
||||||
|
</ContextMenuPrimitive.CheckboxItem>
|
||||||
|
))
|
||||||
|
ContextMenuCheckboxItem.displayName =
|
||||||
|
ContextMenuPrimitive.CheckboxItem.displayName
|
||||||
|
|
||||||
|
const ContextMenuRadioItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.RadioItem>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.RadioItem>
|
||||||
|
>(({ className, children, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.RadioItem
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
||||||
|
<ContextMenuPrimitive.ItemIndicator>
|
||||||
|
<Circle className="h-4 w-4 fill-current" />
|
||||||
|
</ContextMenuPrimitive.ItemIndicator>
|
||||||
|
</span>
|
||||||
|
{children}
|
||||||
|
</ContextMenuPrimitive.RadioItem>
|
||||||
|
))
|
||||||
|
ContextMenuRadioItem.displayName = ContextMenuPrimitive.RadioItem.displayName
|
||||||
|
|
||||||
|
const ContextMenuLabel = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.Label>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Label> & {
|
||||||
|
inset?: boolean
|
||||||
|
}
|
||||||
|
>(({ className, inset, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.Label
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"px-2 py-1.5 text-sm font-semibold text-foreground",
|
||||||
|
inset && "pl-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ContextMenuLabel.displayName = ContextMenuPrimitive.Label.displayName
|
||||||
|
|
||||||
|
const ContextMenuSeparator = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ContextMenuPrimitive.Separator>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Separator>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ContextMenuPrimitive.Separator
|
||||||
|
ref={ref}
|
||||||
|
className={cn("-mx-1 my-1 h-px bg-border", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ContextMenuSeparator.displayName = ContextMenuPrimitive.Separator.displayName
|
||||||
|
|
||||||
|
const ContextMenuShortcut = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLSpanElement>) => {
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={cn(
|
||||||
|
"ml-auto text-xs tracking-widest text-muted-foreground",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
ContextMenuShortcut.displayName = "ContextMenuShortcut"
|
||||||
|
|
||||||
|
export {
|
||||||
|
ContextMenu,
|
||||||
|
ContextMenuTrigger,
|
||||||
|
ContextMenuContent,
|
||||||
|
ContextMenuItem,
|
||||||
|
ContextMenuCheckboxItem,
|
||||||
|
ContextMenuRadioItem,
|
||||||
|
ContextMenuLabel,
|
||||||
|
ContextMenuSeparator,
|
||||||
|
ContextMenuShortcut,
|
||||||
|
ContextMenuGroup,
|
||||||
|
ContextMenuPortal,
|
||||||
|
ContextMenuSub,
|
||||||
|
ContextMenuSubContent,
|
||||||
|
ContextMenuSubTrigger,
|
||||||
|
ContextMenuRadioGroup,
|
||||||
|
}
|
||||||
122
components/ui/dialog.tsx
Normal file
@@ -0,0 +1,122 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as DialogPrimitive from "@radix-ui/react-dialog"
|
||||||
|
import { X } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Dialog = DialogPrimitive.Root
|
||||||
|
|
||||||
|
const DialogTrigger = DialogPrimitive.Trigger
|
||||||
|
|
||||||
|
const DialogPortal = DialogPrimitive.Portal
|
||||||
|
|
||||||
|
const DialogClose = DialogPrimitive.Close
|
||||||
|
|
||||||
|
const DialogOverlay = React.forwardRef<
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Overlay>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<DialogPrimitive.Overlay
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
DialogOverlay.displayName = DialogPrimitive.Overlay.displayName
|
||||||
|
|
||||||
|
const DialogContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Content>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content>
|
||||||
|
>(({ className, children, ...props }, ref) => (
|
||||||
|
<DialogPortal>
|
||||||
|
<DialogOverlay />
|
||||||
|
<DialogPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"fixed left-[50%] top-[50%] z-50 grid w-full max-w-lg translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<DialogPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground">
|
||||||
|
<X className="h-4 w-4" />
|
||||||
|
<span className="sr-only">Close</span>
|
||||||
|
</DialogPrimitive.Close>
|
||||||
|
</DialogPrimitive.Content>
|
||||||
|
</DialogPortal>
|
||||||
|
))
|
||||||
|
DialogContent.displayName = DialogPrimitive.Content.displayName
|
||||||
|
|
||||||
|
const DialogHeader = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLDivElement>) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"flex flex-col space-y-1.5 text-center sm:text-left",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
DialogHeader.displayName = "DialogHeader"
|
||||||
|
|
||||||
|
const DialogFooter = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLDivElement>) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
DialogFooter.displayName = "DialogFooter"
|
||||||
|
|
||||||
|
const DialogTitle = React.forwardRef<
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Title>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Title>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<DialogPrimitive.Title
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"text-lg font-semibold leading-none tracking-tight",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
DialogTitle.displayName = DialogPrimitive.Title.displayName
|
||||||
|
|
||||||
|
const DialogDescription = React.forwardRef<
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Description>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Description>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<DialogPrimitive.Description
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm text-muted-foreground", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
DialogDescription.displayName = DialogPrimitive.Description.displayName
|
||||||
|
|
||||||
|
export {
|
||||||
|
Dialog,
|
||||||
|
DialogPortal,
|
||||||
|
DialogOverlay,
|
||||||
|
DialogTrigger,
|
||||||
|
DialogClose,
|
||||||
|
DialogContent,
|
||||||
|
DialogHeader,
|
||||||
|
DialogFooter,
|
||||||
|
DialogTitle,
|
||||||
|
DialogDescription,
|
||||||
|
}
|
||||||
178
components/ui/form.tsx
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as LabelPrimitive from "@radix-ui/react-label"
|
||||||
|
import { Slot } from "@radix-ui/react-slot"
|
||||||
|
import {
|
||||||
|
Controller,
|
||||||
|
ControllerProps,
|
||||||
|
FieldPath,
|
||||||
|
FieldValues,
|
||||||
|
FormProvider,
|
||||||
|
useFormContext,
|
||||||
|
} from "react-hook-form"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import { Label } from "@/components/ui/label"
|
||||||
|
|
||||||
|
const Form = FormProvider
|
||||||
|
|
||||||
|
type FormFieldContextValue<
|
||||||
|
TFieldValues extends FieldValues = FieldValues,
|
||||||
|
TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>
|
||||||
|
> = {
|
||||||
|
name: TName
|
||||||
|
}
|
||||||
|
|
||||||
|
const FormFieldContext = React.createContext<FormFieldContextValue>(
|
||||||
|
{} as FormFieldContextValue
|
||||||
|
)
|
||||||
|
|
||||||
|
const FormField = <
|
||||||
|
TFieldValues extends FieldValues = FieldValues,
|
||||||
|
TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>
|
||||||
|
>({
|
||||||
|
...props
|
||||||
|
}: ControllerProps<TFieldValues, TName>) => {
|
||||||
|
return (
|
||||||
|
<FormFieldContext.Provider value={{ name: props.name }}>
|
||||||
|
<Controller {...props} />
|
||||||
|
</FormFieldContext.Provider>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const useFormField = () => {
|
||||||
|
const fieldContext = React.useContext(FormFieldContext)
|
||||||
|
const itemContext = React.useContext(FormItemContext)
|
||||||
|
const { getFieldState, formState } = useFormContext()
|
||||||
|
|
||||||
|
const fieldState = getFieldState(fieldContext.name, formState)
|
||||||
|
|
||||||
|
if (!fieldContext) {
|
||||||
|
throw new Error("useFormField should be used within <FormField>")
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = itemContext
|
||||||
|
|
||||||
|
return {
|
||||||
|
id,
|
||||||
|
name: fieldContext.name,
|
||||||
|
formItemId: `${id}-form-item`,
|
||||||
|
formDescriptionId: `${id}-form-item-description`,
|
||||||
|
formMessageId: `${id}-form-item-message`,
|
||||||
|
...fieldState,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type FormItemContextValue = {
|
||||||
|
id: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const FormItemContext = React.createContext<FormItemContextValue>(
|
||||||
|
{} as FormItemContextValue
|
||||||
|
)
|
||||||
|
|
||||||
|
const FormItem = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => {
|
||||||
|
const id = React.useId()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<FormItemContext.Provider value={{ id }}>
|
||||||
|
<div ref={ref} className={cn("space-y-2", className)} {...props} />
|
||||||
|
</FormItemContext.Provider>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
FormItem.displayName = "FormItem"
|
||||||
|
|
||||||
|
const FormLabel = React.forwardRef<
|
||||||
|
React.ElementRef<typeof LabelPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof LabelPrimitive.Root>
|
||||||
|
>(({ className, ...props }, ref) => {
|
||||||
|
const { error, formItemId } = useFormField()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Label
|
||||||
|
ref={ref}
|
||||||
|
className={cn(error && "text-destructive", className)}
|
||||||
|
htmlFor={formItemId}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
FormLabel.displayName = "FormLabel"
|
||||||
|
|
||||||
|
const FormControl = React.forwardRef<
|
||||||
|
React.ElementRef<typeof Slot>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof Slot>
|
||||||
|
>(({ ...props }, ref) => {
|
||||||
|
const { error, formItemId, formDescriptionId, formMessageId } = useFormField()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Slot
|
||||||
|
ref={ref}
|
||||||
|
id={formItemId}
|
||||||
|
aria-describedby={
|
||||||
|
!error
|
||||||
|
? `${formDescriptionId}`
|
||||||
|
: `${formDescriptionId} ${formMessageId}`
|
||||||
|
}
|
||||||
|
aria-invalid={!!error}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
FormControl.displayName = "FormControl"
|
||||||
|
|
||||||
|
const FormDescription = React.forwardRef<
|
||||||
|
HTMLParagraphElement,
|
||||||
|
React.HTMLAttributes<HTMLParagraphElement>
|
||||||
|
>(({ className, ...props }, ref) => {
|
||||||
|
const { formDescriptionId } = useFormField()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<p
|
||||||
|
ref={ref}
|
||||||
|
id={formDescriptionId}
|
||||||
|
className={cn("text-[0.8rem] text-muted-foreground", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
FormDescription.displayName = "FormDescription"
|
||||||
|
|
||||||
|
const FormMessage = React.forwardRef<
|
||||||
|
HTMLParagraphElement,
|
||||||
|
React.HTMLAttributes<HTMLParagraphElement>
|
||||||
|
>(({ className, children, ...props }, ref) => {
|
||||||
|
const { error, formMessageId } = useFormField()
|
||||||
|
const body = error ? String(error?.message) : children
|
||||||
|
|
||||||
|
if (!body) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<p
|
||||||
|
ref={ref}
|
||||||
|
id={formMessageId}
|
||||||
|
className={cn("text-[0.8rem] font-medium text-destructive", className)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{body}
|
||||||
|
</p>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
FormMessage.displayName = "FormMessage"
|
||||||
|
|
||||||
|
export {
|
||||||
|
useFormField,
|
||||||
|
Form,
|
||||||
|
FormItem,
|
||||||
|
FormLabel,
|
||||||
|
FormControl,
|
||||||
|
FormDescription,
|
||||||
|
FormMessage,
|
||||||
|
FormField,
|
||||||
|
}
|
||||||
22
components/ui/input.tsx
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Input = React.forwardRef<HTMLInputElement, React.ComponentProps<"input">>(
|
||||||
|
({ className, type, ...props }, ref) => {
|
||||||
|
return (
|
||||||
|
<input
|
||||||
|
type={type}
|
||||||
|
className={cn(
|
||||||
|
"flex h-9 w-full rounded-md border border-input bg-transparent px-3 py-1 text-base shadow-sm transition-colors file:border-0 file:bg-transparent file:text-sm file:font-medium file:text-foreground placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
Input.displayName = "Input"
|
||||||
|
|
||||||
|
export { Input }
|
||||||
26
components/ui/label.tsx
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as LabelPrimitive from "@radix-ui/react-label"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const labelVariants = cva(
|
||||||
|
"text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
|
||||||
|
)
|
||||||
|
|
||||||
|
const Label = React.forwardRef<
|
||||||
|
React.ElementRef<typeof LabelPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof LabelPrimitive.Root> &
|
||||||
|
VariantProps<typeof labelVariants>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<LabelPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn(labelVariants(), className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Label.displayName = LabelPrimitive.Root.displayName
|
||||||
|
|
||||||
|
export { Label }
|
||||||
236
components/ui/menubar.tsx
Normal file
@@ -0,0 +1,236 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as MenubarPrimitive from "@radix-ui/react-menubar"
|
||||||
|
import { Check, ChevronRight, Circle } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const MenubarMenu = MenubarPrimitive.Menu
|
||||||
|
|
||||||
|
const MenubarGroup = MenubarPrimitive.Group
|
||||||
|
|
||||||
|
const MenubarPortal = MenubarPrimitive.Portal
|
||||||
|
|
||||||
|
const MenubarSub = MenubarPrimitive.Sub
|
||||||
|
|
||||||
|
const MenubarRadioGroup = MenubarPrimitive.RadioGroup
|
||||||
|
|
||||||
|
const Menubar = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.Root>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex h-9 items-center space-x-1 rounded-md border bg-background p-1 shadow-sm",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Menubar.displayName = MenubarPrimitive.Root.displayName
|
||||||
|
|
||||||
|
const MenubarTrigger = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.Trigger>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.Trigger>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.Trigger
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex cursor-default select-none items-center rounded-sm px-3 py-1 text-sm font-medium outline-none focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
MenubarTrigger.displayName = MenubarPrimitive.Trigger.displayName
|
||||||
|
|
||||||
|
const MenubarSubTrigger = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.SubTrigger>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.SubTrigger> & {
|
||||||
|
inset?: boolean
|
||||||
|
}
|
||||||
|
>(({ className, inset, children, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.SubTrigger
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground",
|
||||||
|
inset && "pl-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<ChevronRight className="ml-auto h-4 w-4" />
|
||||||
|
</MenubarPrimitive.SubTrigger>
|
||||||
|
))
|
||||||
|
MenubarSubTrigger.displayName = MenubarPrimitive.SubTrigger.displayName
|
||||||
|
|
||||||
|
const MenubarSubContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.SubContent>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.SubContent>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.SubContent
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-lg data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
MenubarSubContent.displayName = MenubarPrimitive.SubContent.displayName
|
||||||
|
|
||||||
|
const MenubarContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.Content>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.Content>
|
||||||
|
>(
|
||||||
|
(
|
||||||
|
{ className, align = "start", alignOffset = -4, sideOffset = 8, ...props },
|
||||||
|
ref
|
||||||
|
) => (
|
||||||
|
<MenubarPrimitive.Portal>
|
||||||
|
<MenubarPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
align={align}
|
||||||
|
alignOffset={alignOffset}
|
||||||
|
sideOffset={sideOffset}
|
||||||
|
className={cn(
|
||||||
|
"z-50 min-w-[12rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-md data-[state=open]:animate-in data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
</MenubarPrimitive.Portal>
|
||||||
|
)
|
||||||
|
)
|
||||||
|
MenubarContent.displayName = MenubarPrimitive.Content.displayName
|
||||||
|
|
||||||
|
const MenubarItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.Item>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.Item> & {
|
||||||
|
inset?: boolean
|
||||||
|
}
|
||||||
|
>(({ className, inset, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.Item
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
inset && "pl-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
MenubarItem.displayName = MenubarPrimitive.Item.displayName
|
||||||
|
|
||||||
|
const MenubarCheckboxItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.CheckboxItem>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.CheckboxItem>
|
||||||
|
>(({ className, children, checked, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.CheckboxItem
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
checked={checked}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
||||||
|
<MenubarPrimitive.ItemIndicator>
|
||||||
|
<Check className="h-4 w-4" />
|
||||||
|
</MenubarPrimitive.ItemIndicator>
|
||||||
|
</span>
|
||||||
|
{children}
|
||||||
|
</MenubarPrimitive.CheckboxItem>
|
||||||
|
))
|
||||||
|
MenubarCheckboxItem.displayName = MenubarPrimitive.CheckboxItem.displayName
|
||||||
|
|
||||||
|
const MenubarRadioItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.RadioItem>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.RadioItem>
|
||||||
|
>(({ className, children, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.RadioItem
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
||||||
|
<MenubarPrimitive.ItemIndicator>
|
||||||
|
<Circle className="h-4 w-4 fill-current" />
|
||||||
|
</MenubarPrimitive.ItemIndicator>
|
||||||
|
</span>
|
||||||
|
{children}
|
||||||
|
</MenubarPrimitive.RadioItem>
|
||||||
|
))
|
||||||
|
MenubarRadioItem.displayName = MenubarPrimitive.RadioItem.displayName
|
||||||
|
|
||||||
|
const MenubarLabel = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.Label>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.Label> & {
|
||||||
|
inset?: boolean
|
||||||
|
}
|
||||||
|
>(({ className, inset, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.Label
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"px-2 py-1.5 text-sm font-semibold",
|
||||||
|
inset && "pl-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
MenubarLabel.displayName = MenubarPrimitive.Label.displayName
|
||||||
|
|
||||||
|
const MenubarSeparator = React.forwardRef<
|
||||||
|
React.ElementRef<typeof MenubarPrimitive.Separator>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof MenubarPrimitive.Separator>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<MenubarPrimitive.Separator
|
||||||
|
ref={ref}
|
||||||
|
className={cn("-mx-1 my-1 h-px bg-muted", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
MenubarSeparator.displayName = MenubarPrimitive.Separator.displayName
|
||||||
|
|
||||||
|
const MenubarShortcut = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLSpanElement>) => {
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={cn(
|
||||||
|
"ml-auto text-xs tracking-widest text-muted-foreground",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
MenubarShortcut.displayname = "MenubarShortcut"
|
||||||
|
|
||||||
|
export {
|
||||||
|
Menubar,
|
||||||
|
MenubarMenu,
|
||||||
|
MenubarTrigger,
|
||||||
|
MenubarContent,
|
||||||
|
MenubarItem,
|
||||||
|
MenubarSeparator,
|
||||||
|
MenubarLabel,
|
||||||
|
MenubarCheckboxItem,
|
||||||
|
MenubarRadioGroup,
|
||||||
|
MenubarRadioItem,
|
||||||
|
MenubarPortal,
|
||||||
|
MenubarSubContent,
|
||||||
|
MenubarSubTrigger,
|
||||||
|
MenubarGroup,
|
||||||
|
MenubarSub,
|
||||||
|
MenubarShortcut,
|
||||||
|
}
|
||||||
28
components/ui/progress.tsx
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as ProgressPrimitive from "@radix-ui/react-progress"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Progress = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ProgressPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ProgressPrimitive.Root>
|
||||||
|
>(({ className, value, ...props }, ref) => (
|
||||||
|
<ProgressPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative h-2 w-full overflow-hidden rounded-full bg-primary/20",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ProgressPrimitive.Indicator
|
||||||
|
className="h-full w-full flex-1 bg-primary transition-all"
|
||||||
|
style={{ transform: `translateX(-${100 - (value || 0)}%)` }}
|
||||||
|
/>
|
||||||
|
</ProgressPrimitive.Root>
|
||||||
|
))
|
||||||
|
Progress.displayName = ProgressPrimitive.Root.displayName
|
||||||
|
|
||||||
|
export { Progress }
|
||||||
48
components/ui/scroll-area.tsx
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as ScrollAreaPrimitive from "@radix-ui/react-scroll-area"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const ScrollArea = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ScrollAreaPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ScrollAreaPrimitive.Root>
|
||||||
|
>(({ className, children, ...props }, ref) => (
|
||||||
|
<ScrollAreaPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn("relative overflow-hidden", className)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ScrollAreaPrimitive.Viewport className="h-full w-full rounded-[inherit]">
|
||||||
|
{children}
|
||||||
|
</ScrollAreaPrimitive.Viewport>
|
||||||
|
<ScrollBar />
|
||||||
|
<ScrollAreaPrimitive.Corner />
|
||||||
|
</ScrollAreaPrimitive.Root>
|
||||||
|
))
|
||||||
|
ScrollArea.displayName = ScrollAreaPrimitive.Root.displayName
|
||||||
|
|
||||||
|
const ScrollBar = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ScrollAreaPrimitive.ScrollAreaScrollbar>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ScrollAreaPrimitive.ScrollAreaScrollbar>
|
||||||
|
>(({ className, orientation = "vertical", ...props }, ref) => (
|
||||||
|
<ScrollAreaPrimitive.ScrollAreaScrollbar
|
||||||
|
ref={ref}
|
||||||
|
orientation={orientation}
|
||||||
|
className={cn(
|
||||||
|
"flex touch-none select-none transition-colors",
|
||||||
|
orientation === "vertical" &&
|
||||||
|
"h-full w-2.5 border-l border-l-transparent p-[1px]",
|
||||||
|
orientation === "horizontal" &&
|
||||||
|
"h-2.5 flex-col border-t border-t-transparent p-[1px]",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ScrollAreaPrimitive.ScrollAreaThumb className="relative flex-1 rounded-full bg-border" />
|
||||||
|
</ScrollAreaPrimitive.ScrollAreaScrollbar>
|
||||||
|
))
|
||||||
|
ScrollBar.displayName = ScrollAreaPrimitive.ScrollAreaScrollbar.displayName
|
||||||
|
|
||||||
|
export { ScrollArea, ScrollBar }
|
||||||
159
components/ui/select.tsx
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as SelectPrimitive from "@radix-ui/react-select"
|
||||||
|
import { Check, ChevronDown, ChevronUp } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Select = SelectPrimitive.Root
|
||||||
|
|
||||||
|
const SelectGroup = SelectPrimitive.Group
|
||||||
|
|
||||||
|
const SelectValue = SelectPrimitive.Value
|
||||||
|
|
||||||
|
const SelectTrigger = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.Trigger>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Trigger>
|
||||||
|
>(({ className, children, ...props }, ref) => (
|
||||||
|
<SelectPrimitive.Trigger
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex h-9 w-full items-center justify-between whitespace-nowrap rounded-md border border-input bg-transparent px-3 py-2 text-sm shadow-sm ring-offset-background placeholder:text-muted-foreground focus:outline-none focus:ring-1 focus:ring-ring disabled:cursor-not-allowed disabled:opacity-50 [&>span]:line-clamp-1",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<SelectPrimitive.Icon asChild>
|
||||||
|
<ChevronDown className="h-4 w-4 opacity-50" />
|
||||||
|
</SelectPrimitive.Icon>
|
||||||
|
</SelectPrimitive.Trigger>
|
||||||
|
))
|
||||||
|
SelectTrigger.displayName = SelectPrimitive.Trigger.displayName
|
||||||
|
|
||||||
|
const SelectScrollUpButton = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.ScrollUpButton>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.ScrollUpButton>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<SelectPrimitive.ScrollUpButton
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex cursor-default items-center justify-center py-1",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ChevronUp className="h-4 w-4" />
|
||||||
|
</SelectPrimitive.ScrollUpButton>
|
||||||
|
))
|
||||||
|
SelectScrollUpButton.displayName = SelectPrimitive.ScrollUpButton.displayName
|
||||||
|
|
||||||
|
const SelectScrollDownButton = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.ScrollDownButton>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.ScrollDownButton>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<SelectPrimitive.ScrollDownButton
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"flex cursor-default items-center justify-center py-1",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<ChevronDown className="h-4 w-4" />
|
||||||
|
</SelectPrimitive.ScrollDownButton>
|
||||||
|
))
|
||||||
|
SelectScrollDownButton.displayName =
|
||||||
|
SelectPrimitive.ScrollDownButton.displayName
|
||||||
|
|
||||||
|
const SelectContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.Content>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Content>
|
||||||
|
>(({ className, children, position = "popper", ...props }, ref) => (
|
||||||
|
<SelectPrimitive.Portal>
|
||||||
|
<SelectPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative z-50 max-h-96 min-w-[8rem] overflow-hidden rounded-md border bg-popover text-popover-foreground shadow-md data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
|
||||||
|
position === "popper" &&
|
||||||
|
"data-[side=bottom]:translate-y-1 data-[side=left]:-translate-x-1 data-[side=right]:translate-x-1 data-[side=top]:-translate-y-1",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
position={position}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<SelectScrollUpButton />
|
||||||
|
<SelectPrimitive.Viewport
|
||||||
|
className={cn(
|
||||||
|
"p-1",
|
||||||
|
position === "popper" &&
|
||||||
|
"h-[var(--radix-select-trigger-height)] w-full min-w-[var(--radix-select-trigger-width)]"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</SelectPrimitive.Viewport>
|
||||||
|
<SelectScrollDownButton />
|
||||||
|
</SelectPrimitive.Content>
|
||||||
|
</SelectPrimitive.Portal>
|
||||||
|
))
|
||||||
|
SelectContent.displayName = SelectPrimitive.Content.displayName
|
||||||
|
|
||||||
|
const SelectLabel = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.Label>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Label>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<SelectPrimitive.Label
|
||||||
|
ref={ref}
|
||||||
|
className={cn("px-2 py-1.5 text-sm font-semibold", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
SelectLabel.displayName = SelectPrimitive.Label.displayName
|
||||||
|
|
||||||
|
const SelectItem = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.Item>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Item>
|
||||||
|
>(({ className, children, ...props }, ref) => (
|
||||||
|
<SelectPrimitive.Item
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-2 pr-8 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<span className="absolute right-2 flex h-3.5 w-3.5 items-center justify-center">
|
||||||
|
<SelectPrimitive.ItemIndicator>
|
||||||
|
<Check className="h-4 w-4" />
|
||||||
|
</SelectPrimitive.ItemIndicator>
|
||||||
|
</span>
|
||||||
|
<SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
|
||||||
|
</SelectPrimitive.Item>
|
||||||
|
))
|
||||||
|
SelectItem.displayName = SelectPrimitive.Item.displayName
|
||||||
|
|
||||||
|
const SelectSeparator = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SelectPrimitive.Separator>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Separator>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<SelectPrimitive.Separator
|
||||||
|
ref={ref}
|
||||||
|
className={cn("-mx-1 my-1 h-px bg-muted", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
SelectSeparator.displayName = SelectPrimitive.Separator.displayName
|
||||||
|
|
||||||
|
export {
|
||||||
|
Select,
|
||||||
|
SelectGroup,
|
||||||
|
SelectValue,
|
||||||
|
SelectTrigger,
|
||||||
|
SelectContent,
|
||||||
|
SelectLabel,
|
||||||
|
SelectItem,
|
||||||
|
SelectSeparator,
|
||||||
|
SelectScrollUpButton,
|
||||||
|
SelectScrollDownButton,
|
||||||
|
}
|
||||||
31
components/ui/separator.tsx
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as SeparatorPrimitive from "@radix-ui/react-separator"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Separator = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SeparatorPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SeparatorPrimitive.Root>
|
||||||
|
>(
|
||||||
|
(
|
||||||
|
{ className, orientation = "horizontal", decorative = true, ...props },
|
||||||
|
ref
|
||||||
|
) => (
|
||||||
|
<SeparatorPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
decorative={decorative}
|
||||||
|
orientation={orientation}
|
||||||
|
className={cn(
|
||||||
|
"shrink-0 bg-border",
|
||||||
|
orientation === "horizontal" ? "h-[1px] w-full" : "h-full w-[1px]",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
)
|
||||||
|
Separator.displayName = SeparatorPrimitive.Root.displayName
|
||||||
|
|
||||||
|
export { Separator }
|
||||||
55
components/ui/tabs.tsx
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as TabsPrimitive from "@radix-ui/react-tabs"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Tabs = TabsPrimitive.Root
|
||||||
|
|
||||||
|
const TabsList = React.forwardRef<
|
||||||
|
React.ElementRef<typeof TabsPrimitive.List>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof TabsPrimitive.List>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<TabsPrimitive.List
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"inline-flex h-9 items-center justify-center rounded-lg bg-muted p-1 text-muted-foreground",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
TabsList.displayName = TabsPrimitive.List.displayName
|
||||||
|
|
||||||
|
const TabsTrigger = React.forwardRef<
|
||||||
|
React.ElementRef<typeof TabsPrimitive.Trigger>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof TabsPrimitive.Trigger>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<TabsPrimitive.Trigger
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"inline-flex items-center justify-center whitespace-nowrap rounded-md px-3 py-1 text-sm font-medium ring-offset-background transition-all focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:shadow",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
TabsTrigger.displayName = TabsPrimitive.Trigger.displayName
|
||||||
|
|
||||||
|
const TabsContent = React.forwardRef<
|
||||||
|
React.ElementRef<typeof TabsPrimitive.Content>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof TabsPrimitive.Content>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<TabsPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"mt-2 ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
TabsContent.displayName = TabsPrimitive.Content.displayName
|
||||||
|
|
||||||
|
export { Tabs, TabsList, TabsTrigger, TabsContent }
|
||||||
129
components/ui/toast.tsx
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as ToastPrimitives from "@radix-ui/react-toast"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
import { X } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const ToastProvider = ToastPrimitives.Provider
|
||||||
|
|
||||||
|
const ToastViewport = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ToastPrimitives.Viewport>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ToastPrimitives.Viewport>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ToastPrimitives.Viewport
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"fixed top-0 z-[100] flex max-h-screen w-full flex-col-reverse p-4 sm:bottom-0 sm:right-0 sm:top-auto sm:flex-col md:max-w-[420px]",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ToastViewport.displayName = ToastPrimitives.Viewport.displayName
|
||||||
|
|
||||||
|
const toastVariants = cva(
|
||||||
|
"group pointer-events-auto relative flex w-full items-center justify-between space-x-2 overflow-hidden rounded-md border p-4 pr-6 shadow-lg transition-all data-[swipe=cancel]:translate-x-0 data-[swipe=end]:translate-x-[var(--radix-toast-swipe-end-x)] data-[swipe=move]:translate-x-[var(--radix-toast-swipe-move-x)] data-[swipe=move]:transition-none data-[state=open]:animate-in data-[state=closed]:animate-out data-[swipe=end]:animate-out data-[state=closed]:fade-out-80 data-[state=closed]:slide-out-to-right-full data-[state=open]:slide-in-from-top-full data-[state=open]:sm:slide-in-from-bottom-full",
|
||||||
|
{
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
default: "border bg-background text-foreground",
|
||||||
|
destructive:
|
||||||
|
"destructive group border-destructive bg-destructive text-destructive-foreground",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
variant: "default",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const Toast = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ToastPrimitives.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ToastPrimitives.Root> &
|
||||||
|
VariantProps<typeof toastVariants>
|
||||||
|
>(({ className, variant, ...props }, ref) => {
|
||||||
|
return (
|
||||||
|
<ToastPrimitives.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn(toastVariants({ variant }), className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
Toast.displayName = ToastPrimitives.Root.displayName
|
||||||
|
|
||||||
|
const ToastAction = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ToastPrimitives.Action>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ToastPrimitives.Action>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ToastPrimitives.Action
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"inline-flex h-8 shrink-0 items-center justify-center rounded-md border bg-transparent px-3 text-sm font-medium transition-colors hover:bg-secondary focus:outline-none focus:ring-1 focus:ring-ring disabled:pointer-events-none disabled:opacity-50 group-[.destructive]:border-muted/40 group-[.destructive]:hover:border-destructive/30 group-[.destructive]:hover:bg-destructive group-[.destructive]:hover:text-destructive-foreground group-[.destructive]:focus:ring-destructive",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ToastAction.displayName = ToastPrimitives.Action.displayName
|
||||||
|
|
||||||
|
const ToastClose = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ToastPrimitives.Close>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ToastPrimitives.Close>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ToastPrimitives.Close
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"absolute right-1 top-1 rounded-md p-1 text-foreground/50 opacity-0 transition-opacity hover:text-foreground focus:opacity-100 focus:outline-none focus:ring-1 group-hover:opacity-100 group-[.destructive]:text-red-300 group-[.destructive]:hover:text-red-50 group-[.destructive]:focus:ring-red-400 group-[.destructive]:focus:ring-offset-red-600",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
toast-close=""
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<X className="h-4 w-4" />
|
||||||
|
</ToastPrimitives.Close>
|
||||||
|
))
|
||||||
|
ToastClose.displayName = ToastPrimitives.Close.displayName
|
||||||
|
|
||||||
|
const ToastTitle = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ToastPrimitives.Title>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ToastPrimitives.Title>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ToastPrimitives.Title
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm font-semibold [&+div]:text-xs", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ToastTitle.displayName = ToastPrimitives.Title.displayName
|
||||||
|
|
||||||
|
const ToastDescription = React.forwardRef<
|
||||||
|
React.ElementRef<typeof ToastPrimitives.Description>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof ToastPrimitives.Description>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<ToastPrimitives.Description
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm opacity-90", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
ToastDescription.displayName = ToastPrimitives.Description.displayName
|
||||||
|
|
||||||
|
type ToastProps = React.ComponentPropsWithoutRef<typeof Toast>
|
||||||
|
|
||||||
|
type ToastActionElement = React.ReactElement<typeof ToastAction>
|
||||||
|
|
||||||
|
export {
|
||||||
|
type ToastProps,
|
||||||
|
type ToastActionElement,
|
||||||
|
ToastProvider,
|
||||||
|
ToastViewport,
|
||||||
|
Toast,
|
||||||
|
ToastTitle,
|
||||||
|
ToastDescription,
|
||||||
|
ToastClose,
|
||||||
|
ToastAction,
|
||||||
|
}
|
||||||
35
components/ui/toaster.tsx
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import { useToast } from "@/hooks/use-toast"
|
||||||
|
import {
|
||||||
|
Toast,
|
||||||
|
ToastClose,
|
||||||
|
ToastDescription,
|
||||||
|
ToastProvider,
|
||||||
|
ToastTitle,
|
||||||
|
ToastViewport,
|
||||||
|
} from "@/components/ui/toast"
|
||||||
|
|
||||||
|
export function Toaster() {
|
||||||
|
const { toasts } = useToast()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ToastProvider>
|
||||||
|
{toasts.map(function ({ id, title, description, action, ...props }) {
|
||||||
|
return (
|
||||||
|
<Toast key={id} {...props}>
|
||||||
|
<div className="grid gap-1">
|
||||||
|
{title && <ToastTitle>{title}</ToastTitle>}
|
||||||
|
{description && (
|
||||||
|
<ToastDescription>{description}</ToastDescription>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{action}
|
||||||
|
<ToastClose />
|
||||||
|
</Toast>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
<ToastViewport />
|
||||||
|
</ToastProvider>
|
||||||
|
)
|
||||||
|
}
|
||||||
194
hooks/use-toast.ts
Normal file
@@ -0,0 +1,194 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
// Inspired by react-hot-toast library
|
||||||
|
import * as React from "react"
|
||||||
|
|
||||||
|
import type {
|
||||||
|
ToastActionElement,
|
||||||
|
ToastProps,
|
||||||
|
} from "@/components/ui/toast"
|
||||||
|
|
||||||
|
const TOAST_LIMIT = 1
|
||||||
|
const TOAST_REMOVE_DELAY = 1000000
|
||||||
|
|
||||||
|
type ToasterToast = ToastProps & {
|
||||||
|
id: string
|
||||||
|
title?: React.ReactNode
|
||||||
|
description?: React.ReactNode
|
||||||
|
action?: ToastActionElement
|
||||||
|
}
|
||||||
|
|
||||||
|
const actionTypes = {
|
||||||
|
ADD_TOAST: "ADD_TOAST",
|
||||||
|
UPDATE_TOAST: "UPDATE_TOAST",
|
||||||
|
DISMISS_TOAST: "DISMISS_TOAST",
|
||||||
|
REMOVE_TOAST: "REMOVE_TOAST",
|
||||||
|
} as const
|
||||||
|
|
||||||
|
let count = 0
|
||||||
|
|
||||||
|
function genId() {
|
||||||
|
count = (count + 1) % Number.MAX_SAFE_INTEGER
|
||||||
|
return count.toString()
|
||||||
|
}
|
||||||
|
|
||||||
|
type ActionType = typeof actionTypes
|
||||||
|
|
||||||
|
type Action =
|
||||||
|
| {
|
||||||
|
type: ActionType["ADD_TOAST"]
|
||||||
|
toast: ToasterToast
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: ActionType["UPDATE_TOAST"]
|
||||||
|
toast: Partial<ToasterToast>
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: ActionType["DISMISS_TOAST"]
|
||||||
|
toastId?: ToasterToast["id"]
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: ActionType["REMOVE_TOAST"]
|
||||||
|
toastId?: ToasterToast["id"]
|
||||||
|
}
|
||||||
|
|
||||||
|
interface State {
|
||||||
|
toasts: ToasterToast[]
|
||||||
|
}
|
||||||
|
|
||||||
|
const toastTimeouts = new Map<string, ReturnType<typeof setTimeout>>()
|
||||||
|
|
||||||
|
const addToRemoveQueue = (toastId: string) => {
|
||||||
|
if (toastTimeouts.has(toastId)) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const timeout = setTimeout(() => {
|
||||||
|
toastTimeouts.delete(toastId)
|
||||||
|
dispatch({
|
||||||
|
type: "REMOVE_TOAST",
|
||||||
|
toastId: toastId,
|
||||||
|
})
|
||||||
|
}, TOAST_REMOVE_DELAY)
|
||||||
|
|
||||||
|
toastTimeouts.set(toastId, timeout)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const reducer = (state: State, action: Action): State => {
|
||||||
|
switch (action.type) {
|
||||||
|
case "ADD_TOAST":
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
toasts: [action.toast, ...state.toasts].slice(0, TOAST_LIMIT),
|
||||||
|
}
|
||||||
|
|
||||||
|
case "UPDATE_TOAST":
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
toasts: state.toasts.map((t) =>
|
||||||
|
t.id === action.toast.id ? { ...t, ...action.toast } : t
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
case "DISMISS_TOAST": {
|
||||||
|
const { toastId } = action
|
||||||
|
|
||||||
|
// ! Side effects ! - This could be extracted into a dismissToast() action,
|
||||||
|
// but I'll keep it here for simplicity
|
||||||
|
if (toastId) {
|
||||||
|
addToRemoveQueue(toastId)
|
||||||
|
} else {
|
||||||
|
state.toasts.forEach((toast) => {
|
||||||
|
addToRemoveQueue(toast.id)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
toasts: state.toasts.map((t) =>
|
||||||
|
t.id === toastId || toastId === undefined
|
||||||
|
? {
|
||||||
|
...t,
|
||||||
|
open: false,
|
||||||
|
}
|
||||||
|
: t
|
||||||
|
),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case "REMOVE_TOAST":
|
||||||
|
if (action.toastId === undefined) {
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
toasts: [],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
toasts: state.toasts.filter((t) => t.id !== action.toastId),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const listeners: Array<(state: State) => void> = []
|
||||||
|
|
||||||
|
let memoryState: State = { toasts: [] }
|
||||||
|
|
||||||
|
function dispatch(action: Action) {
|
||||||
|
memoryState = reducer(memoryState, action)
|
||||||
|
listeners.forEach((listener) => {
|
||||||
|
listener(memoryState)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type Toast = Omit<ToasterToast, "id">
|
||||||
|
|
||||||
|
function toast({ ...props }: Toast) {
|
||||||
|
const id = genId()
|
||||||
|
|
||||||
|
const update = (props: ToasterToast) =>
|
||||||
|
dispatch({
|
||||||
|
type: "UPDATE_TOAST",
|
||||||
|
toast: { ...props, id },
|
||||||
|
})
|
||||||
|
const dismiss = () => dispatch({ type: "DISMISS_TOAST", toastId: id })
|
||||||
|
|
||||||
|
dispatch({
|
||||||
|
type: "ADD_TOAST",
|
||||||
|
toast: {
|
||||||
|
...props,
|
||||||
|
id,
|
||||||
|
open: true,
|
||||||
|
onOpenChange: (open) => {
|
||||||
|
if (!open) dismiss()
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: id,
|
||||||
|
dismiss,
|
||||||
|
update,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function useToast() {
|
||||||
|
const [state, setState] = React.useState<State>(memoryState)
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
listeners.push(setState)
|
||||||
|
return () => {
|
||||||
|
const index = listeners.indexOf(setState)
|
||||||
|
if (index > -1) {
|
||||||
|
listeners.splice(index, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [state])
|
||||||
|
|
||||||
|
return {
|
||||||
|
...state,
|
||||||
|
toast,
|
||||||
|
dismiss: (toastId?: string) => dispatch({ type: "DISMISS_TOAST", toastId }),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { useToast, toast }
|
||||||
7
jest.config.js
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
module.exports = {
|
||||||
|
preset: 'ts-jest',
|
||||||
|
testEnvironment: 'node',
|
||||||
|
moduleNameMapper: {
|
||||||
|
'^@/(.*)$': '<rootDir>/$1',
|
||||||
|
},
|
||||||
|
};
|
||||||
339
lib/navidrome.ts
Normal file
@@ -0,0 +1,339 @@
|
|||||||
|
import crypto from 'crypto';
|
||||||
|
|
||||||
|
export interface NavidromeConfig {
|
||||||
|
serverUrl: string;
|
||||||
|
username: string;
|
||||||
|
password: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SubsonicResponse<T = any> {
|
||||||
|
'subsonic-response': {
|
||||||
|
status: string;
|
||||||
|
version: string;
|
||||||
|
type: string;
|
||||||
|
serverVersion?: string;
|
||||||
|
openSubsonic?: boolean;
|
||||||
|
error?: {
|
||||||
|
code: number;
|
||||||
|
message: string;
|
||||||
|
};
|
||||||
|
} & T;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Album {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
artist: string;
|
||||||
|
artistId: string;
|
||||||
|
coverArt?: string;
|
||||||
|
songCount: number;
|
||||||
|
duration: number;
|
||||||
|
playCount?: number;
|
||||||
|
created: string;
|
||||||
|
starred?: string;
|
||||||
|
year?: number;
|
||||||
|
genre?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Artist {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
albumCount: number;
|
||||||
|
starred?: string;
|
||||||
|
coverArt?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Song {
|
||||||
|
id: string;
|
||||||
|
parent: string;
|
||||||
|
isDir: boolean;
|
||||||
|
title: string;
|
||||||
|
album: string;
|
||||||
|
artist: string;
|
||||||
|
track?: number;
|
||||||
|
year?: number;
|
||||||
|
genre?: string;
|
||||||
|
coverArt?: string;
|
||||||
|
size: number;
|
||||||
|
contentType: string;
|
||||||
|
suffix: string;
|
||||||
|
duration: number;
|
||||||
|
bitRate?: number;
|
||||||
|
path: string;
|
||||||
|
playCount?: number;
|
||||||
|
discNumber?: number;
|
||||||
|
created: string;
|
||||||
|
albumId: string;
|
||||||
|
artistId: string;
|
||||||
|
type: string;
|
||||||
|
starred?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Playlist {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
comment?: string;
|
||||||
|
owner: string;
|
||||||
|
public: boolean;
|
||||||
|
songCount: number;
|
||||||
|
duration: number;
|
||||||
|
created: string;
|
||||||
|
changed: string;
|
||||||
|
coverArt?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
class NavidromeAPI {
|
||||||
|
private config: NavidromeConfig;
|
||||||
|
private clientName = 'stillnavidrome';
|
||||||
|
private version = '1.16.0';
|
||||||
|
|
||||||
|
constructor(config: NavidromeConfig) {
|
||||||
|
this.config = config;
|
||||||
|
}
|
||||||
|
|
||||||
|
private generateSalt(): string {
|
||||||
|
return crypto.randomBytes(8).toString('hex');
|
||||||
|
}
|
||||||
|
|
||||||
|
private generateToken(password: string, salt: string): string {
|
||||||
|
return crypto.createHash('md5').update(password + salt).digest('hex');
|
||||||
|
}
|
||||||
|
|
||||||
|
private async makeRequest(endpoint: string, params: Record<string, any> = {}): Promise<any> {
|
||||||
|
const salt = this.generateSalt();
|
||||||
|
const token = this.generateToken(this.config.password, salt);
|
||||||
|
|
||||||
|
const queryParams = new URLSearchParams({
|
||||||
|
u: this.config.username,
|
||||||
|
t: token,
|
||||||
|
s: salt,
|
||||||
|
v: this.version,
|
||||||
|
c: this.clientName,
|
||||||
|
f: 'json',
|
||||||
|
...params
|
||||||
|
});
|
||||||
|
|
||||||
|
const url = `${this.config.serverUrl}/rest/${endpoint}?${queryParams.toString()}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(url);
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data: SubsonicResponse = await response.json();
|
||||||
|
|
||||||
|
if (data['subsonic-response'].status === 'failed') {
|
||||||
|
throw new Error(data['subsonic-response'].error?.message || 'Unknown error');
|
||||||
|
}
|
||||||
|
|
||||||
|
return data['subsonic-response'];
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Navidrome API request failed:', error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async ping(): Promise<boolean> {
|
||||||
|
try {
|
||||||
|
await this.makeRequest('ping');
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async getArtists(): Promise<Artist[]> {
|
||||||
|
const response = await this.makeRequest('getArtists');
|
||||||
|
const artists: Artist[] = [];
|
||||||
|
|
||||||
|
if (response.artists?.index) {
|
||||||
|
for (const index of response.artists.index) {
|
||||||
|
if (index.artist) {
|
||||||
|
artists.push(...index.artist);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return artists;
|
||||||
|
}
|
||||||
|
|
||||||
|
async getArtist(artistId: string): Promise<{ artist: Artist; albums: Album[] }> {
|
||||||
|
const response = await this.makeRequest('getArtist', { id: artistId });
|
||||||
|
return {
|
||||||
|
artist: response.artist,
|
||||||
|
albums: response.artist.album || []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async getAlbums(type?: 'newest' | 'recent' | 'frequent' | 'random', size: number = 50, offset: number = 0): Promise<Album[]> {
|
||||||
|
const response = await this.makeRequest('getAlbumList2', {
|
||||||
|
type: type || 'newest',
|
||||||
|
size,
|
||||||
|
offset
|
||||||
|
});
|
||||||
|
return response.albumList2?.album || [];
|
||||||
|
}
|
||||||
|
|
||||||
|
async getAlbum(albumId: string): Promise<{ album: Album; songs: Song[] }> {
|
||||||
|
const response = await this.makeRequest('getAlbum', { id: albumId });
|
||||||
|
return {
|
||||||
|
album: response.album,
|
||||||
|
songs: response.album.song || []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async search(query: string, artistCount = 20, albumCount = 20, songCount = 20): Promise<{
|
||||||
|
artists: Artist[];
|
||||||
|
albums: Album[];
|
||||||
|
songs: Song[];
|
||||||
|
}> {
|
||||||
|
const response = await this.makeRequest('search3', {
|
||||||
|
query,
|
||||||
|
artistCount,
|
||||||
|
albumCount,
|
||||||
|
songCount
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
artists: response.searchResult3?.artist || [],
|
||||||
|
albums: response.searchResult3?.album || [],
|
||||||
|
songs: response.searchResult3?.song || []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async getPlaylists(): Promise<Playlist[]> {
|
||||||
|
const response = await this.makeRequest('getPlaylists');
|
||||||
|
return response.playlists?.playlist || [];
|
||||||
|
}
|
||||||
|
|
||||||
|
async getPlaylist(playlistId: string): Promise<{ playlist: Playlist; songs: Song[] }> {
|
||||||
|
const response = await this.makeRequest('getPlaylist', { id: playlistId });
|
||||||
|
return {
|
||||||
|
playlist: response.playlist,
|
||||||
|
songs: response.playlist.entry || []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async createPlaylist(name: string, songIds?: string[]): Promise<Playlist> {
|
||||||
|
const params: Record<string, any> = { name };
|
||||||
|
if (songIds && songIds.length > 0) {
|
||||||
|
songIds.forEach((id, index) => {
|
||||||
|
params[`songId[${index}]`] = id;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await this.makeRequest('createPlaylist', params);
|
||||||
|
return response.playlist;
|
||||||
|
}
|
||||||
|
|
||||||
|
async updatePlaylist(playlistId: string, name?: string, comment?: string, songIds?: string[]): Promise<void> {
|
||||||
|
const params: Record<string, any> = { playlistId };
|
||||||
|
if (name) params.name = name;
|
||||||
|
if (comment) params.comment = comment;
|
||||||
|
if (songIds) {
|
||||||
|
songIds.forEach((id, index) => {
|
||||||
|
params[`songId[${index}]`] = id;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.makeRequest('updatePlaylist', params);
|
||||||
|
}
|
||||||
|
|
||||||
|
async deletePlaylist(playlistId: string): Promise<void> {
|
||||||
|
await this.makeRequest('deletePlaylist', { id: playlistId });
|
||||||
|
}
|
||||||
|
|
||||||
|
getStreamUrl(songId: string, maxBitRate?: number): string {
|
||||||
|
const salt = this.generateSalt();
|
||||||
|
const token = this.generateToken(this.config.password, salt);
|
||||||
|
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
u: this.config.username,
|
||||||
|
t: token,
|
||||||
|
s: salt,
|
||||||
|
v: this.version,
|
||||||
|
c: this.clientName,
|
||||||
|
id: songId
|
||||||
|
});
|
||||||
|
|
||||||
|
if (maxBitRate) {
|
||||||
|
params.append('maxBitRate', maxBitRate.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${this.config.serverUrl}/rest/stream?${params.toString()}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getCoverArtUrl(coverArtId: string, size?: number): string {
|
||||||
|
const salt = this.generateSalt();
|
||||||
|
const token = this.generateToken(this.config.password, salt);
|
||||||
|
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
u: this.config.username,
|
||||||
|
t: token,
|
||||||
|
s: salt,
|
||||||
|
v: this.version,
|
||||||
|
c: this.clientName,
|
||||||
|
id: coverArtId
|
||||||
|
});
|
||||||
|
|
||||||
|
if (size) {
|
||||||
|
params.append('size', size.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${this.config.serverUrl}/rest/getCoverArt?${params.toString()}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async star(id: string, type: 'song' | 'album' | 'artist'): Promise<void> {
|
||||||
|
const paramName = type === 'song' ? 'id' : type === 'album' ? 'albumId' : 'artistId';
|
||||||
|
await this.makeRequest('star', { [paramName]: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
async unstar(id: string, type: 'song' | 'album' | 'artist'): Promise<void> {
|
||||||
|
const paramName = type === 'song' ? 'id' : type === 'album' ? 'albumId' : 'artistId';
|
||||||
|
await this.makeRequest('unstar', { [paramName]: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
async scrobble(songId: string, submission: boolean = true): Promise<void> {
|
||||||
|
await this.makeRequest('scrobble', {
|
||||||
|
id: songId,
|
||||||
|
submission: submission.toString(),
|
||||||
|
time: Date.now()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async getAllSongs(size = 500, offset = 0): Promise<Song[]> {
|
||||||
|
const response = await this.makeRequest('search3', {
|
||||||
|
query: '',
|
||||||
|
songCount: size,
|
||||||
|
songOffset: offset,
|
||||||
|
artistCount: 0,
|
||||||
|
albumCount: 0
|
||||||
|
});
|
||||||
|
|
||||||
|
return response.searchResult3?.song || [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Singleton instance management
|
||||||
|
let navidromeInstance: NavidromeAPI | null = null;
|
||||||
|
|
||||||
|
export function getNavidromeAPI(): NavidromeAPI {
|
||||||
|
if (!navidromeInstance) {
|
||||||
|
const config: NavidromeConfig = {
|
||||||
|
serverUrl: process.env.NEXT_PUBLIC_NAVIDROME_URL || '',
|
||||||
|
username: process.env.NEXT_PUBLIC_NAVIDROME_USERNAME || '',
|
||||||
|
password: process.env.NEXT_PUBLIC_NAVIDROME_PASSWORD || ''
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!config.serverUrl || !config.username || !config.password) {
|
||||||
|
throw new Error('Navidrome configuration is incomplete. Please check environment variables.');
|
||||||
|
}
|
||||||
|
|
||||||
|
navidromeInstance = new NavidromeAPI(config);
|
||||||
|
}
|
||||||
|
|
||||||
|
return navidromeInstance;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default NavidromeAPI;
|
||||||
6
lib/utils.ts
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
import { type ClassValue, clsx } from "clsx"
|
||||||
|
import { twMerge } from "tailwind-merge"
|
||||||
|
|
||||||
|
export function cn(...inputs: ClassValue[]) {
|
||||||
|
return twMerge(clsx(inputs))
|
||||||
|
}
|
||||||
51
next.config.mjs
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
/** @type {import('next').NextConfig} */
|
||||||
|
const nextConfig = {
|
||||||
|
images: {
|
||||||
|
remotePatterns: [
|
||||||
|
{
|
||||||
|
protocol: "https",
|
||||||
|
hostname: "**",
|
||||||
|
}
|
||||||
|
],
|
||||||
|
},
|
||||||
|
async headers() {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
source: '/(.*)',
|
||||||
|
headers: [
|
||||||
|
{
|
||||||
|
key: 'X-Content-Type-Options',
|
||||||
|
value: 'nosniff',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'X-Frame-Options',
|
||||||
|
value: 'DENY',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'Referrer-Policy',
|
||||||
|
value: 'strict-origin-when-cross-origin',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
source: '/sw.js',
|
||||||
|
headers: [
|
||||||
|
{
|
||||||
|
key: 'Content-Type',
|
||||||
|
value: 'application/javascript; charset=utf-8',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'Cache-Control',
|
||||||
|
value: 'no-cache, no-store, must-revalidate',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'Content-Security-Policy',
|
||||||
|
value: "default-src 'self'; script-src 'self'",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default nextConfig;
|
||||||
59
package.json
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
{
|
||||||
|
"name": "offbrand-spotify",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"private": true,
|
||||||
|
"scripts": {
|
||||||
|
"dev": "next dev -p 40625",
|
||||||
|
"build": "next build",
|
||||||
|
"start": "next start -p 40625",
|
||||||
|
"lint": "next lint",
|
||||||
|
"test": "jest"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@hookform/resolvers": "^3.9.1",
|
||||||
|
"@neutrixs/colorthief": "^2.5.0",
|
||||||
|
"@radix-ui/react-avatar": "^1.1.1",
|
||||||
|
"@radix-ui/react-context-menu": "^2.2.2",
|
||||||
|
"@radix-ui/react-dialog": "^1.1.2",
|
||||||
|
"@radix-ui/react-icons": "^1.3.2",
|
||||||
|
"@radix-ui/react-label": "^2.1.0",
|
||||||
|
"@radix-ui/react-menubar": "^1.1.2",
|
||||||
|
"@radix-ui/react-progress": "^1.1.1",
|
||||||
|
"@radix-ui/react-scroll-area": "^1.2.1",
|
||||||
|
"@radix-ui/react-select": "^2.1.2",
|
||||||
|
"@radix-ui/react-separator": "^1.1.0",
|
||||||
|
"@radix-ui/react-slot": "^1.1.0",
|
||||||
|
"@radix-ui/react-tabs": "^1.1.1",
|
||||||
|
"@radix-ui/react-toast": "^1.2.4",
|
||||||
|
"@vercel/analytics": "^1.4.1",
|
||||||
|
"@vercel/speed-insights": "^1.1.0",
|
||||||
|
"axios": "^1.7.7",
|
||||||
|
"chalk": "^5.3.0",
|
||||||
|
"class-variance-authority": "^0.7.0",
|
||||||
|
"clsx": "^2.1.1",
|
||||||
|
"colorthief": "^2.6.0",
|
||||||
|
"lucide-react": "^0.469.0",
|
||||||
|
"next": "^15.0.3",
|
||||||
|
"react": "^19",
|
||||||
|
"react-dom": "^19",
|
||||||
|
"react-hook-form": "^7.53.2",
|
||||||
|
"react-icons": "^5.3.0",
|
||||||
|
"tailwind-merge": "^2.5.4",
|
||||||
|
"tailwindcss-animate": "^1.0.7",
|
||||||
|
"zod": "^3.23.8"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/jest": "^29.5.14",
|
||||||
|
"@types/node": "^22.10.4",
|
||||||
|
"@types/react": "^19.0.4",
|
||||||
|
"@types/react-dom": "^19.0.2",
|
||||||
|
"eslint": "^9.17",
|
||||||
|
"eslint-config-next": "15.1.4",
|
||||||
|
"jest": "^29.7.0",
|
||||||
|
"postcss": "^8",
|
||||||
|
"tailwindcss": "^3.4.15",
|
||||||
|
"ts-jest": "^29.2.5",
|
||||||
|
"typescript": "^5"
|
||||||
|
},
|
||||||
|
"packageManager": "pnpm@9.15.3+sha512.1f79bc245a66eb0b07c5d4d83131240774642caaa86ef7d0434ab47c0d16f66b04e21e0c086eb61e62c77efc4d7f7ec071afad3796af64892fae66509173893a"
|
||||||
|
}
|
||||||
7093
pnpm-lock.yaml
generated
Normal file
8
postcss.config.mjs
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
/** @type {import('postcss-load-config').Config} */
|
||||||
|
const config = {
|
||||||
|
plugins: {
|
||||||
|
tailwindcss: {},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default config;
|
||||||
BIN
preview.png
Normal file
|
After Width: | Height: | Size: 483 KiB |
BIN
public/apple-touch-icon.png
Normal file
|
After Width: | Height: | Size: 3.4 KiB |
BIN
public/buh.mp3
Normal file
BIN
public/default-user.jpg
Normal file
|
After Width: | Height: | Size: 1.6 KiB |
BIN
public/favicon.ico
Normal file
|
After Width: | Height: | Size: 15 KiB |
BIN
public/icon-192-maskable.png
Normal file
|
After Width: | Height: | Size: 3.6 KiB |
BIN
public/icon-192.png
Normal file
|
After Width: | Height: | Size: 4.7 KiB |
BIN
public/icon-512-maskable.png
Normal file
|
After Width: | Height: | Size: 10 KiB |
BIN
public/icon-512.png
Normal file
|
After Width: | Height: | Size: 14 KiB |
BIN
public/play.png
Normal file
|
After Width: | Height: | Size: 6.6 KiB |
BIN
public/screenshot.png
Normal file
|
After Width: | Height: | Size: 2.4 MiB |
BIN
public/splash.png
Normal file
|
After Width: | Height: | Size: 172 KiB |
67
tailwind.config.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-require-imports */
|
||||||
|
import type { Config } from "tailwindcss";
|
||||||
|
|
||||||
|
const config: Config = {
|
||||||
|
darkMode: "media",
|
||||||
|
content: [
|
||||||
|
"./pages/**/*.{js,ts,jsx,tsx,mdx}",
|
||||||
|
"./components/**/*.{js,ts,jsx,tsx,mdx}",
|
||||||
|
"./app/**/*.{js,ts,jsx,tsx,mdx}",
|
||||||
|
],
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
colors: {
|
||||||
|
background: 'hsl(var(--background))',
|
||||||
|
foreground: 'hsl(var(--foreground))',
|
||||||
|
card: {
|
||||||
|
DEFAULT: 'hsl(var(--card))',
|
||||||
|
foreground: 'hsl(var(--card-foreground))'
|
||||||
|
},
|
||||||
|
popover: {
|
||||||
|
DEFAULT: 'hsl(var(--popover))',
|
||||||
|
foreground: 'hsl(var(--popover-foreground))'
|
||||||
|
},
|
||||||
|
primary: {
|
||||||
|
DEFAULT: 'hsl(var(--primary))',
|
||||||
|
foreground: 'hsl(var(--primary-foreground))'
|
||||||
|
},
|
||||||
|
secondary: {
|
||||||
|
DEFAULT: 'hsl(var(--secondary))',
|
||||||
|
foreground: 'hsl(var(--secondary-foreground))'
|
||||||
|
},
|
||||||
|
muted: {
|
||||||
|
DEFAULT: 'hsl(var(--muted))',
|
||||||
|
foreground: 'hsl(var(--muted-foreground))'
|
||||||
|
},
|
||||||
|
accent: {
|
||||||
|
DEFAULT: 'hsl(var(--accent))',
|
||||||
|
foreground: 'hsl(var(--accent-foreground))'
|
||||||
|
},
|
||||||
|
destructive: {
|
||||||
|
DEFAULT: 'hsl(var(--destructive))',
|
||||||
|
foreground: 'hsl(var(--destructive-foreground))'
|
||||||
|
},
|
||||||
|
border: 'hsl(var(--border))',
|
||||||
|
input: 'hsl(var(--input))',
|
||||||
|
ring: 'hsl(var(--ring))',
|
||||||
|
hover: 'hsl(var(--hover))',
|
||||||
|
chart: {
|
||||||
|
'1': 'hsl(var(--chart-1))',
|
||||||
|
'2': 'hsl(var(--chart-2))',
|
||||||
|
'3': 'hsl(var(--chart-3))',
|
||||||
|
'4': 'hsl(var(--chart-4))',
|
||||||
|
'5': 'hsl(var(--chart-5))'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
borderRadius: {
|
||||||
|
lg: 'var(--radius)',
|
||||||
|
md: 'calc(var(--radius) - 2px)',
|
||||||
|
sm: 'calc(var(--radius) - 4px)'
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
plugins: [
|
||||||
|
require("tailwindcss-animate")
|
||||||
|
],
|
||||||
|
};
|
||||||
|
export default config;
|
||||||
40
tsconfig.json
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"lib": [
|
||||||
|
"dom",
|
||||||
|
"dom.iterable",
|
||||||
|
"esnext"
|
||||||
|
],
|
||||||
|
"allowJs": true,
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"strict": true,
|
||||||
|
"noEmit": true,
|
||||||
|
"esModuleInterop": true,
|
||||||
|
"module": "commonjs", // Changed to commonjs for Jest compatibility
|
||||||
|
"moduleResolution": "node",
|
||||||
|
"resolveJsonModule": true,
|
||||||
|
"isolatedModules": true,
|
||||||
|
"jsx": "preserve",
|
||||||
|
"incremental": true,
|
||||||
|
"plugins": [
|
||||||
|
{
|
||||||
|
"name": "next"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"paths": {
|
||||||
|
"@/*": [
|
||||||
|
"./*"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"target": "ES2017"
|
||||||
|
},
|
||||||
|
"include": [
|
||||||
|
"next-env.d.ts",
|
||||||
|
"**/*.ts",
|
||||||
|
"**/*.tsx",
|
||||||
|
".next/types/**/*.ts"
|
||||||
|
],
|
||||||
|
"exclude": [
|
||||||
|
"node_modules"
|
||||||
|
]
|
||||||
|
}
|
||||||