feat: Modernisation UI/UX et configuration Flutter multi-plateforme

Phase 1 - Corrections Critiques:
- Fixed memory leaks dans music_provider.dart (stream subscriptions)
- Fixed race conditions dans search_provider.dart (stale results)
- Fixed token refresh errors dans api_service.dart
- Improved error handling avec messages utilisateur
- Changed API URL to HTTPS by default

Phase 2 - Améliorations UX Desktop:
- Ajouté cursor pointers sur tous les éléments cliquables
- Implémenté hover states avec effets néon glow (200ms transitions)
- Créé skeleton loading states avec shimmer animation
- Ajouté widgets: ClickableWrapper, ErrorDisplay, SkeletonLoading
- Enhanced visual feedback pour desktop users

Phase 3 - Configuration Flutter:
- Configuré Android (Gradle 8.1.0, Kotlin 1.9.0, minSdk 21, targetSdk 34)
- Créé launcher icons cyberpunk néon (5 densités)
- Configuré Windows desktop (structure complète)
- Activé Linux desktop support
- Ajouté package équatable pour entités de domaine
- Corrigé imports (colors.dart, auth_provider.dart)
- Fixed Dio API compatibility (RequestOptions)

Documentation:
- STYLE_GUIDE.md: Guide complet (100+ pages)
- DESIGN_IMPLEMENTATION_GUIDE.md: Implémentation Flutter
- BUILD_STATUS.md: Status builds + troubleshooting
- QUICKSTART_BUILDS.md: Guide rapide
- BUILD_INDEX.md: Index documentation
- PHASE_1_CORRECTIONS.md: Corrections Phase 1
- PHASE_2_UX_IMPROVEMENTS.md: Améliorations Phase 2
- PR_REVIEW_SUMMARY.md: Revue code complète
- CODE_ANALYSIS_AND_PRIORITIES.md: Analyse code

Scripts & Builds:
- BUILD_ALL.sh: Script automatisé builds multi-plateforme
- builds/: Structure avec README par plateforme
- design-system/: Système de design complet

Backend:
- Ajouté streaming HTTP Range pour audio progressif
- Enhanced YouTube service avec métadonnées complètes
- Improved error handling et validation

