2025-11-12 17:50:44 +01:00
|
|
|
import { useState, useRef, useEffect } from 'react';
|
2025-11-14 14:36:49 +01:00
|
|
|
import { useParams, useNavigate, Link } from 'react-router-dom';
|
2025-11-12 17:50:44 +01:00
|
|
|
import Layout from '../components/layout/Layout';
|
|
|
|
|
import { useAuth } from '../contexts/AuthContext';
|
2025-11-14 17:41:35 +01:00
|
|
|
import { Send, UserPlus, Loader2, LogOut, AlertTriangle, QrCode, Edit2, Filter, X } from 'lucide-react';
|
2025-11-12 22:42:15 +01:00
|
|
|
import { connectSocket, disconnectSocket, getSocket } from '../services/socket';
|
2025-11-14 19:22:23 +01:00
|
|
|
import { eventsAPI, heatsAPI, matchesAPI } from '../services/api';
|
2025-11-14 17:41:35 +01:00
|
|
|
import HeatsBanner from '../components/heats/HeatsBanner';
|
2025-11-15 23:08:00 +01:00
|
|
|
import Avatar from '../components/common/Avatar';
|
2025-11-21 16:50:46 +01:00
|
|
|
import ChatMessageList from '../components/chat/ChatMessageList';
|
|
|
|
|
import ChatInput from '../components/chat/ChatInput';
|
|
|
|
|
import ConfirmationModal from '../components/modals/ConfirmationModal';
|
|
|
|
|
import Modal from '../components/modals/Modal';
|
2025-11-21 17:02:04 +01:00
|
|
|
import useEventChat from '../hooks/useEventChat';
|
2025-11-12 17:50:44 +01:00
|
|
|
|
|
|
|
|
const EventChatPage = () => {
|
2025-11-13 21:43:58 +01:00
|
|
|
const { slug } = useParams();
|
2025-11-12 17:50:44 +01:00
|
|
|
const { user } = useAuth();
|
|
|
|
|
const navigate = useNavigate();
|
2025-11-13 21:43:58 +01:00
|
|
|
const [event, setEvent] = useState(null);
|
2025-11-14 14:36:49 +01:00
|
|
|
const [isParticipant, setIsParticipant] = useState(false);
|
2025-11-13 21:43:58 +01:00
|
|
|
const [loading, setLoading] = useState(true);
|
2025-11-14 18:04:10 +01:00
|
|
|
const [checkedInUsers, setCheckedInUsers] = useState([]);
|
2025-11-14 14:11:24 +01:00
|
|
|
const [showLeaveModal, setShowLeaveModal] = useState(false);
|
|
|
|
|
const [isLeaving, setIsLeaving] = useState(false);
|
2025-11-12 17:50:44 +01:00
|
|
|
const messagesEndRef = useRef(null);
|
2025-11-13 20:16:58 +01:00
|
|
|
const messagesContainerRef = useRef(null);
|
2025-11-12 17:50:44 +01:00
|
|
|
|
2025-11-21 17:02:04 +01:00
|
|
|
// Event Chat hook - manages messages, Socket.IO, and active users
|
|
|
|
|
const {
|
|
|
|
|
messages,
|
|
|
|
|
newMessage,
|
|
|
|
|
setNewMessage,
|
|
|
|
|
activeUsers,
|
|
|
|
|
isConnected,
|
|
|
|
|
loadingOlder,
|
|
|
|
|
hasMore,
|
|
|
|
|
sendMessage: handleSendMessage,
|
|
|
|
|
loadOlderMessages
|
|
|
|
|
} = useEventChat(slug, user?.id, event, messagesContainerRef);
|
|
|
|
|
|
2025-11-14 17:41:35 +01:00
|
|
|
// Heats state
|
|
|
|
|
const [myHeats, setMyHeats] = useState([]);
|
|
|
|
|
const [userHeats, setUserHeats] = useState(new Map());
|
|
|
|
|
const [showHeatsBanner, setShowHeatsBanner] = useState(false);
|
|
|
|
|
const [hideMyHeats, setHideMyHeats] = useState(false);
|
|
|
|
|
const [showHeatsModal, setShowHeatsModal] = useState(false);
|
|
|
|
|
|
2025-11-12 17:50:44 +01:00
|
|
|
const scrollToBottom = () => {
|
|
|
|
|
messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' });
|
|
|
|
|
};
|
|
|
|
|
|
2025-11-14 14:36:49 +01:00
|
|
|
// Fetch event data and check participation
|
2025-11-13 21:43:58 +01:00
|
|
|
useEffect(() => {
|
|
|
|
|
const fetchEvent = async () => {
|
|
|
|
|
try {
|
|
|
|
|
setLoading(true);
|
2025-11-14 14:36:49 +01:00
|
|
|
|
|
|
|
|
// Get all events with participation info
|
|
|
|
|
const allEvents = await eventsAPI.getAll();
|
|
|
|
|
const eventData = allEvents.find(e => e.slug === slug);
|
|
|
|
|
|
|
|
|
|
if (eventData) {
|
|
|
|
|
setEvent(eventData);
|
|
|
|
|
setIsParticipant(eventData.isJoined);
|
|
|
|
|
} else {
|
|
|
|
|
setEvent(null);
|
|
|
|
|
setIsParticipant(false);
|
|
|
|
|
}
|
2025-11-13 21:43:58 +01:00
|
|
|
} catch (err) {
|
|
|
|
|
console.error('Error loading event:', err);
|
|
|
|
|
setEvent(null);
|
2025-11-14 14:36:49 +01:00
|
|
|
setIsParticipant(false);
|
2025-11-13 21:43:58 +01:00
|
|
|
} finally {
|
|
|
|
|
setLoading(false);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
fetchEvent();
|
|
|
|
|
}, [slug]);
|
|
|
|
|
|
2025-11-14 18:04:10 +01:00
|
|
|
// Load heats data and checked-in users
|
2025-11-14 17:41:35 +01:00
|
|
|
useEffect(() => {
|
|
|
|
|
if (!event || !isParticipant) return;
|
|
|
|
|
|
2025-11-14 18:04:10 +01:00
|
|
|
const loadData = async () => {
|
2025-11-14 17:41:35 +01:00
|
|
|
try {
|
|
|
|
|
// Load my heats
|
|
|
|
|
const myHeatsData = await heatsAPI.getMyHeats(slug);
|
|
|
|
|
setMyHeats(myHeatsData);
|
|
|
|
|
setShowHeatsBanner(myHeatsData.length === 0);
|
|
|
|
|
|
|
|
|
|
// Load all users' heats
|
|
|
|
|
const allHeatsData = await heatsAPI.getAllHeats(slug);
|
|
|
|
|
const heatsMap = new Map();
|
|
|
|
|
allHeatsData.forEach((userHeat) => {
|
|
|
|
|
heatsMap.set(userHeat.userId, userHeat.heats);
|
|
|
|
|
});
|
|
|
|
|
setUserHeats(heatsMap);
|
2025-11-14 18:04:10 +01:00
|
|
|
|
|
|
|
|
// Load all checked-in users (participants)
|
|
|
|
|
const eventDetails = await eventsAPI.getDetails(slug);
|
|
|
|
|
if (eventDetails.data && eventDetails.data.participants) {
|
|
|
|
|
const participants = eventDetails.data.participants
|
|
|
|
|
.map(p => ({
|
2025-11-14 18:10:35 +01:00
|
|
|
userId: p.userId,
|
|
|
|
|
username: p.username,
|
|
|
|
|
avatar: p.avatar || `https://api.dicebear.com/7.x/avataaars/svg?seed=${p.username}`,
|
|
|
|
|
firstName: p.firstName,
|
|
|
|
|
lastName: p.lastName,
|
2025-11-14 18:04:10 +01:00
|
|
|
}))
|
|
|
|
|
.filter(p => p.userId !== user.id); // Exclude current user
|
|
|
|
|
setCheckedInUsers(participants);
|
|
|
|
|
}
|
2025-11-14 17:41:35 +01:00
|
|
|
} catch (error) {
|
2025-11-14 18:04:10 +01:00
|
|
|
console.error('Failed to load data:', error);
|
2025-11-14 17:41:35 +01:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2025-11-14 18:04:10 +01:00
|
|
|
loadData();
|
|
|
|
|
}, [event, isParticipant, slug, user.id]);
|
2025-11-14 17:41:35 +01:00
|
|
|
|
2025-11-12 17:50:44 +01:00
|
|
|
useEffect(() => {
|
|
|
|
|
scrollToBottom();
|
|
|
|
|
}, [messages]);
|
|
|
|
|
|
2025-11-21 17:02:04 +01:00
|
|
|
// Heats updates listener (specific to EventChatPage)
|
2025-11-12 22:42:15 +01:00
|
|
|
useEffect(() => {
|
2025-11-13 21:43:58 +01:00
|
|
|
if (!event) return;
|
|
|
|
|
|
2025-11-21 17:02:04 +01:00
|
|
|
const socket = getSocket();
|
|
|
|
|
if (!socket) return;
|
2025-11-12 22:42:15 +01:00
|
|
|
|
2025-11-14 17:41:35 +01:00
|
|
|
// Heats updated notification
|
2025-11-21 17:02:04 +01:00
|
|
|
const handleHeatsUpdated = (data) => {
|
2025-11-14 17:41:35 +01:00
|
|
|
const { userId, heats } = data;
|
|
|
|
|
|
|
|
|
|
// Update userHeats map
|
|
|
|
|
setUserHeats((prev) => {
|
|
|
|
|
const newMap = new Map(prev);
|
|
|
|
|
if (heats && heats.length > 0) {
|
|
|
|
|
newMap.set(userId, heats);
|
|
|
|
|
} else {
|
|
|
|
|
newMap.delete(userId);
|
|
|
|
|
}
|
|
|
|
|
return newMap;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// If it's the current user, update myHeats
|
|
|
|
|
if (userId === user.id) {
|
|
|
|
|
setMyHeats(heats || []);
|
|
|
|
|
setShowHeatsBanner(heats.length === 0);
|
|
|
|
|
}
|
2025-11-12 22:42:15 +01:00
|
|
|
};
|
2025-11-12 17:50:44 +01:00
|
|
|
|
2025-11-21 17:02:04 +01:00
|
|
|
socket.on('heats_updated', handleHeatsUpdated);
|
2025-11-12 22:42:15 +01:00
|
|
|
|
2025-11-21 17:02:04 +01:00
|
|
|
return () => {
|
|
|
|
|
socket.off('heats_updated', handleHeatsUpdated);
|
|
|
|
|
};
|
|
|
|
|
}, [event, user.id]);
|
2025-11-13 20:16:58 +01:00
|
|
|
|
2025-11-14 19:22:23 +01:00
|
|
|
const handleMatchWith = async (userId) => {
|
|
|
|
|
try {
|
|
|
|
|
const result = await matchesAPI.createMatch(userId, slug);
|
|
|
|
|
|
|
|
|
|
// Show success message
|
|
|
|
|
alert(`Match request sent successfully! The user will be notified.`);
|
|
|
|
|
|
|
|
|
|
// Optional: Navigate to matches page or refresh matches list
|
|
|
|
|
// For now, we just show a success message
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error('Failed to send match request:', error);
|
|
|
|
|
|
|
|
|
|
// Show appropriate error message
|
|
|
|
|
if (error.status === 400 && error.message.includes('already exists')) {
|
|
|
|
|
alert('You already have a match request with this user.');
|
|
|
|
|
} else if (error.status === 403) {
|
|
|
|
|
alert('You must be a participant of this event to send match requests.');
|
|
|
|
|
} else if (error.status === 404) {
|
|
|
|
|
alert('Event not found.');
|
|
|
|
|
} else {
|
|
|
|
|
alert('Failed to send match request. Please try again.');
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-11-12 17:50:44 +01:00
|
|
|
};
|
|
|
|
|
|
2025-11-14 17:41:35 +01:00
|
|
|
const handleHeatsSave = () => {
|
|
|
|
|
setShowHeatsBanner(false);
|
|
|
|
|
setShowHeatsModal(false);
|
|
|
|
|
// Heats will be updated via Socket.IO heats_updated event
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const formatHeat = (heat) => {
|
|
|
|
|
const parts = [
|
|
|
|
|
heat.competitionType?.abbreviation || '',
|
|
|
|
|
heat.division?.abbreviation || '',
|
|
|
|
|
heat.heatNumber,
|
|
|
|
|
];
|
|
|
|
|
if (heat.role) {
|
|
|
|
|
parts.push(heat.role.charAt(0)); // L or F
|
|
|
|
|
}
|
|
|
|
|
return parts.join(' ');
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const shouldHideUser = (userId) => {
|
|
|
|
|
if (!hideMyHeats || myHeats.length === 0) return false;
|
|
|
|
|
|
|
|
|
|
const targetUserHeats = userHeats.get(userId);
|
|
|
|
|
if (!targetUserHeats || targetUserHeats.length === 0) return false;
|
|
|
|
|
|
|
|
|
|
// Hide if ANY of their heats match ANY of my heats (same division + competition_type + heat_number)
|
|
|
|
|
return targetUserHeats.some((targetHeat) =>
|
|
|
|
|
myHeats.some(
|
|
|
|
|
(myHeat) =>
|
|
|
|
|
myHeat.divisionId === targetHeat.divisionId &&
|
|
|
|
|
myHeat.competitionTypeId === targetHeat.competitionTypeId &&
|
|
|
|
|
myHeat.heatNumber === targetHeat.heatNumber
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
2025-11-14 18:04:10 +01:00
|
|
|
// Combine checked-in users with online status
|
|
|
|
|
const getAllDisplayUsers = () => {
|
|
|
|
|
const activeUserIds = new Set(activeUsers.map(u => u.userId));
|
|
|
|
|
|
|
|
|
|
// Merge checked-in users with online status
|
|
|
|
|
const allUsers = checkedInUsers.map(user => ({
|
|
|
|
|
...user,
|
|
|
|
|
isOnline: activeUserIds.has(user.userId),
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
// Sort: online first, then offline
|
|
|
|
|
return allUsers.sort((a, b) => {
|
|
|
|
|
if (a.isOnline === b.isOnline) return 0;
|
|
|
|
|
return a.isOnline ? -1 : 1;
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2025-11-14 14:11:24 +01:00
|
|
|
const handleLeaveEvent = async () => {
|
|
|
|
|
try {
|
|
|
|
|
setIsLeaving(true);
|
|
|
|
|
await eventsAPI.leave(slug);
|
|
|
|
|
|
|
|
|
|
// Disconnect socket
|
|
|
|
|
const socket = getSocket();
|
|
|
|
|
if (socket) {
|
|
|
|
|
socket.emit('leave_event_room');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Redirect to events page
|
|
|
|
|
navigate('/events');
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error('Failed to leave event:', error);
|
|
|
|
|
alert('Failed to leave event. Please try again.');
|
|
|
|
|
} finally {
|
|
|
|
|
setIsLeaving(false);
|
|
|
|
|
setShowLeaveModal(false);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2025-11-13 20:16:58 +01:00
|
|
|
// Infinite scroll - detect scroll to top
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
const container = messagesContainerRef.current;
|
|
|
|
|
if (!container) return;
|
|
|
|
|
|
|
|
|
|
const handleScroll = () => {
|
|
|
|
|
if (container.scrollTop < 100 && !loadingOlder && hasMore) {
|
|
|
|
|
loadOlderMessages();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
container.addEventListener('scroll', handleScroll);
|
|
|
|
|
return () => container.removeEventListener('scroll', handleScroll);
|
|
|
|
|
}, [loadingOlder, hasMore, messages]);
|
|
|
|
|
|
2025-11-13 21:43:58 +01:00
|
|
|
if (loading) {
|
|
|
|
|
return (
|
|
|
|
|
<Layout>
|
|
|
|
|
<div className="max-w-4xl mx-auto flex items-center justify-center min-h-[400px]">
|
|
|
|
|
<div className="flex flex-col items-center gap-3">
|
|
|
|
|
<Loader2 className="w-12 h-12 animate-spin text-primary-600" />
|
|
|
|
|
<p className="text-gray-600">Loading event...</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Layout>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-12 17:50:44 +01:00
|
|
|
if (!event) {
|
|
|
|
|
return (
|
|
|
|
|
<Layout>
|
2025-11-13 21:43:58 +01:00
|
|
|
<div className="max-w-4xl mx-auto">
|
|
|
|
|
<div className="bg-red-50 border border-red-200 rounded-md p-4 text-red-700">
|
|
|
|
|
Event not found
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2025-11-12 17:50:44 +01:00
|
|
|
</Layout>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-14 14:36:49 +01:00
|
|
|
// Check if user is participant
|
|
|
|
|
if (!isParticipant) {
|
|
|
|
|
return (
|
|
|
|
|
<Layout>
|
|
|
|
|
<div className="max-w-4xl mx-auto">
|
|
|
|
|
<div className="bg-white rounded-lg shadow-md p-8">
|
|
|
|
|
<div className="text-center">
|
|
|
|
|
<div className="w-16 h-16 bg-amber-100 rounded-full flex items-center justify-center mx-auto mb-4">
|
|
|
|
|
<QrCode className="w-8 h-8 text-amber-600" />
|
|
|
|
|
</div>
|
|
|
|
|
<h2 className="text-2xl font-bold text-gray-900 mb-2">
|
|
|
|
|
Check-in Required
|
|
|
|
|
</h2>
|
|
|
|
|
<p className="text-gray-600 mb-6">
|
|
|
|
|
You need to check in at the event venue by scanning the QR code to access this chat.
|
|
|
|
|
</p>
|
|
|
|
|
|
|
|
|
|
<div className="bg-amber-50 border border-amber-200 rounded-lg p-4 mb-6">
|
|
|
|
|
<p className="text-sm text-amber-800 font-medium mb-1">
|
|
|
|
|
{event.name}
|
|
|
|
|
</p>
|
|
|
|
|
<p className="text-sm text-amber-700">
|
|
|
|
|
{event.location}
|
|
|
|
|
</p>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className="flex gap-3 justify-center">
|
|
|
|
|
<Link
|
|
|
|
|
to="/events"
|
|
|
|
|
className="px-6 py-2 border border-gray-300 rounded-lg hover:bg-gray-50 transition-colors"
|
|
|
|
|
>
|
|
|
|
|
Back to Events
|
|
|
|
|
</Link>
|
|
|
|
|
{import.meta.env.DEV && (
|
|
|
|
|
<Link
|
|
|
|
|
to={`/events/${slug}/details`}
|
|
|
|
|
className="px-6 py-2 bg-primary-600 text-white rounded-lg hover:bg-primary-700 transition-colors"
|
|
|
|
|
>
|
|
|
|
|
View QR Code (dev)
|
|
|
|
|
</Link>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Layout>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-12 17:50:44 +01:00
|
|
|
return (
|
|
|
|
|
<Layout>
|
|
|
|
|
<div className="max-w-6xl mx-auto">
|
|
|
|
|
<div className="bg-white rounded-lg shadow-md overflow-hidden">
|
|
|
|
|
{/* Header */}
|
|
|
|
|
<div className="bg-primary-600 text-white p-4">
|
2025-11-14 17:41:35 +01:00
|
|
|
<div className="flex items-start justify-between">
|
|
|
|
|
<div className="flex-1">
|
|
|
|
|
<h2 className="text-2xl font-bold">{event.name}</h2>
|
|
|
|
|
<p className="text-primary-100 text-sm">{event.location}</p>
|
2025-11-14 18:41:06 +01:00
|
|
|
<div className="mt-2 flex items-center gap-3">
|
2025-11-14 17:41:35 +01:00
|
|
|
<span className={`text-xs px-2 py-1 rounded ${isConnected ? 'bg-green-500' : 'bg-red-500'}`}>
|
|
|
|
|
{isConnected ? '● Connected' : '● Disconnected'}
|
|
|
|
|
</span>
|
2025-11-14 18:41:06 +01:00
|
|
|
|
|
|
|
|
{/* My Heats Display */}
|
|
|
|
|
{myHeats.length > 0 && (
|
|
|
|
|
<div className="flex items-center gap-2">
|
|
|
|
|
<span className="text-xs text-primary-100">Your heats:</span>
|
|
|
|
|
<div className="flex flex-wrap gap-1">
|
|
|
|
|
{myHeats.map((heat, idx) => (
|
|
|
|
|
<span
|
|
|
|
|
key={idx}
|
|
|
|
|
className="text-xs px-2 py-0.5 bg-primary-700 text-white rounded font-mono"
|
|
|
|
|
>
|
|
|
|
|
{formatHeat(heat)}
|
|
|
|
|
</span>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2025-11-14 17:41:35 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
{myHeats.length > 0 && (
|
|
|
|
|
<button
|
|
|
|
|
onClick={() => setShowHeatsModal(true)}
|
|
|
|
|
className="flex items-center gap-2 px-3 py-2 bg-primary-700 hover:bg-primary-800 rounded-md transition-colors text-sm"
|
|
|
|
|
>
|
|
|
|
|
<Edit2 className="w-4 h-4" />
|
|
|
|
|
Edit Heats
|
|
|
|
|
</button>
|
|
|
|
|
)}
|
2025-11-12 22:42:15 +01:00
|
|
|
</div>
|
2025-11-12 17:50:44 +01:00
|
|
|
</div>
|
|
|
|
|
|
2025-11-14 17:41:35 +01:00
|
|
|
{/* Heats Banner */}
|
|
|
|
|
{showHeatsBanner && (
|
|
|
|
|
<HeatsBanner
|
|
|
|
|
slug={slug}
|
|
|
|
|
onSave={handleHeatsSave}
|
|
|
|
|
onDismiss={() => setShowHeatsBanner(false)}
|
|
|
|
|
/>
|
|
|
|
|
)}
|
|
|
|
|
|
2025-11-12 17:50:44 +01:00
|
|
|
<div className="flex h-[calc(100vh-280px)]">
|
2025-11-14 18:04:10 +01:00
|
|
|
{/* Participants Sidebar */}
|
2025-11-12 17:50:44 +01:00
|
|
|
<div className="w-64 border-r bg-gray-50 p-4 overflow-y-auto">
|
2025-11-14 17:41:35 +01:00
|
|
|
<div className="mb-4">
|
2025-11-14 18:04:10 +01:00
|
|
|
<h3 className="font-semibold text-gray-900 mb-2">
|
|
|
|
|
Participants ({getAllDisplayUsers().filter(u => !shouldHideUser(u.userId)).length})
|
2025-11-14 17:41:35 +01:00
|
|
|
</h3>
|
2025-11-14 18:04:10 +01:00
|
|
|
<p className="text-xs text-gray-500 mb-3">
|
|
|
|
|
{activeUsers.length} online
|
|
|
|
|
</p>
|
2025-11-14 17:41:35 +01:00
|
|
|
|
|
|
|
|
{/* Filter Checkbox */}
|
|
|
|
|
{myHeats.length > 0 && (
|
|
|
|
|
<label className="flex items-center gap-2 text-sm text-gray-700 cursor-pointer hover:text-gray-900">
|
|
|
|
|
<input
|
|
|
|
|
type="checkbox"
|
|
|
|
|
checked={hideMyHeats}
|
|
|
|
|
onChange={(e) => setHideMyHeats(e.target.checked)}
|
|
|
|
|
className="rounded border-gray-300 text-primary-600 focus:ring-primary-500"
|
|
|
|
|
/>
|
|
|
|
|
<Filter className="w-3 h-3" />
|
|
|
|
|
<span>Hide users from my heats</span>
|
|
|
|
|
</label>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
|
2025-11-14 18:04:10 +01:00
|
|
|
{getAllDisplayUsers().filter(u => !shouldHideUser(u.userId)).length === 0 && (
|
|
|
|
|
<p className="text-sm text-gray-500">No other participants</p>
|
2025-11-12 22:42:15 +01:00
|
|
|
)}
|
2025-11-12 17:50:44 +01:00
|
|
|
<div className="space-y-2">
|
2025-11-14 18:04:10 +01:00
|
|
|
{getAllDisplayUsers()
|
|
|
|
|
.filter((displayUser) => !shouldHideUser(displayUser.userId))
|
|
|
|
|
.map((displayUser) => {
|
|
|
|
|
const thisUserHeats = userHeats.get(displayUser.userId) || [];
|
2025-11-14 17:41:35 +01:00
|
|
|
const hasHeats = thisUserHeats.length > 0;
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<div
|
2025-11-14 18:04:10 +01:00
|
|
|
key={displayUser.userId}
|
2025-11-14 17:41:35 +01:00
|
|
|
className="flex items-center justify-between p-2 hover:bg-gray-100 rounded-lg"
|
|
|
|
|
>
|
|
|
|
|
<div className="flex items-center space-x-2 flex-1 min-w-0">
|
2025-11-15 23:08:00 +01:00
|
|
|
<Avatar
|
|
|
|
|
src={displayUser.avatar}
|
|
|
|
|
username={displayUser.username}
|
|
|
|
|
size={32}
|
|
|
|
|
status={displayUser.isOnline ? 'online' : 'offline'}
|
|
|
|
|
title={displayUser.username}
|
|
|
|
|
/>
|
2025-11-14 17:41:35 +01:00
|
|
|
<div className="flex-1 min-w-0">
|
2025-11-14 18:04:10 +01:00
|
|
|
<p className={`text-sm font-medium truncate ${
|
|
|
|
|
displayUser.isOnline ? 'text-gray-900' : 'text-gray-500'
|
|
|
|
|
}`}>
|
|
|
|
|
{displayUser.username}
|
2025-11-14 17:41:35 +01:00
|
|
|
</p>
|
|
|
|
|
{/* Heat Badges */}
|
|
|
|
|
{hasHeats && (
|
|
|
|
|
<div className="flex flex-wrap gap-1 mt-1">
|
|
|
|
|
{thisUserHeats.slice(0, 3).map((heat, idx) => (
|
|
|
|
|
<span
|
|
|
|
|
key={idx}
|
|
|
|
|
className="text-xs px-1.5 py-0.5 bg-amber-100 text-amber-800 rounded font-mono"
|
|
|
|
|
>
|
|
|
|
|
{formatHeat(heat)}
|
|
|
|
|
</span>
|
|
|
|
|
))}
|
|
|
|
|
{thisUserHeats.length > 3 && (
|
|
|
|
|
<span className="text-xs px-1.5 py-0.5 bg-gray-200 text-gray-700 rounded font-mono">
|
|
|
|
|
+{thisUserHeats.length - 3}
|
|
|
|
|
</span>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<button
|
2025-11-14 18:04:10 +01:00
|
|
|
onClick={() => handleMatchWith(displayUser.userId)}
|
2025-11-14 17:41:35 +01:00
|
|
|
disabled={!hasHeats}
|
|
|
|
|
className={`p-1 rounded flex-shrink-0 ${
|
|
|
|
|
hasHeats
|
|
|
|
|
? 'text-primary-600 hover:bg-primary-50'
|
|
|
|
|
: 'text-gray-300 cursor-not-allowed'
|
|
|
|
|
}`}
|
|
|
|
|
title={hasHeats ? 'Connect' : 'User has not declared heats'}
|
|
|
|
|
>
|
|
|
|
|
<UserPlus className="w-4 h-4" />
|
|
|
|
|
</button>
|
2025-11-12 17:50:44 +01:00
|
|
|
</div>
|
2025-11-14 17:41:35 +01:00
|
|
|
);
|
|
|
|
|
})}
|
2025-11-12 17:50:44 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{/* Chat Area */}
|
|
|
|
|
<div className="flex-1 flex flex-col">
|
2025-11-21 16:50:46 +01:00
|
|
|
<ChatMessageList
|
|
|
|
|
messages={messages}
|
|
|
|
|
currentUserId={user.id}
|
|
|
|
|
messagesEndRef={messagesEndRef}
|
|
|
|
|
messagesContainerRef={messagesContainerRef}
|
|
|
|
|
loadingOlder={loadingOlder}
|
|
|
|
|
hasMore={hasMore}
|
|
|
|
|
/>
|
2025-11-12 17:50:44 +01:00
|
|
|
|
|
|
|
|
{/* Message Input */}
|
|
|
|
|
<div className="border-t p-4">
|
2025-11-21 16:50:46 +01:00
|
|
|
<ChatInput
|
|
|
|
|
value={newMessage}
|
|
|
|
|
onChange={(e) => setNewMessage(e.target.value)}
|
|
|
|
|
onSubmit={handleSendMessage}
|
|
|
|
|
disabled={!isConnected}
|
|
|
|
|
placeholder="Write a message..."
|
|
|
|
|
/>
|
2025-11-12 17:50:44 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2025-11-14 14:11:24 +01:00
|
|
|
|
|
|
|
|
{/* Leave Event Button */}
|
|
|
|
|
<div className="mt-4 flex justify-center">
|
|
|
|
|
<button
|
|
|
|
|
onClick={() => setShowLeaveModal(true)}
|
|
|
|
|
className="flex items-center gap-2 px-4 py-2 bg-red-600 text-white rounded-lg hover:bg-red-700 transition-colors font-medium"
|
|
|
|
|
>
|
|
|
|
|
<LogOut size={16} />
|
|
|
|
|
Leave Event
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
2025-11-12 17:50:44 +01:00
|
|
|
</div>
|
2025-11-14 14:11:24 +01:00
|
|
|
|
2025-11-21 16:50:46 +01:00
|
|
|
<ConfirmationModal
|
|
|
|
|
isOpen={showLeaveModal}
|
|
|
|
|
onClose={() => setShowLeaveModal(false)}
|
|
|
|
|
onConfirm={handleLeaveEvent}
|
|
|
|
|
title="Leave Event?"
|
|
|
|
|
description="This action cannot be undone"
|
|
|
|
|
message={`Are you sure you want to leave ${event?.name || 'this event'}? You will need to scan the QR code again to rejoin.`}
|
|
|
|
|
confirmText="Leave Event"
|
|
|
|
|
isLoading={isLeaving}
|
|
|
|
|
loadingText="Leaving..."
|
|
|
|
|
icon={AlertTriangle}
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
<Modal
|
|
|
|
|
isOpen={showHeatsModal}
|
|
|
|
|
onClose={() => setShowHeatsModal(false)}
|
|
|
|
|
title="Edit Your Competition Heats"
|
|
|
|
|
>
|
|
|
|
|
<HeatsBanner
|
|
|
|
|
slug={slug}
|
|
|
|
|
onSave={handleHeatsSave}
|
|
|
|
|
onDismiss={() => setShowHeatsModal(false)}
|
|
|
|
|
existingHeats={myHeats}
|
|
|
|
|
/>
|
|
|
|
|
</Modal>
|
2025-11-12 17:50:44 +01:00
|
|
|
</div>
|
|
|
|
|
</Layout>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export default EventChatPage;
|