Files
SAVExSTATE/lib/community_view.dart
T

261 lines
9.7 KiB
Dart

import 'package:flutter/material.dart';
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:firebase_auth/firebase_auth.dart';
class CommunityView extends StatelessWidget {
const CommunityView({super.key});
@override
Widget build(BuildContext context) {
final user = FirebaseAuth.instance.currentUser;
// We check the user's role at the top level of the Community tab
return StreamBuilder<DocumentSnapshot>(
stream: FirebaseFirestore.instance.collection('users').doc(user?.uid).snapshots(),
builder: (context, userSnapshot) {
bool isAdmin = false;
if (userSnapshot.hasData && userSnapshot.data!.exists) {
final userData = userSnapshot.data!.data() as Map<String, dynamic>;
isAdmin = userData['role'] == 'admin';
}
return DefaultTabController(
length: 2,
child: Column(
children: [
const TabBar(
indicatorColor: Colors.blueAccent,
labelColor: Colors.blueAccent,
unselectedLabelColor: Colors.grey,
tabs: [
Tab(text: "Fan Lounge"),
Tab(text: "Q&A"),
],
),
Expanded(
child: TabBarView(
children: [
_ChatRoom(),
_QnAFlow(isAdmin: isAdmin), // Pass admin status here
],
),
),
],
),
);
},
);
}
}
// --- PART 1: THE CHATROOM ---
class _ChatRoom extends StatefulWidget {
@override
State<_ChatRoom> createState() => _ChatRoomState();
}
class _ChatRoomState extends State<_ChatRoom> {
final _msgController = TextEditingController();
void _sendMessage() async {
if (_msgController.text.trim().isEmpty) return;
final user = FirebaseAuth.instance.currentUser;
// Before sending, we try to get the most recent name from Firestore
final userDoc = await FirebaseFirestore.instance.collection('users').doc(user?.uid).get();
final String name = userDoc.exists ? (userDoc.data()?['displayName'] ?? "Fan") : "Anonymous Fan";
await FirebaseFirestore.instance.collection('chat').add({
'text': _msgController.text.trim(),
'senderId': user?.uid,
'senderName': name,
'timestamp': FieldValue.serverTimestamp(),
});
_msgController.clear();
}
@override
Widget build(BuildContext context) {
return Column(
children: [
Expanded(
child: StreamBuilder<QuerySnapshot>(
stream: FirebaseFirestore.instance.collection('chat').orderBy('timestamp', descending: true).snapshots(),
builder: (context, snapshot) {
if (!snapshot.hasData) return const Center(child: CircularProgressIndicator());
final msgs = snapshot.data!.docs;
return ListView.builder(
reverse: true, // Newest messages at bottom
padding: const EdgeInsets.all(10),
itemCount: msgs.length,
itemBuilder: (context, index) {
final data = msgs[index].data() as Map<String, dynamic>;
final isMe = data['senderId'] == FirebaseAuth.instance.currentUser?.uid;
return Align(
alignment: isMe ? Alignment.centerRight : Alignment.centerLeft,
child: Container(
margin: const EdgeInsets.symmetric(vertical: 4),
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8),
decoration: BoxDecoration(
color: isMe ? Colors.blueAccent : Colors.grey[850],
borderRadius: BorderRadius.circular(12),
),
child: Column(
crossAxisAlignment: isMe ? CrossAxisAlignment.end : CrossAxisAlignment.start,
children: [
if (!isMe) Text(data['senderName'] ?? "Fan", style: const TextStyle(color: Colors.blueAccent, fontSize: 10, fontWeight: FontWeight.bold)),
Text(data['text'] ?? "", style: const TextStyle(color: Colors.white, fontSize: 15)),
],
),
),
);
},
);
},
),
),
Container(
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
color: Colors.black,
child: Row(
children: [
Expanded(
child: TextField(
controller: _msgController,
decoration: const InputDecoration(hintText: "Say something...", border: InputBorder.none)
)
),
IconButton(onPressed: _sendMessage, icon: const Icon(Icons.send, color: Colors.blueAccent)),
],
),
),
],
);
}
}
// --- PART 2: THE Q&A ---
class _QnAFlow extends StatelessWidget {
final bool isAdmin;
_QnAFlow({required this.isAdmin});
final _questionController = TextEditingController();
void _submitQuestion() async {
if (_questionController.text.isEmpty) return;
final user = FirebaseAuth.instance.currentUser;
final userDoc = await FirebaseFirestore.instance.collection('users').doc(user?.uid).get();
final String name = userDoc.exists ? (userDoc.data()?['displayName'] ?? "Fan") : "Fan";
await FirebaseFirestore.instance.collection('qna').add({
'question': _questionController.text.trim(),
'answer': null,
'senderName': name,
'timestamp': FieldValue.serverTimestamp(),
});
_questionController.clear();
}
// ONLY Admin sees this dialog
void _showAnswerDialog(BuildContext context, String qId, String questionText) {
final ansController = TextEditingController();
showDialog(
context: context,
builder: (context) => AlertDialog(
backgroundColor: Colors.grey[900],
title: Text("Answer Question", style: TextStyle(color: Colors.blueAccent[100], fontSize: 16)),
content: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text('"$questionText"', style: const TextStyle(fontStyle: FontStyle.italic, color: Colors.white70)),
const SizedBox(height: 15),
TextField(
controller: ansController,
maxLines: 3,
decoration: const InputDecoration(hintText: "Type your answer...", border: OutlineInputBorder()),
),
],
),
actions: [
TextButton(onPressed: () => Navigator.pop(context), child: const Text("Cancel")),
ElevatedButton(
onPressed: () async {
await FirebaseFirestore.instance.collection('qna').doc(qId).update({
'answer': ansController.text.trim(),
});
Navigator.pop(context);
},
child: const Text("Post Answer"),
),
],
),
);
}
@override
Widget build(BuildContext context) {
return Column(
children: [
Padding(
padding: const EdgeInsets.all(16.0),
child: TextField(
controller: _questionController,
decoration: InputDecoration(
hintText: "Ask Sage a question...",
filled: true,
fillColor: Colors.grey[900],
border: OutlineInputBorder(borderRadius: BorderRadius.circular(10), borderSide: BorderSide.none),
suffixIcon: IconButton(onPressed: _submitQuestion, icon: const Icon(Icons.help_outline, color: Colors.blueAccent)),
),
),
),
Expanded(
child: StreamBuilder<QuerySnapshot>(
stream: FirebaseFirestore.instance.collection('qna').orderBy('timestamp', descending: true).snapshots(),
builder: (context, snapshot) {
if (!snapshot.hasData) return const Center(child: CircularProgressIndicator());
final questions = snapshot.data!.docs;
return ListView.builder(
padding: const EdgeInsets.symmetric(horizontal: 10),
itemCount: questions.length,
itemBuilder: (context, index) {
final data = questions[index].data() as Map<String, dynamic>;
final String qId = questions[index].id;
final hasAnswer = data['answer'] != null;
return Card(
color: hasAnswer ? Colors.blueGrey[900] : Colors.grey[850],
margin: const EdgeInsets.symmetric(vertical: 6),
child: ListTile(
title: Text(data['question'] ?? "", style: const TextStyle(fontWeight: FontWeight.bold)),
subtitle: Padding(
padding: const EdgeInsets.only(top: 8.0),
child: Text(
hasAnswer ? "SAGE: ${data['answer']}" : "Waiting for answer...",
style: TextStyle(
color: hasAnswer ? Colors.greenAccent : Colors.grey,
fontWeight: hasAnswer ? FontWeight.bold : FontWeight.normal
),
),
),
trailing: isAdmin ? const Icon(Icons.edit, color: Colors.blueAccent, size: 18) : null,
onTap: () {
if (isAdmin) {
_showAnswerDialog(context, qId, data['question']);
}
},
),
);
},
);
},
),
),
],
);
}
}