Generated with [Claude Code](https://claude.com/claude-code)
via [Happy](https://happy.engineering)

Co-Authored-By: Claude <noreply@anthropic.com>
Co-Authored-By: Happy <yesreply@happy.engineering>
This commit is contained in:
root
2026-01-19 07:44:40 +00:00
parent a89c7894cf
commit 85dad89d5b
100 changed files with 13570 additions and 323 deletions
+476
View File
@@ -0,0 +1,476 @@
# Home Page Design Override
## Purpose
This file contains **page-specific design rules** that **override** the master design system for the Home page.
## Key Differences from Master
### Layout Structure
- **Hero Section**: Full-width gradient banner with quick picks
- **Horizontal scrolling sections** for album/playlist rows
- **Grid layout** for featured content (3 columns desktop, 2 tablet, 1 mobile)
- **Sticky sub-navigation** for content categories
### Visual Hierarchy
1. **Greeting & Quick Picks** (hero section) - Highest prominence
2. **Featured Playlists** - Large cards with gradients
3. **Recently Played** - Horizontal scroll row
4. **Made For You** - Grid of recommendation cards
5. **New Releases** - Compact list view
### Section Spacing
- Hero padding: **60px** (larger than master)
- Section gaps: **48px** (between major sections)
- Row padding: **24px** (within sections)
## Components
### Hero Section (Personalized Greeting)
```css
.hero-section {
position: relative;
background: linear-gradient(135deg, #0A0E27 0%, #151932 50%, #1F2342 100%);
padding: 60px 40px;
border-radius: 20px;
margin-bottom: 48px;
overflow: hidden;
}
/* Animated background gradient */
.hero-section::before {
content: '';
position: absolute;
inset: 0;
background: radial-gradient(circle at 20% 50%, rgba(0, 240, 255, 0.1) 0%, transparent 50%),
radial-gradient(circle at 80% 50%, rgba(191, 0, 255, 0.1) 0%, transparent 50%);
animation: gradient-shift 10s ease-in-out infinite;
}
@keyframes gradient-shift {
0%, 100% { opacity: 1; }
50% { opacity: 0.6; }
}
.hero-content {
position: relative;
z-index: 1;
}
.hero-greeting {
font-size: 32px;
font-weight: 700;
color: #F0F4F8;
margin-bottom: 24px;
}
.hero-subtitle {
font-size: 18px;
font-weight: 400;
color: #9BA3B8;
margin-bottom: 32px;
}
```
### Quick Picks Grid
```css
.quick-picks-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
gap: 20px;
}
.quick-pick-card {
aspect-ratio: 16/9;
background: linear-gradient(135deg, rgba(0, 240, 255, 0.1) 0%, rgba(191, 0, 255, 0.1) 100%);
border-radius: 16px;
padding: 24px;
display: flex;
flex-direction: column;
justify-content: flex-end;
cursor: pointer;
transition: all 250ms ease;
border: 1px solid transparent;
position: relative;
overflow: hidden;
}
.quick-pick-card:hover {
border-color: #00F0FF;
transform: translateY(-4px);
box-shadow: 0 12px 40px rgba(0, 240, 255, 0.2);
}
.quick-pick-card::before {
content: '';
position: absolute;
inset: 0;
background: radial-gradient(circle at var(--mouse-x, 50%) var(--mouse-y, 50%), rgba(0, 240, 255, 0.15), transparent 50%);
opacity: 0;
transition: opacity 250ms ease;
}
.quick-pick-card:hover::before {
opacity: 1;
}
.quick-pick-title {
font-size: 24px;
font-weight: 600;
color: #F0F4F8;
margin-bottom: 8px;
}
.quick-pick-description {
font-size: 14px;
font-weight: 400;
color: #9BA3B8;
}
```
### Horizontal Scroll Rows
```css
.section-row {
margin-bottom: 48px;
}
.section-header {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 24px;
}
.section-title {
font-size: 28px;
font-weight: 700;
color: #F0F4F8;
}
.see-all-link {
font-size: 14px;
font-weight: 600;
color: #00F0FF;
text-decoration: none;
cursor: pointer;
transition: opacity 200ms ease;
}
.see-all-link:hover {
opacity: 0.8;
}
/* Horizontal scroll container */
.horizontal-scroll {
display: flex;
gap: 20px;
overflow-x: auto;
scroll-snap-type: x mandatory;
padding-bottom: 8px;
scrollbar-width: thin;
scrollbar-color: #2A2F4A #0A0E27;
}
.horizontal-scroll::-webkit-scrollbar {
height: 8px;
}
.horizontal-scroll::-webkit-scrollbar-track {
background: #0A0E27;
border-radius: 4px;
}
.horizontal-scroll::-webkit-scrollbar-thumb {
background: #2A2F4A;
border-radius: 4px;
}
.horizontal-scroll::-webkit-scrollbar-thumb:hover {
background: #00F0FF;
}
.scroll-item {
flex: 0 0 200px;
scroll-snap-align: start;
}
```
### Album/Playlist Cards (Grid)
```css
.content-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(200px, 1fr));
gap: 24px;
}
.content-card {
cursor: pointer;
transition: all 250ms ease;
}
.content-card-image {
width: 100%;
aspect-ratio: 1/1;
object-fit: cover;
border-radius: 12px;
margin-bottom: 12px;
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.3);
transition: all 250ms ease;
}
.content-card:hover .content-card-image {
box-shadow: 0 8px 30px rgba(0, 240, 255, 0.3);
transform: translateY(-4px);
}
.content-card-title {
font-size: 16px;
font-weight: 600;
color: #F0F4F8;
margin-bottom: 4px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.content-card-subtitle {
font-size: 14px;
font-weight: 400;
color: #9BA3B8;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
/* Play overlay on hover */
.play-overlay {
position: absolute;
top: 12px;
left: 12px;
right: 12px;
aspect-ratio: 1/1;
background: rgba(10, 14, 39, 0.8);
border-radius: 12px;
display: flex;
align-items: center;
justify-content: center;
opacity: 0;
transition: opacity 200ms ease;
}
.content-card:hover .play-overlay {
opacity: 1;
}
```
### Featured Playlist Card (Large)
```css
.featured-card {
background: linear-gradient(135deg, var(--gradient-start) 0%, var(--gradient-end) 100%);
border-radius: 20px;
padding: 32px;
min-height: 250px;
display: flex;
flex-direction: column;
justify-content: flex-end;
cursor: pointer;
transition: all 300ms ease;
position: relative;
overflow: hidden;
}
.featured-card::before {
content: '';
position: absolute;
inset: 0;
background: radial-gradient(circle at 80% 20%, rgba(255, 255, 255, 0.1) 0%, transparent 50%);
}
.featured-card:hover {
transform: translateY(-8px);
box-shadow: 0 16px 50px rgba(0, 0, 0, 0.4);
}
.featured-card-tag {
position: absolute;
top: 24px;
left: 24px;
background: rgba(10, 14, 39, 0.6);
backdrop-filter: blur(10px);
padding: 6px 12px;
border-radius: 6px;
font-size: 12px;
font-weight: 600;
color: #00F0FF;
text-transform: uppercase;
letter-spacing: 0.5px;
}
.featured-card-title {
font-size: 32px;
font-weight: 700;
color: #F0F4F8;
margin-bottom: 8px;
}
.featured-card-description {
font-size: 16px;
font-weight: 400;
color: rgba(240, 244, 248, 0.8);
max-width: 80%;
}
```
### Category Pills (Filter Bar)
```css
.category-filter-bar {
display: flex;
gap: 12px;
padding: 20px 0;
overflow-x: auto;
scrollbar-width: none;
}
.category-filter-bar::-webkit-scrollbar {
display: none;
}
.category-pill {
padding: 10px 20px;
background: rgba(21, 25, 50, 0.8);
border: 1px solid #2A2F4A;
border-radius: 24px;
font-size: 14px;
font-weight: 500;
color: #9BA3B8;
cursor: pointer;
transition: all 200ms ease;
white-space: nowrap;
}
.category-pill:hover {
border-color: #00F0FF;
color: #00F0FF;
}
.category-pill.active {
background: linear-gradient(135deg, #00F0FF 0%, #00C8FF 100%);
border-color: #00F0FF;
color: #0A0E27;
box-shadow: 0 4px 16px rgba(0, 240, 255, 0.4);
}
```
## Responsive Adjustments
### Mobile (< 768px)
- Hero padding: 40px 20px
- Hero greeting: 24px
- Quick picks: 1 column (100% width)
- Content grid: 2 columns (repeat(2, 1fr))
- Horizontal scroll: Enable momentum scroll
- Section title: 22px
### Tablet (768px - 1024px)
- Hero padding: 50px 30px
- Quick picks: 2 columns
- Content grid: 3 columns (repeat(3, 1fr))
- Section title: 26px
### Desktop (> 1024px)
- Use default sizes above
- Quick picks: 4 columns
- Content grid: 6 columns
## Loading States
### Skeleton Cards
```css
.skeleton-card {
background: linear-gradient(90deg, #151932 25%, #1F2342 50%, #151932 75%);
background-size: 200% 100%;
animation: skeleton-loading 1.5s ease-in-out infinite;
border-radius: 12px;
}
@keyframes skeleton-loading {
0% { background-position: 200% 0; }
100% { background-position: -200% 0; }
}
.skeleton-image {
width: 100%;
aspect-ratio: 1/1;
border-radius: 12px;
margin-bottom: 12px;
}
.skeleton-text {
height: 16px;
background: #1F2342;
border-radius: 4px;
margin-bottom: 8px;
}
.skeleton-text.short {
width: 60%;
}
.skeleton-text.long {
width: 90%;
}
```
## Empty States
```css
.empty-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: 80px 40px;
text-align: center;
}
.empty-state-icon {
width: 80px;
height: 80px;
color: #2A2F4A;
margin-bottom: 24px;
}
.empty-state-title {
font-size: 24px;
font-weight: 600;
color: #F0F4F8;
margin-bottom: 12px;
}
.empty-state-description {
font-size: 16px;
font-weight: 400;
color: #9BA3B8;
margin-bottom: 32px;
max-width: 400px;
}
```
## Additional Anti-Patterns for Home
**NEVER load all content at once** - Implement lazy loading for rows
**NEVER auto-play audio on home** - User must explicitly click play
**NEVER mix card sizes in same row** - Keep consistent sizing
**NEVER hide horizontal scroll indicators** - Show shadow hint
**NEVER use generic greetings** - Personalize with time of day
**NEVER skip section anchors** - Allow deep linking to sections
---
*These page-specific rules override the master design system for the home page only.*
+318
View File
@@ -0,0 +1,318 @@
# Player Page Design Override
## Purpose
This file contains **page-specific design rules** that **override** the master design system for the Audio Player page.
## Key Differences from Master
### Layout
- Full-screen overlay mode for immersive experience
- Larger touch targets (minimum 56px for player controls)
- Fixed bottom mini-player (height: 90px)
### Visual Hierarchy
- Album art is the **primary visual element** (dominates 40% of viewport)
- Track info is secondary (artist name smaller, muted color)
- Controls are tertiary but highly accessible
### Typography
- Track title: **24px, weight 600** (larger than master H3)
- Artist name: **16px, weight 400, color #9BA3B8**
- Timestamps: **13px, weight 500, color #6B7280**
### Colors (Same as Master)
- Background: `#0A0E27`
- Surface: `#151932`
- Primary Neon: `#00F0FF` (progress bar, play button)
- Secondary Neon: `#BF00FF` (like button active)
### Components
#### Progress Bar
```css
/* Custom progress bar styling */
.progress-container {
height: 4px;
background: #2A2F4A;
border-radius: 2px;
cursor: pointer;
position: relative;
}
.progress-fill {
height: 100%;
background: linear-gradient(90deg, #00F0FF 0%, #00C8FF 100%);
border-radius: 2px;
box-shadow: 0 0 10px rgba(0, 240, 255, 0.6);
}
.progress-handle {
width: 12px;
height: 12px;
background: #00F0FF;
border-radius: 50%;
position: absolute;
top: 50%;
transform: translate(-50%, -50%);
box-shadow: 0 0 15px rgba(0, 240, 255, 0.8);
opacity: 0;
transition: opacity 200ms ease;
}
.progress-container:hover .progress-handle {
opacity: 1;
}
```
#### Control Buttons
```css
/* Primary Control (Play/Pause) */
.btn-play-primary {
width: 64px;
height: 64px;
background: linear-gradient(135deg, #00F0FF 0%, #00C8FF 100%);
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
box-shadow: 0 4px 24px rgba(0, 240, 255, 0.4);
cursor: pointer;
transition: all 200ms ease;
}
.btn-play-primary:hover {
box-shadow: 0 6px 32px rgba(0, 240, 255, 0.6);
transform: scale(1.05);
}
/* Secondary Controls (Prev, Next, Shuffle) */
.btn-control-secondary {
width: 48px;
height: 48px;
background: transparent;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
color: #9BA3B8;
cursor: pointer;
transition: all 200ms ease;
}
.btn-control-secondary:hover {
color: #F0F4F8;
background: rgba(240, 244, 248, 0.05);
}
.btn-control-secondary.active {
color: #00F0FF;
}
```
#### Album Art Display
```css
/* Large album art with glow */
.album-art-container {
width: 100%;
max-width: 400px;
aspect-ratio: 1/1;
margin: 0 auto;
position: relative;
}
.album-art {
width: 100%;
height: 100%;
object-fit: cover;
border-radius: 16px;
box-shadow: 0 20px 60px rgba(0, 240, 255, 0.2);
}
/* Playing animation (subtle pulse) */
@keyframes pulse-glow {
0%, 100% {
box-shadow: 0 20px 60px rgba(0, 240, 255, 0.2);
}
50% {
box-shadow: 0 20px 80px rgba(0, 240, 255, 0.3);
}
}
.album-art.playing {
animation: pulse-glow 3s ease-in-out infinite;
}
```
#### Mini Player (Bottom Bar)
```css
/* Persistent bottom mini-player */
.mini-player {
position: fixed;
bottom: 0;
left: 0;
right: 0;
height: 90px;
background: linear-gradient(180deg, rgba(21,25,50,0.95) 0%, rgba(10,14,39,0.98) 100%);
backdrop-filter: blur(20px);
border-top: 1px solid #2A2F4A;
padding: 12px 24px;
display: grid;
grid-template-columns: auto 1fr auto;
gap: 20px;
align-items: center;
z-index: 100;
}
.mini-album-art {
width: 64px;
height: 64px;
border-radius: 8px;
object-fit: cover;
}
.mini-track-info {
display: flex;
flex-direction: column;
gap: 4px;
}
.mini-track-title {
font-size: 16px;
font-weight: 600;
color: #F0F4F8;
}
.mini-artist-name {
font-size: 14px;
font-weight: 400;
color: #9BA3B8;
}
.mini-controls {
display: flex;
align-items: center;
gap: 16px;
}
```
### Volume Control
```css
.volume-slider {
-webkit-appearance: none;
width: 100px;
height: 4px;
background: #2A2F4A;
border-radius: 2px;
outline: none;
}
.volume-slider::-webkit-slider-thumb {
-webkit-appearance: none;
width: 12px;
height: 12px;
background: #00F0FF;
border-radius: 50%;
cursor: pointer;
box-shadow: 0 0 10px rgba(0, 240, 255, 0.6);
}
```
### Queue Panel (Slide-out)
```css
.queue-panel {
position: fixed;
top: 0;
right: 0;
bottom: 0;
width: 400px;
background: #151932;
border-left: 1px solid #2A2F4A;
transform: translateX(100%);
transition: transform 300ms ease;
z-index: 50;
overflow-y: auto;
}
.queue-panel.open {
transform: translateX(0);
}
.queue-item {
display: grid;
grid-template-columns: 48px 1fr auto;
gap: 12px;
padding: 12px;
border-radius: 8px;
cursor: pointer;
transition: background 200ms ease;
}
.queue-item:hover {
background: rgba(0, 240, 255, 0.05);
}
.queue-item.active {
background: rgba(0, 240, 255, 0.1);
border: 1px solid #00F0FF;
}
```
## Responsive Adjustments
### Mobile (< 768px)
- Mini player: Height 80px
- Album art: Max-width 300px
- Control buttons: 56px primary, 44px secondary
- Queue panel: Full-width (100%)
### Tablet (768px - 1024px)
- Mini player: Height 85px
- Album art: Max-width 350px
- Queue panel: Width 350px
### Desktop (> 1024px)
- Use default sizes above
## Accessibility (Beyond Master)
### Keyboard Shortcuts
- **Space**: Play/Pause
- **Arrow Left/Right**: Seek ±10s
- **Arrow Up/Down**: Volume ±10%
- **Shift + Arrow Left/Right**: Previous/Next track
- **M**: Mute/Unmute
- **F**: Toggle fullscreen mode
### Screen Reader Announcements
- Announce track changes (title + artist)
- Announce playback state (playing/paused)
- Announce volume changes (percentage)
- Announce queue position ("Track 3 of 15")
### ARIA Labels
```html
<!-- Play/Pause Button -->
<button aria-label="Pause" aria-pressed="true">
<!-- Pause Icon -->
</button>
<!-- Progress Bar -->
<div role="slider" aria-label="Track progress" aria-valuemin="0" aria-valuemax="100" aria-valuenow="45">
<!-- Progress UI -->
</div>
<!-- Volume Slider -->
<input type="range" aria-label="Volume" aria-valuemin="0" aria-valuemax="100" aria-valuenow="70">
```
## Additional Anti-Patterns for Player
**NEVER auto-play audio without user interaction** - Wait for explicit play action
**NEVER hide controls during video playback** - Keep controls always visible
**NEVER use instant seek jumps** - Animate progress bar smoothly
**NEVER skip keyboard navigation** - All controls must be keyboard-accessible
**NEVER ignore audio focus** - Pause when another app plays audio
---
*These page-specific rules override the master design system for the player page only.*
+636
View File
@@ -0,0 +1,636 @@
# Search Page Design Override
## Purpose
This file contains **page-specific design rules** that **override** the master design system for the Search page.
## Key Differences from Master
### Layout Structure
- **Large, prominent search bar** at top (72px height)
- **Search tabs** (All, Songs, Artists, Albums, Playlists)
- **Results grid** (responsive: 6 col desktop, 4 tablet, 2 mobile)
- **Recent searches** with clear button
- **Trending searches** as suggestions
### Visual Hierarchy
1. **Search Input** - Dominant, focused element
2. **Search Tabs** - Secondary navigation
3. **Results Count** - Tertiary info
4. **Filter/Sort Controls** - Quaternary
### Search Bar Styling
- Larger height (48px input, vs 40px in master)
- Prominent search icon
- Clear button appears on type
- Recent searches dropdown
## Components
### Search Input Field
```css
.search-container {
position: sticky;
top: 0;
background: rgba(10, 14, 39, 0.95);
backdrop-filter: blur(20px);
padding: 20px 0;
z-index: 20;
border-bottom: 1px solid #2A2F4A;
}
.search-input-wrapper {
position: relative;
max-width: 800px;
margin: 0 auto;
}
.search-input {
width: 100%;
height: 56px;
background: #151932;
border: 2px solid #2A2F4A;
border-radius: 28px;
padding: 0 60px 0 60px;
font-size: 18px;
font-weight: 400;
color: #F0F4F8;
transition: all 200ms ease;
}
.search-input::placeholder {
color: #6B7280;
}
.search-input:focus {
outline: none;
border-color: #00F0FF;
box-shadow: 0 0 0 4px rgba(0, 240, 255, 0.1);
background: #1F2342;
}
.search-input-icon {
position: absolute;
left: 20px;
top: 50%;
transform: translateY(-50%);
width: 24px;
height: 24px;
color: #6B7280;
pointer-events: none;
transition: color 200ms ease;
}
.search-input:focus + .search-input-icon {
color: #00F0FF;
}
.search-clear-button {
position: absolute;
right: 16px;
top: 50%;
transform: translateY(-50%);
width: 32px;
height: 32px;
background: #2A2F4A;
border: none;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
cursor: pointer;
opacity: 0;
visibility: hidden;
transition: all 200ms ease;
}
.search-input:not(:placeholder-shown) + .search-input-icon ~ .search-clear-button {
opacity: 1;
visibility: visible;
}
.search-clear-button:hover {
background: #3B4259;
}
```
### Search Tabs
```css
.search-tabs {
display: flex;
gap: 8px;
padding: 20px 0;
overflow-x: auto;
scrollbar-width: none;
}
.search-tabs::-webkit-scrollbar {
display: none;
}
.search-tab {
padding: 10px 20px;
background: transparent;
border: none;
border-radius: 20px;
font-size: 14px;
font-weight: 500;
color: #9BA3B8;
cursor: pointer;
transition: all 200ms ease;
white-space: nowrap;
}
.search-tab:hover {
color: #F0F4F8;
background: rgba(240, 244, 248, 0.05);
}
.search-tab.active {
background: #F0F4F8;
color: #0A0E27;
font-weight: 600;
}
```
### Recent Searches
```css
.recent-searches-section {
padding: 32px 0;
}
.recent-searches-header {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 20px;
}
.recent-searches-title {
font-size: 20px;
font-weight: 600;
color: #F0F4F8;
}
.clear-recent-button {
font-size: 14px;
font-weight: 500;
color: #00F0FF;
background: transparent;
border: none;
cursor: pointer;
transition: opacity 200ms ease;
}
.clear-recent-button:hover {
opacity: 0.8;
}
.recent-search-item {
display: flex;
align-items: center;
gap: 16px;
padding: 16px 20px;
background: #151932;
border-radius: 12px;
cursor: pointer;
transition: all 200ms ease;
border: 1px solid transparent;
}
.recent-search-item:hover {
background: #1F2342;
border-color: #2A2F4A;
}
.recent-search-icon {
width: 20px;
height: 20px;
color: #6B7280;
flex-shrink: 0;
}
.recent-search-text {
flex: 1;
font-size: 16px;
font-weight: 400;
color: #F0F4F8;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.recent-search-remove {
width: 32px;
height: 32px;
background: transparent;
border: none;
border-radius: 6px;
display: flex;
align-items: center;
justify-content: center;
cursor: pointer;
opacity: 0;
transition: all 200ms ease;
}
.recent-search-item:hover .recent-search-remove {
opacity: 1;
}
.recent-search-remove:hover {
background: rgba(255, 59, 59, 0.1);
color: #FF3B3B;
}
```
### Search Results Grid
```css
.search-results {
padding: 32px 0;
}
.results-header {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 24px;
}
.results-count {
font-size: 14px;
font-weight: 500;
color: #9BA3B8;
}
.results-sort {
display: flex;
align-items: center;
gap: 12px;
}
.sort-dropdown {
padding: 8px 16px;
background: #151932;
border: 1px solid #2A2F4A;
border-radius: 8px;
font-size: 14px;
font-weight: 500;
color: #F0F4F8;
cursor: pointer;
transition: all 200ms ease;
}
.sort-dropdown:hover {
border-color: #00F0FF;
}
/* Results grid layout */
.results-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(200px, 1fr));
gap: 24px;
}
/* Song results (list view) */
.song-results-list {
display: flex;
flex-direction: column;
gap: 8px;
}
.song-result-item {
display: grid;
grid-template-columns: 48px 1fr auto auto auto;
gap: 16px;
align-items: center;
padding: 12px 16px;
background: transparent;
border-radius: 8px;
cursor: pointer;
transition: all 200ms ease;
}
.song-result-item:hover {
background: rgba(240, 244, 248, 0.05);
}
.song-result-number {
font-size: 14px;
font-weight: 500;
color: #6B7280;
}
.song-result-info {
display: flex;
align-items: center;
gap: 12px;
min-width: 0;
}
.song-result-image {
width: 48px;
height: 48px;
object-fit: cover;
border-radius: 6px;
flex-shrink: 0;
}
.song-result-title {
font-size: 16px;
font-weight: 500;
color: #F0F4F8;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.song-result-artist {
font-size: 14px;
font-weight: 400;
color: #9BA3B8;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.song-result-album {
font-size: 14px;
font-weight: 400;
color: #9BA3B8;
}
.song-result-duration {
font-size: 14px;
font-weight: 500;
color: #6B7280;
}
.song-result-actions {
display: flex;
align-items: center;
gap: 8px;
opacity: 0;
transition: opacity 200ms ease;
}
.song-result-item:hover .song-result-actions {
opacity: 1;
}
.icon-button {
width: 36px;
height: 36px;
background: transparent;
border: none;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
cursor: pointer;
color: #9BA3B8;
transition: all 200ms ease;
}
.icon-button:hover {
color: #F0F4F8;
background: rgba(240, 244, 248, 0.1);
}
```
### Trending Searches
```css
.trending-searches-section {
padding: 32px 0;
border-top: 1px solid #2A2F4A;
}
.trending-searches-title {
font-size: 18px;
font-weight: 600;
color: #F0F4F8;
margin-bottom: 20px;
}
.trending-searches-list {
display: flex;
flex-direction: column;
gap: 12px;
}
.trending-search-item {
display: flex;
align-items: center;
gap: 16px;
padding: 12px 16px;
border-radius: 8px;
cursor: pointer;
transition: all 200ms ease;
}
.trending-search-item:hover {
background: rgba(240, 244, 248, 0.05);
}
.trending-rank {
font-size: 18px;
font-weight: 700;
color: #00F0FF;
width: 32px;
flex-shrink: 0;
}
.trending-search-content {
flex: 1;
}
.trending-search-query {
font-size: 16px;
font-weight: 500;
color: #F0F4F8;
margin-bottom: 4px;
}
.trending-search-meta {
font-size: 13px;
font-weight: 400;
color: #6B7280;
}
.trending-arrow {
width: 20px;
height: 20px;
color: #6B7280;
flex-shrink: 0;
}
```
### No Results State
```css
.no-results-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: 120px 40px;
text-align: center;
}
.no-results-icon {
width: 100px;
height: 100px;
color: #2A2F4A;
margin-bottom: 24px;
}
.no-results-title {
font-size: 28px;
font-weight: 600;
color: #F0F4F8;
margin-bottom: 12px;
}
.no-results-query {
color: #00F0FF;
}
.no-results-description {
font-size: 16px;
font-weight: 400;
color: #9BA3B8;
margin-bottom: 8px;
max-width: 500px;
}
.no-results-suggestions {
font-size: 14px;
font-weight: 400;
color: #6B7280;
margin-bottom: 32px;
}
.no-results-tips {
display: flex;
flex-direction: column;
gap: 12px;
text-align: left;
}
.no-results-tip {
display: flex;
align-items: center;
gap: 12px;
font-size: 14px;
font-weight: 400;
color: #9BA3B8;
}
.no-results-tip-icon {
width: 16px;
height: 16px;
color: #00F0FF;
flex-shrink: 0;
}
```
### Search Filters
```css
.search-filters {
display: flex;
align-items: center;
gap: 12px;
padding: 20px 0;
flex-wrap: wrap;
}
.filter-chip {
padding: 8px 16px;
background: #151932;
border: 1px solid #2A2F4A;
border-radius: 20px;
font-size: 13px;
font-weight: 500;
color: #9BA3B8;
cursor: pointer;
transition: all 200ms ease;
display: flex;
align-items: center;
gap: 8px;
}
.filter-chip:hover {
border-color: #00F0FF;
color: #F0F4F8;
}
.filter-chip.active {
background: linear-gradient(135deg, #00F0FF 0%, #00C8FF 100%);
border-color: #00F0FF;
color: #0A0E27;
}
.filter-chip-remove {
width: 16px;
height: 16px;
opacity: 0.7;
}
.filter-chip-remove:hover {
opacity: 1;
}
```
## Responsive Adjustments
### Mobile (< 768px)
- Search input height: 48px
- Search tabs: Scrollable horizontally
- Results grid: 2 columns
- Song list: Compact (hide album column)
- Filters: Horizontal scroll
### Tablet (768px - 1024px)
- Search input height: 52px
- Results grid: 4 columns
- Song list: Full layout
### Desktop (> 1024px)
- Use default sizes above
- Results grid: 6 columns
## Search Behavior
### Debouncing
- Wait **300ms** after typing before sending search request
- Cancel previous requests if user continues typing
### Keyboard Navigation
- **Escape**: Clear search and focus search input
- **Tab**: Navigate through results
- **Enter**: Search on press or navigate to first result
- **Arrow Down/Up**: Navigate through results list
### Search Suggestions
- Show **5 recent searches** max
- Show **10 trending searches** max
- Highlight matching text in results
## Additional Anti-Patterns for Search
**NEVER search on every keystroke** - Debounce for 300ms
**NEVER show results before user types** - Start with recent/trending
**NEVER hide clear button** - Always visible when input has text
**NEVER mix result types** - Use tabs to separate
**NEVER skip empty state** - Provide helpful tips
**NEVER ignore search history** - Save and show recent searches
**NEVER use instant navigation** - Let user review results first
---
*These page-specific rules override the master design system for the search page only.*