Files
2026-04-23 23:58:59 -05:00

154 lines
8.1 KiB
Dart

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<ProfileView> createState() => _ProfileViewState();
}
class _ProfileViewState extends State<ProfileView> {
bool _isUploading = false;
Uint8List? _localBytes;
final ImagePicker _picker = ImagePicker();
Future<void> _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<DocumentSnapshot>(
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<String, dynamic>;
// --- 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);
}
}