🎨 Update TournamentGrid component styling and layout

This commit is contained in:
2026-01-29 05:32:54 +00:00
parent 66eae1ddcd
commit 38603a46d8

View File

@@ -0,0 +1,401 @@
<!--
TournamentGrid Component
Displays a grid of tournaments with search, filtering, and pagination.
Features:
- Tournament card display with state badges
- Client-side search filtering
- Pagination support (v2.1 only)
- Click-to-expand details
- Empty state messaging
Props:
- tournaments: Array of tournament objects
- loading: Boolean for loading state
- loadingMore: Boolean for pagination loading
- searchQuery: String for search input (v-model)
- filteredTournaments: Computed array of filtered results
- expandedTournamentId: ID of currently expanded tournament
- hasNextPage: Boolean for pagination availability (v2.1)
- apiVersion: String ('v1' or 'v2.1')
Events:
- update:searchQuery: Emitted when search input changes
- load-more: Emitted when "Load More" button clicked
- toggle-details: Emitted when tournament details button clicked
-->
<template>
<div class="tournament-grid">
<!-- Loading State -->
<div v-if="loading" class="status loading"> Loading tournaments...</div>
<!-- Error State -->
<div v-else-if="error" class="status error">
Error: {{ error }}
</div>
<!-- Empty State -->
<div v-else-if="!tournaments || tournaments.length === 0" class="status empty">
No tournaments found. Create one at
<a href="https://challonge.com" target="_blank">Challonge.com</a>
</div>
<!-- Tournament List -->
<div v-else>
<!-- Search Filter -->
<div class="search-box">
<input
:value="searchQuery"
@input="$emit('update:searchQuery', $event.target.value)"
type="text"
placeholder="🔍 Search tournaments by name (client-side)..."
class="search-input"
/>
<span v-if="searchQuery" class="search-info">
Showing {{ filteredTournaments.length }} of
{{ tournaments.length }} tournaments
</span>
</div>
<!-- Tournament Cards -->
<div class="tournament-list">
<div
v-for="tournament in filteredTournaments"
:key="getTournamentId(tournament)"
class="tournament-card"
>
<div class="tournament-header">
<h4>{{ getTournamentName(tournament) }}</h4>
<span
class="tournament-state"
:class="getTournamentProp(tournament, 'state')"
>
{{ getTournamentProp(tournament, 'state') }}
</span>
</div>
<div class="tournament-details">
<p>
<strong>URL:</strong>
{{ getTournamentProp(tournament, 'url') }}
</p>
<p>
<strong>Type:</strong>
{{ getTournamentProp(tournament, 'tournament_type') }}
</p>
<p>
<strong>Participants:</strong>
{{ getTournamentProp(tournament, 'participants_count') }}
</p>
<p v-if="getTournamentProp(tournament, 'started_at')">
<strong>Started:</strong>
{{ formatDate(getTournamentProp(tournament, 'started_at')) }}
</p>
</div>
<button
@click="$emit('toggle-details', getTournamentId(tournament))"
class="btn btn-small"
:class="{
'btn-active': expandedTournamentId === getTournamentId(tournament)
}"
>
{{
expandedTournamentId === getTournamentId(tournament)
? 'Hide Details'
: 'Load Details'
}}
</button>
<!-- Details Slot for expanded content -->
<slot
name="tournament-details"
:tournament="tournament"
:is-expanded="expandedTournamentId === getTournamentId(tournament)"
></slot>
</div>
</div>
<!-- Load More Button (v2.1 only) -->
<div
v-if="apiVersion === 'v2.1' && hasNextPage"
class="load-more-section"
>
<button
@click="$emit('load-more')"
:disabled="loadingMore"
class="btn btn-secondary"
>
{{ loadingMore ? 'Loading...' : 'Load More Tournaments' }}
</button>
</div>
</div>
</div>
</template>
<script setup>
/**
* Props for TournamentGrid component
*/
const props = defineProps({
tournaments: {
type: Array,
default: null
},
loading: {
type: Boolean,
default: false
},
loadingMore: {
type: Boolean,
default: false
},
error: {
type: String,
default: null
},
searchQuery: {
type: String,
default: ''
},
filteredTournaments: {
type: Array,
default: () => []
},
expandedTournamentId: {
type: [String, Number],
default: null
},
hasNextPage: {
type: Boolean,
default: false
},
apiVersion: {
type: String,
default: 'v2.1'
}
});
/**
* Events emitted by TournamentGrid
*/
defineEmits(['update:searchQuery', 'load-more', 'toggle-details']);
/**
* Helper to get tournament name (handles both v1 and v2.1 response structures)
*/
function getTournamentName(tournament) {
return tournament.tournament?.name || tournament.name || '';
}
/**
* Helper to get tournament ID
*/
function getTournamentId(tournament) {
return tournament.tournament?.id || tournament.id;
}
/**
* Helper to get tournament property
*/
function getTournamentProp(tournament, prop) {
return tournament.tournament?.[prop] || tournament[prop];
}
/**
* Format date string
*/
function formatDate(dateString) {
if (!dateString) return '';
return new Date(dateString).toLocaleString();
}
</script>
<style scoped>
.tournament-grid {
margin-top: 1.5rem;
}
.status {
padding: 1.5rem;
border-radius: 8px;
text-align: center;
margin: 1rem 0;
}
.status.loading {
background: #e3f2fd;
color: #1976d2;
font-weight: 500;
}
.status.error {
background: #ffebee;
color: #c62828;
font-weight: 500;
}
.status.empty {
background: #f5f5f5;
color: #616161;
font-weight: 500;
}
.status a {
color: #667eea;
text-decoration: underline;
}
.search-box {
margin-bottom: 1.5rem;
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.search-input {
width: 100%;
padding: 0.75rem 1rem;
border: 2px solid #e0e0e0;
border-radius: 8px;
font-size: 1rem;
transition: all 0.2s;
}
.search-input:focus {
outline: none;
border-color: #667eea;
box-shadow: 0 0 0 3px rgba(102, 126, 234, 0.1);
}
.search-info {
font-size: 0.875rem;
color: #666;
font-style: italic;
}
.tournament-list {
display: grid;
gap: 1rem;
}
.tournament-card {
background: white;
border: 2px solid #e0e0e0;
border-radius: 8px;
padding: 1.5rem;
transition: all 0.2s;
}
.tournament-card:hover {
border-color: #667eea;
box-shadow: 0 4px 12px rgba(102, 126, 234, 0.15);
}
.tournament-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 1rem;
gap: 1rem;
}
.tournament-header h4 {
margin: 0;
color: #333;
font-size: 1.25rem;
flex: 1;
}
.tournament-state {
padding: 0.25rem 0.75rem;
border-radius: 12px;
font-size: 0.75rem;
font-weight: 600;
text-transform: uppercase;
white-space: nowrap;
}
.tournament-state.pending {
background: #fff3cd;
color: #856404;
}
.tournament-state.underway,
.tournament-state.in_progress {
background: #d1ecf1;
color: #0c5460;
}
.tournament-state.complete,
.tournament-state.ended {
background: #d4edda;
color: #155724;
}
.tournament-details {
margin-bottom: 1rem;
}
.tournament-details p {
margin: 0.5rem 0;
color: #666;
font-size: 0.9rem;
}
.tournament-details strong {
color: #333;
}
.btn {
padding: 0.5rem 1rem;
border: none;
border-radius: 6px;
font-size: 0.9rem;
font-weight: 500;
cursor: pointer;
transition: all 0.2s;
}
.btn-small {
background: #667eea;
color: white;
}
.btn-small:hover {
background: #5568d3;
transform: translateY(-1px);
box-shadow: 0 4px 8px rgba(102, 126, 234, 0.3);
}
.btn-small.btn-active {
background: #764ba2;
}
.btn-small.btn-active:hover {
background: #653c8a;
}
.load-more-section {
margin-top: 1.5rem;
text-align: center;
}
.btn-secondary {
background: #6c757d;
color: white;
padding: 0.75rem 1.5rem;
font-size: 1rem;
}
.btn-secondary:hover:not(:disabled) {
background: #5a6268;
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(108, 117, 125, 0.3);
}
.btn-secondary:disabled {
background: #adb5bd;
cursor: not-allowed;
opacity: 0.6;
}
</style>