Files
SAVExSTATE/lib/music_vault_view.dart

85 lines
3.7 KiB
Dart

import 'package:flutter/material.dart';
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:firebase_auth/firebase_auth.dart';
import 'player_service.dart'; // REQUIRED
class MusicVaultView extends StatefulWidget {
const MusicVaultView({super.key});
@override
State<MusicVaultView> createState() => _MusicVaultViewState();
}
class _MusicVaultViewState extends State<MusicVaultView> {
static const Color terminalGreen = Color(0xFFE87D25);
int _userTier = 1;
@override
void initState() {
super.initState();
_fetchUserTier();
}
Future<void> _fetchUserTier() async {
final user = FirebaseAuth.instance.currentUser;
if (user != null) {
final doc = await FirebaseFirestore.instance.collection('users').doc(user.uid).get();
if (mounted) setState(() => _userTier = doc.data()?['tier'] ?? 1);
}
}
@override
Widget build(BuildContext context) {
return StreamBuilder<QuerySnapshot>(
stream: FirebaseFirestore.instance.collection('tracks').orderBy('uploadedAt', descending: true).snapshots(),
builder: (context, snapshot) {
if (!snapshot.hasData) return const Center(child: CircularProgressIndicator(color: terminalGreen));
final tracks = snapshot.data!.docs;
if (tracks.isEmpty) return const Center(child: Text("ARCHIVE_NULL / AWAITING_DROP"));
return ListView.builder(
itemCount: tracks.length,
padding: const EdgeInsets.only(top: 10, bottom: 20),
itemBuilder: (context, index) {
final t = tracks[index].data() as Map<String, dynamic>;
final String docId = tracks[index].id;
// Check if this specific track is the one playing
return ValueListenableBuilder(
valueListenable: PlayerService().currentTrack,
builder: (context, activeTrack, child) {
final bool isCurrent = activeTrack != null && activeTrack['audioUrl'] == t['audioUrl'];
return Container(
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 4),
decoration: BoxDecoration(
color: isCurrent ? terminalGreen.withValues(alpha: 0.05) : Colors.transparent,
border: Border(left: BorderSide(color: isCurrent ? terminalGreen : Colors.white12, width: 2)),
),
child: ListTile(
leading: Container(
width: 45, height: 45,
decoration: BoxDecoration(border: Border.all(color: isCurrent ? terminalGreen : Colors.white10)),
child: Image.network(
"${t['imageUrl']}${t['imageUrl'].contains('?') ? '&' : '?'}t=${DateTime.now().millisecondsSinceEpoch}",
fit: BoxFit.cover,
errorBuilder: (context, e, s) => const Icon(Icons.audiotrack, color: terminalGreen, size: 20),
),
),
title: Text(
"${t['title'].toString().toUpperCase()}.${(t['fileName'] ?? 'MP3').split('.').last.toUpperCase()}",
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13, color: isCurrent ? terminalGreen : Colors.white),
),
subtitle: Text(isCurrent ? "SYSTEM_STATUS: RUNNING" : "SYSTEM_STATUS: ENCRYPTED", style: const TextStyle(fontSize: 9, color: Colors.white24)),
trailing: Icon(isCurrent ? Icons.graphic_eq : Icons.play_arrow_outlined, color: terminalGreen, size: 18),
onTap: () => PlayerService().play(t, _userTier),
),
);
},
);
},
);
},
);
}
}