import 'dart:typed_data'; import 'package:flutter/material.dart'; import 'package:firebase_auth/firebase_auth.dart'; import 'package:cloud_firestore/cloud_firestore.dart'; import 'package:firebase_storage/firebase_storage.dart'; import 'package:image_picker/image_picker.dart'; import 'package:cached_network_image/cached_network_image.dart'; import 'package:onsolgo/core/constants.dart'; import 'package:onsolgo/screens/admin/admin_dashboard.dart'; import 'package:onsolgo/screens/profile/collection_view.dart'; import 'package:onsolgo/screens/profile/achievements_view.dart'; import 'package:onsolgo/screens/artist/artist_hub.dart'; class ProfileView extends StatefulWidget { const ProfileView({super.key}); @override State createState() => _ProfileViewState(); } class _ProfileViewState extends State { bool _isUploading = false; Uint8List? _localBytes; final ImagePicker _picker = ImagePicker(); Future _pickAndUpload(String uid) async { try { final XFile? image = await _picker.pickImage(source: ImageSource.gallery, imageQuality: 40, maxWidth: 500); if (image == null) return; final bytes = await image.readAsBytes(); setState(() { _localBytes = bytes; _isUploading = true; }); final ref = FirebaseStorage.instance.ref().child('avatars').child('$uid.jpg'); await ref.putData(bytes, SettableMetadata(contentType: 'image/jpeg')); final downloadUrl = await ref.getDownloadURL(); final sep = downloadUrl.contains('?') ? '&' : '?'; final hashedUrl = '$downloadUrl${sep}t=${DateTime.now().millisecondsSinceEpoch}'; await FirebaseFirestore.instance.collection('users').doc(uid).update({'pfpUrl': hashedUrl}); } finally { if (mounted) setState(() => _isUploading = false); } } @override Widget build(BuildContext context) { final String uid = FirebaseAuth.instance.currentUser?.uid ?? ""; return Scaffold( backgroundColor: Colors.black, appBar: AppBar(backgroundColor: Colors.black, elevation: 0, title: const Text("PROFILE")), body: SafeArea( child: StreamBuilder( stream: FirebaseFirestore.instance.collection('users').doc(uid).snapshots(), builder: (context, snapshot) { if (!snapshot.hasData || !snapshot.data!.exists) return const Center(child: CircularProgressIndicator()); var data = snapshot.data!.data() as Map; // --- WEB-SAFE DATA PARSING --- int rank = safeInt(data['rankLevel']); final int energy = safeInt(data['energy'] ?? 2); String role = data['role']?.toString().toLowerCase() ?? "reader"; String tier = data['tier']?.toString().toLowerCase() ?? "free"; String? pfp = data['pfpUrl']; return SingleChildScrollView( padding: const EdgeInsets.symmetric(horizontal: 20), child: Column( mainAxisSize: MainAxisSize.min, children: [ const SizedBox(height: 30), // AVATAR Center( child: Stack(alignment: Alignment.center, children: [ Container(width: 120, height: 120, decoration: BoxDecoration(shape: BoxShape.circle, border: Border.all(color: getRankColor(rank), width: 3)), child: ClipOval( child: _localBytes != null ? Image.memory(_localBytes!, fit: BoxFit.cover) : (pfp != null && pfp.isNotEmpty ? CachedNetworkImage( imageUrl: pfp, key: ValueKey(pfp), fit: BoxFit.cover, placeholder: (context, url) => const Center(child: SizedBox(width: 24, height: 24, child: CircularProgressIndicator(strokeWidth: 2))), errorWidget: (context, url, error) => const Icon(Icons.person, size: 60), ) : const Icon(Icons.person, size: 60)), ), ), Positioned.fill(child: Material(color: Colors.transparent, child: InkWell(borderRadius: BorderRadius.circular(100), onTap: () => _pickAndUpload(uid)))), if (_isUploading) const Positioned.fill( child: Center(child: SizedBox(width: 40, height: 40, child: CircularProgressIndicator(strokeWidth: 2, color: kOnsolGold))), ), ]), ), const SizedBox(height: 20), Text(data['username']?.toString().toUpperCase() ?? "CITIZEN", style: const TextStyle(fontSize: 24, fontWeight: FontWeight.bold)), // --- ENERGY BAR --- const SizedBox(height: 8), Row(mainAxisAlignment: MainAxisAlignment.center, children: [ const Icon(Icons.local_fire_department, color: Colors.orange, size: 18), Text("${data['streak'] ?? 0} DAY STREAK", style: const TextStyle(color: Colors.orange, fontWeight: FontWeight.bold, fontSize: 12)), const SizedBox(width: 15), const Icon(Icons.bolt, color: Colors.orangeAccent, size: 18), Text((tier != 'free' || rank == 5) ? "ENERGY: MAX" : "ENERGY: $energy/2", style: const TextStyle(color: Colors.orangeAccent, fontWeight: FontWeight.bold, fontSize: 12)), ]), Text(getRankName(rank), style: TextStyle(color: getRankColor(rank), letterSpacing: 2, fontWeight: FontWeight.bold)), const SizedBox(height: 40), _MenuTile(icon: Icons.inventory_2_outlined, label: "VIEW VAULT", onTap: () => Navigator.push(context, MaterialPageRoute(builder: (c) => const CollectionView()))), _MenuTile(icon: Icons.emoji_events_outlined, label: "CITIZEN ARCHIVE", onTap: () => Navigator.push(context, MaterialPageRoute(builder: (c) => const AchievementsView()))), const SizedBox(height: 40), // --- HUB BUTTONS (Stricter checks for Web) --- if (rank == 5 || role == 'artist') _HubBtn(label: "ARTIST HUB", color: Colors.amber[900]!, icon: Icons.palette, onTap: () => Navigator.push(context, MaterialPageRoute(builder: (c) => const ArtistHub()))), const SizedBox(height: 10), if (role == 'admin') _HubBtn(label: "COMMAND CENTER", color: Colors.red[900]!, icon: Icons.admin_panel_settings, onTap: () => Navigator.push(context, MaterialPageRoute(builder: (c) => const AdminDashboard()))), const SizedBox(height: 30), TextButton(onPressed: () => FirebaseAuth.instance.signOut(), child: const Text("TERMINATE SESSION", style: TextStyle(color: Colors.grey))), ], ), ); }, ), ), ); } } class _HubBtn extends StatelessWidget { final String label; final Color color; final IconData icon; final VoidCallback onTap; const _HubBtn({required this.label, required this.color, required this.icon, required this.onTap}); @override Widget build(BuildContext context) { return ListTile(onTap: onTap, tileColor: color.withValues(alpha: 0.15), shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12)), leading: Icon(icon, color: color), title: Text(label, style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 12))); } } class _MenuTile extends StatelessWidget { final IconData icon; final String label; final VoidCallback onTap; const _MenuTile({required this.icon, required this.label, required this.onTap}); @override Widget build(BuildContext context) { return ListTile(leading: Icon(icon, color: Colors.white, size: 22), title: Text(label, style: const TextStyle(fontSize: 13)), trailing: const Icon(Icons.chevron_right, color: Colors.grey, size: 18), onTap: onTap); } }