Dashboard real-time updates con SSE y tabla de rooms

- Implementación de Server-Sent Events (SSE) para actualizaciones en tiempo real
- Nuevo componente RoomsTable para vista de águila con animaciones de tokens
- Componente RoomCard extraído para reutilización
- Modal para ver detalles de rooms
- SystemMessageDisplay usando AnimatedNumber creativamente
- Indicador de conexión SSE con fallback a polling
- Colores dinámicos de texto basados en brillo del fondo
- Backend: broadcast de actualizaciones del dashboard
- Backend: mensajes del sistema (excepto cambios de ronda) visibles en dashboard
- Configuración de Vite para acceso desde red local
This commit is contained in:
2025-08-11 23:07:09 -06:00
parent deb63d4e38
commit 32f69805f0
9 changed files with 1689 additions and 265 deletions

View File

@@ -0,0 +1,353 @@
<template>
<div class="room-card">
<div class="room-header">
<span class="room-id">Room {{ room.roomId.slice(0, 8) }}</span>
<span class="room-status" :class="`status-${room.metadata?.gameStatus || 'waiting'}`">
{{ room.metadata?.gameStatus || 'waiting' }}
</span>
</div>
<div class="room-details">
<div class="detail-row">
<span class="detail-label">Players:</span>
<span class="detail-value">{{ room.clients }}/{{ room.maxClients }}</span>
</div>
<div class="detail-row">
<span class="detail-label">Round:</span>
<span class="detail-value">{{ room.metadata?.currentRound || 1 }}/3</span>
</div>
<div class="detail-row">
<span class="detail-label">Game Variant:</span>
<span class="detail-value">{{ room.metadata?.currentVariant || 'G1' }}</span>
</div>
<div class="detail-row">
<span class="detail-label">Created:</span>
<span class="detail-value">{{ formatTime(room.createdAt) }}</span>
</div>
</div>
<div class="room-actions">
<button
v-if="room.metadata?.gameStatus === 'playing'"
@click="$emit('pause', room.roomId)"
class="btn btn-action btn-pause"
>
Pause
</button>
<button
v-if="room.metadata?.gameStatus === 'paused'"
@click="$emit('resume', room.roomId)"
class="btn btn-action btn-resume"
>
Resume
</button>
<button
@click="$emit('restart', room.roomId)"
class="btn btn-action btn-restart"
>
🔄 Restart
</button>
<button
@click="$emit('viewDetails', room.roomId)"
class="btn btn-action btn-view"
>
📊 Details
</button>
</div>
<div v-if="roomDetails" class="room-stats">
<h4>Room Statistics</h4>
<div v-if="roomDetails.players" class="players-list">
<div v-for="player in roomDetails.players"
:key="player.sessionId"
class="player-row">
<div class="player-info">
<span class="player-name">{{ player.name }}</span>
<span class="player-role">{{ player.role }}</span>
</div>
<div class="player-tokens">
<span class="token pavo">🦃 {{ player.pavoTokens || 0 }}</span>
<span class="token elote">🌽 {{ player.eloteTokens || 0 }}</span>
<span class="token shame">😳 {{ player.shameTokens || 0 }}</span>
</div>
<button
@click="$emit('kickPlayer', room.roomId, player.sessionId)"
class="btn btn-kick"
>
Kick
</button>
</div>
</div>
<div class="stats-info">
<div class="stat-item">
<span>Time Remaining:</span>
<span>{{ formatSeconds(roomDetails.timeRemaining) }}</span>
</div>
<div v-if="roomDetails.winner" class="stat-item">
<span>Winner:</span>
<span class="winner-name">{{ roomDetails.winner }}</span>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
interface Room {
roomId: string;
clients: number;
maxClients: number;
createdAt: number;
metadata?: {
gameStatus?: string;
currentRound?: number;
currentVariant?: string;
};
}
interface RoomDetails {
players?: Array<{
sessionId: string;
name: string;
role: string;
pavoTokens?: number;
eloteTokens?: number;
shameTokens?: number;
}>;
timeRemaining: number;
winner?: string;
}
defineProps<{
room: Room;
roomDetails?: RoomDetails;
}>();
defineEmits<{
pause: [roomId: string];
resume: [roomId: string];
restart: [roomId: string];
viewDetails: [roomId: string];
kickPlayer: [roomId: string, playerId: string];
}>();
function formatTime(timestamp: number): string {
const date = new Date(timestamp);
return date.toLocaleTimeString();
}
function formatSeconds(seconds: number): string {
const mins = Math.floor(seconds / 60);
const secs = Math.floor(seconds % 60);
return `${mins}:${secs.toString().padStart(2, '0')}`;
}
</script>
<style scoped>
.room-card {
background: white;
color: #333;
border-radius: 15px;
padding: 20px;
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.2);
}
.room-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 15px;
padding-bottom: 15px;
border-bottom: 2px solid #f0f0f0;
}
.room-id {
font-weight: bold;
font-family: monospace;
}
.room-status {
padding: 5px 15px;
border-radius: 20px;
font-size: 12px;
font-weight: 600;
text-transform: uppercase;
}
.status-waiting {
background: #e8f5e9;
color: #4caf50;
}
.status-playing {
background: #e3f2fd;
color: #2196f3;
}
.status-paused {
background: #fff3e0;
color: #ff9800;
}
.status-finished {
background: #f3e5f5;
color: #9c27b0;
}
.room-details {
margin-bottom: 20px;
}
.detail-row {
display: flex;
justify-content: space-between;
padding: 5px 0;
}
.detail-label {
color: #666;
}
.detail-value {
font-weight: 600;
}
.room-actions {
display: flex;
gap: 10px;
flex-wrap: wrap;
}
.btn {
padding: 8px 16px;
border: none;
border-radius: 8px;
font-size: 14px;
font-weight: 600;
cursor: pointer;
transition: all 0.3s;
}
.btn-action {
flex: 1;
min-width: 80px;
}
.btn-pause {
background: #ff9800;
color: white;
}
.btn-resume {
background: #4caf50;
color: white;
}
.btn-restart {
background: #2196f3;
color: white;
}
.btn-view {
background: #9c27b0;
color: white;
}
.btn-kick {
background: #f44336;
color: white;
padding: 4px 12px;
}
.btn:hover {
transform: translateY(-2px);
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.2);
}
.room-stats {
margin-top: 20px;
padding-top: 20px;
border-top: 2px solid #f0f0f0;
}
.room-stats h4 {
margin-bottom: 15px;
color: #666;
}
.players-list {
margin-bottom: 15px;
}
.player-row {
display: flex;
align-items: center;
padding: 12px;
background: #f8f9fa;
border-radius: 8px;
margin-bottom: 8px;
gap: 12px;
}
.player-info {
flex: 1;
display: flex;
flex-direction: column;
gap: 4px;
}
.player-name {
font-weight: 600;
font-size: 14px;
}
.player-role {
font-size: 12px;
color: #666;
text-transform: uppercase;
}
.player-tokens {
display: flex;
gap: 8px;
align-items: center;
}
.token {
background: white;
padding: 4px 8px;
border-radius: 12px;
font-size: 12px;
font-weight: 600;
border: 1px solid #e0e0e0;
}
.token.pavo {
color: #d84315;
}
.token.elote {
color: #f57f17;
}
.token.shame {
color: #c62828;
}
.stats-info {
background: #f8f9fa;
padding: 15px;
border-radius: 8px;
}
.stat-item {
display: flex;
justify-content: space-between;
padding: 5px 0;
}
.winner-name {
color: #4caf50;
font-weight: bold;
}
</style>