takeone/resources/views/clubs/explore.blade.php

917 lines
36 KiB
PHP

@extends('layouts.app')
@section('content')
<div class="container py-4">
<!-- Hero Section -->
<div class="text-center mb-3">
<h1 class="display-4 fw-bold text-primary mb-2">Find Your Perfect Fit</h1>
<p class="lead text-muted">Discover sports clubs, trainers, nutrition clinics, and more near you</p>
<p class="text-muted"><span id="currentLocation" class="badge bg-primary text-white rounded-pill px-3 py-2"><i class="bi bi-geo-alt-fill me-1 fs-5"></i>Detecting location...</span></p>
</div>
<!-- Search Bar with Near Me Button -->
<div class="row justify-content-center mb-4">
<div class="col-lg-10">
<div class="card shadow-sm rounded-pill border-0">
<div class="card-body rounded-pill p-2">
<div class="input-group rounded-pill">
<span class="input-group-text bg-white border-0">
<i class="bi bi-search"></i>
</span>
<input type="text" id="searchInput" class="form-control bg-white border-0"
placeholder="Search for clubs, trainers, nutrition clinics...">
<button class="btn btn-primary px-4 rounded-pill" id="nearMeBtn" type="button">
<i class="bi bi-geo-alt-fill me-2"></i>Near Me
</button>
</div>
</div>
</div>
</div>
</div>
<!-- Category Tabs -->
<div class="row justify-content-center mb-4">
<div class="col-lg-10">
<div class="d-flex flex-wrap gap-2 justify-content-center">
<button class="btn btn-primary category-btn active" data-category="all">
<i class="bi bi-search me-2"></i>All
</button>
<button class="btn btn-outline-primary category-btn" data-category="sports-clubs">
<i class="bi bi-trophy me-2"></i>Clubs
</button>
<button class="btn btn-outline-primary category-btn" data-category="personal-trainers">
<i class="bi bi-person me-2"></i>Trainers
</button>
<button class="btn btn-outline-primary category-btn" data-category="events">
<i class="bi bi-calendar-event me-2"></i>Events
</button>
<button class="btn btn-outline-primary category-btn" data-category="nutrition-clinic">
<i class="bi bi-apple me-2"></i>Nutrition
</button>
<button class="btn btn-outline-primary category-btn" data-category="physiotherapy-clinics">
<i class="bi bi-activity me-2"></i>Physiotherapy
</button>
<button class="btn btn-outline-primary category-btn" data-category="sports-shops">
<i class="bi bi-bag me-2"></i>Shops
</button>
<button class="btn btn-outline-primary category-btn" data-category="venues">
<i class="bi bi-building-fill me-2"></i>Venues
</button>
<button class="btn btn-outline-primary category-btn" data-category="supplements">
<i class="bi bi-box me-2"></i>Supplements
</button>
<button class="btn btn-outline-primary category-btn" data-category="food-plans">
<i class="bi bi-egg-fried me-2"></i>Food Plans
</button>
</div>
</div>
</div>
<!-- Location Status Alert -->
<div id="locationAlert" class="alert alert-info alert-dismissible fade" role="alert" style="display: none;">
<i class="bi bi-info-circle me-2"></i>
<span id="locationMessage"></span>
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
</div>
<!-- Loading Spinner -->
<div id="loadingSpinner" class="text-center py-5">
<div class="spinner-border text-primary" role="status">
<span class="visually-hidden">Loading...</span>
</div>
<p class="mt-3 text-muted">Finding what's near you...</p>
</div>
<!-- Clubs Grid -->
<div class="row justify-content-center" id="clubsGrid" style="display: none;">
<div class="col-12">
<div class="row row-cols-1 row-cols-md-2 row-cols-lg-3 g-4" id="clubsContainer">
<!-- Club cards will be inserted here -->
</div>
</div>
</div>
<!-- No Results -->
<div class="row justify-content-center" id="noResultsContainer" style="display: none;">
<div class="col-12 d-flex justify-content-center">
<div id="noResults" class="d-flex flex-column align-items-center justify-content-center text-center" style="min-height: 400px;">
<i class="bi bi-inbox" style="font-size: 4rem; color: #dee2e6;"></i>
<h4 class="mt-3 text-muted">No Results Found</h4>
<p class="text-muted">Try adjusting your search or location</p>
</div>
</div>
</div>
</div>
<!-- Map Modal -->
<div class="modal fade" id="mapModal" tabindex="-1" aria-labelledby="mapModalLabel" aria-hidden="true">
<div class="modal-dialog modal-xl modal-dialog-centered">
<div class="modal-content">
<div class="modal-header border-0">
<h5 class="modal-title" id="mapModalLabel">
<i class="bi bi-geo-alt-fill me-2 text-primary"></i>Set Your Location
</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
</div>
<div class="modal-body p-0">
<div id="map" style="height: 600px; width: 100%;"></div>
</div>
<div class="modal-footer border-0 bg-light">
<div class="w-100 d-flex justify-content-between align-items-center">
<small class="text-muted">
<i class="bi bi-geo-alt-fill me-1"></i>
<span id="modalLocationCoordinates">Drag the marker to set your location</span>
</small>
<button type="button" class="btn btn-primary" id="applyLocationBtn">
<i class="bi bi-check-circle me-2"></i>Apply Location
</button>
</div>
</div>
</div>
</div>
</div>
@push('styles')
<!-- Font Awesome -->
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
<!-- Leaflet CSS -->
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css"
integrity="sha256-p4NxAoJBhIIN+hmNHrzRCf9tD/miZyoHS5obTRR9BMY="
crossorigin=""/>
<style>
.category-btn {
border-radius: 50px;
padding: 0.5rem 1.5rem;
font-size: 0.9rem;
transition: all 0.3s ease;
}
.category-btn:hover {
transform: translateY(-2px);
box-shadow: 0 4px 8px rgba(0,0,0,0.1);
color: white !important;
}
.category-btn.active {
transform: translateY(-2px);
box-shadow: 0 4px 8px hsl(var(--primary) / 0.3);
}
.club-card {
transition: all 0.3s ease-in-out;
min-height: 450px;
}
.club-card:hover {
transform: translateY(-8px);
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04) !important;
}
.club-card:hover .club-cover-img {
transform: scale(1.1);
}
.club-card:hover .club-title {
color: #667eea !important;
}
/* Pulsing animation for location marker */
@keyframes pulse {
0% {
transform: scale(1);
opacity: 1;
}
50% {
transform: scale(1.2);
opacity: 0.7;
}
100% {
transform: scale(1);
opacity: 1;
}
}
.pulse-marker {
animation: pulse 2s ease-in-out infinite;
color: #0d6efd;
}
#currentLocation {
font-size: 1.2rem;
}
.input-group.rounded-pill {
overflow: hidden;
}
.input-group.rounded-pill .input-group-text {
border-radius: 50rem 0 0 50rem !important;
}
.input-group.rounded-pill .form-control {
border-radius: 0;
}
.form-control:focus {
border-color: #ced4da !important;
box-shadow: none !important;
}
/* Trainer Card Styles */
.trainer-card {
border: none;
border-radius: 0;
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
transition: all 0.3s ease;
overflow: hidden;
background: white;
min-height: 400px;
}
.trainer-card:hover {
transform: translateY(-5px);
box-shadow: 0 10px 25px rgba(0, 0, 0, 0.15);
}
.trainer-card .row {
height: 192px;
}
.image-container {
position: relative;
height: 100%;
overflow: hidden;
}
.pt-badge {
position: absolute;
top: 10px;
left: 10px;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
color: white;
padding: 5px 15px;
border-radius: 20px;
font-size: 0.8rem;
font-weight: bold;
z-index: 2;
}
.trainer-img {
width: 100%;
height: 100%;
object-fit: cover;
transition: transform 0.3s ease;
}
.trainer-card:hover .trainer-img {
transform: scale(1.05);
}
.info-section {
padding: 15px;
height: 100%;
display: flex;
flex-direction: column;
justify-content: center;
}
.trainer-name {
font-size: 1.2rem;
font-weight: bold;
color: #333;
margin-bottom: 5px;
}
.trainer-title {
font-size: 0.9rem;
color: #666;
margin-bottom: 10px;
}
.feature-list {
list-style: none;
padding: 0;
margin-bottom: 10px;
flex-grow: 1;
}
.feature-list li {
margin-bottom: 5px;
font-size: 0.8rem;
color: #555;
}
.feature-list i {
color: #28a745;
margin-right: 8px;
}
.rating-box {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 10px;
font-size: 0.8rem;
}
.stars {
color: #ffc107;
}
.stars i {
margin-right: 2px;
}
.trainer-buttons {
margin-top: auto;
}
.btn-book {
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
border: none;
color: white;
padding: 8px 15px;
border-radius: 20px;
font-weight: bold;
font-size: 0.8rem;
transition: all 0.3s ease;
}
.btn-book:hover {
background: linear-gradient(135deg, #764ba2 0%, #667eea 100%);
transform: translateY(-2px);
}
.btn-view {
background: transparent;
border: 2px solid #667eea;
color: #667eea;
padding: 8px 15px;
border-radius: 20px;
font-weight: bold;
font-size: 0.8rem;
transition: all 0.3s ease;
}
.btn-view:hover {
background: #667eea;
color: white;
transform: translateY(-2px);
}
</style>
@endpush
@push('scripts')
<!-- Leaflet JS -->
<script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js"
integrity="sha256-20nQCchB9co0qIjJZRGuk2/Z9VM+kNiyxNV1lvTlZBo="
crossorigin=""></script>
<script>
let map;
let pageMap;
let userMarker;
let searchRadiusCircle;
let userLocation = null;
let watchId = null;
let currentCategory = 'all';
let allClubs = [];
let countriesData = [];
// Initialize the page
document.addEventListener('DOMContentLoaded', function() {
// Load countries from JSON file
fetch('/data/countries.json')
.then(response => response.json())
.then(countries => {
countriesData = countries;
})
.catch(error => console.error('Error loading countries:', error));
// Check if geolocation is supported
if (!navigator.geolocation) {
showAlert('Geolocation is not supported by your browser', 'danger');
document.getElementById('loadingSpinner').style.display = 'none';
// If no geolocation, fetch all clubs
fetchAllClubs();
} else {
// Automatically start watching user's location
// This will fetch clubs once location is detected
startWatchingLocation();
}
// Category buttons
document.querySelectorAll('.category-btn').forEach(btn => {
btn.addEventListener('click', function() {
document.querySelectorAll('.category-btn').forEach(b => {
b.classList.remove('active', 'btn-primary');
b.classList.add('btn-outline-primary');
});
this.classList.remove('btn-outline-primary');
this.classList.add('active', 'btn-primary');
currentCategory = this.dataset.category;
// For 'all' and 'sports-clubs' (Clubs) categories, fetch all clubs with location-based sorting
if (currentCategory === 'all' || currentCategory === 'sports-clubs') {
fetchAllClubs();
} else if (userLocation) {
fetchNearbyClubs(userLocation.latitude, userLocation.longitude);
} else {
// If no location, show message or fetch all as fallback
fetchAllClubs();
}
});
});
// Search input
document.getElementById('searchInput').addEventListener('input', function() {
filterClubs();
});
// Near Me button
document.getElementById('nearMeBtn').addEventListener('click', function() {
const mapModal = new bootstrap.Modal(document.getElementById('mapModal'));
const modalElement = document.getElementById('mapModal');
modalElement.addEventListener('shown.bs.modal', function() {
if (userLocation) {
initMap(userLocation.latitude, userLocation.longitude);
updateModalLocation(userLocation.latitude, userLocation.longitude);
} else {
// No location available, use default and let user drag
initMap(25.276987, 55.296249); // Default to Dubai or any location
updateModalLocation(25.276987, 55.296249);
}
}, { once: true });
mapModal.show();
});
// Apply Location button
document.getElementById('applyLocationBtn').addEventListener('click', function() {
const mapModal = bootstrap.Modal.getInstance(document.getElementById('mapModal'));
mapModal.hide();
if (userLocation) {
if (currentCategory === 'all') {
fetchAllClubs();
} else {
fetchNearbyClubs(userLocation.latitude, userLocation.longitude);
}
}
});
});
// Start watching user's location
function startWatchingLocation() {
watchId = navigator.geolocation.watchPosition(
function(position) {
userLocation = {
latitude: position.coords.latitude,
longitude: position.coords.longitude
};
updateLocationDisplay(userLocation.latitude, userLocation.longitude);
// Fetch all clubs with location-based sorting (since 'all' is default)
fetchAllClubs();
// Stop watching after first successful location
if (watchId) {
navigator.geolocation.clearWatch(watchId);
watchId = null;
}
},
function(error) {
let errorMessage = 'Unable to get your location. ';
switch(error.code) {
case error.PERMISSION_DENIED:
errorMessage += 'Please allow location access.';
break;
case error.POSITION_UNAVAILABLE:
errorMessage += 'Location unavailable.';
break;
case error.TIMEOUT:
errorMessage += 'Request timed out.';
break;
}
showAlert(errorMessage, 'danger');
// If location fails and category is not 'all', perhaps fetch all as fallback
if (currentCategory !== 'all') {
fetchAllClubs();
}
},
{
enableHighAccuracy: true,
timeout: 10000,
maximumAge: 0
}
);
}
// Update location display
function updateLocationDisplay(lat, lng) {
document.getElementById('currentLocation').innerHTML =
`<i class="bi bi-geo-alt-fill me-1 fs-5"></i>${lat.toFixed(4)}, ${lng.toFixed(4)}`;
}
// Initialize map in modal
function initMap(lat, lng) {
if (map) {
map.remove();
}
map = L.map('map', { attributionControl: false }).setView([lat, lng], 13);
L.tileLayer('http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
attribution: '&copy; OpenStreetMap contributors',
maxZoom: 19
}).addTo(map);
// Add draggable marker
userMarker = L.marker([lat, lng], {
draggable: true,
icon: L.divIcon({
className: 'user-location-marker',
html: '<i class="bi bi-geo-alt-fill pulse-marker" style="font-size: 36px; color: #667eea; filter: drop-shadow(0 3px 6px rgba(0,0,0,0.4));"></i>',
iconSize: [36, 36],
iconAnchor: [18, 36]
})
}).addTo(map);
// Drag event
userMarker.on('drag', function(event) {
const position = event.target.getLatLng();
userLocation = {
latitude: position.lat,
longitude: position.lng
};
updateLocationDisplay(position.lat, position.lng);
updateModalLocation(position.lat, position.lng);
});
setTimeout(() => map.invalidateSize(), 100);
}
// Fetch nearby clubs
function fetchNearbyClubs(lat, lng) {
document.getElementById('loadingSpinner').style.display = 'block';
document.getElementById('clubsGrid').style.display = 'none';
fetch(`{{ route('clubs.nearby') }}?latitude=${lat}&longitude=${lng}&radius=50`, {
method: 'GET',
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': '{{ csrf_token() }}'
}
})
.then(response => response.json())
.then(data => {
document.getElementById('loadingSpinner').style.display = 'none';
document.getElementById('clubsGrid').style.display = 'block';
if (data.success) {
allClubs = data.clubs;
displayClubs(allClubs);
} else {
showAlert('Failed to fetch clubs', 'danger');
}
})
.catch(error => {
console.error('Error:', error);
document.getElementById('loadingSpinner').style.display = 'none';
showAlert('Error fetching clubs', 'danger');
});
}
// Fetch all clubs
function fetchAllClubs() {
document.getElementById('loadingSpinner').style.display = 'block';
document.getElementById('clubsGrid').style.display = 'none';
// Build URL with location parameters if available
let url = `{{ route('clubs.all') }}`;
if (userLocation) {
url += `?latitude=${userLocation.latitude}&longitude=${userLocation.longitude}`;
}
fetch(url, {
method: 'GET',
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': '{{ csrf_token() }}'
}
})
.then(response => response.json())
.then(data => {
document.getElementById('loadingSpinner').style.display = 'none';
document.getElementById('clubsGrid').style.display = 'block';
if (data.success) {
allClubs = data.clubs;
displayClubs(allClubs);
} else {
showAlert('Failed to fetch clubs', 'danger');
}
})
.catch(error => {
console.error('Error:', error);
document.getElementById('loadingSpinner').style.display = 'none';
showAlert('Error fetching clubs', 'danger');
});
}
// Display clubs as cards
function displayClubs(clubs) {
const container = document.getElementById('clubsContainer');
const noResultsContainer = document.getElementById('noResultsContainer');
container.innerHTML = '';
let trainerAdded = false;
// Add dummy trainer card if category is 'all' or 'personal-trainers'
if (currentCategory === 'all' || currentCategory === 'personal-trainers') {
const trainerCard = document.createElement('div');
trainerCard.className = 'col';
trainerCard.innerHTML = `
<div class="card border shadow-sm overflow-hidden club-card" style="border-radius: 0; cursor: pointer; transition: all 0.3s ease;">
<!-- Cover Image -->
<div class="position-relative overflow-hidden" style="height: 192px;">
<img src="https://images.unsplash.com/photo-1583454110551-21f2fa2afe61?q=80&w=2070&auto=format&fit=crop"
alt="Personal Trainer"
loading="lazy"
class="w-100 h-100"
style="object-fit: cover; transition: transform 0.3s ease;">
<!-- Personal Trainer Badge -->
<div class="position-absolute" style="top: 8px; left: 8px;">
<span class="badge text-white px-3 py-1" style="background-color: #dc3545; border-radius: 9999px; font-size: 0.75rem; font-weight: 600;"><i class="fa-solid fa-user me-1"></i>Personal Trainer</span>
</div>
</div>
<!-- Card Body -->
<div class="p-4" style="background-color: white;">
<div class="mb-3">
<!-- Trainer Name -->
<h3 class="fw-semibold mb-2 club-title" style="font-size: 1.125rem; color: #1f2937; transition: color 0.3s ease;">Alex Thompson</h3>
<!-- Distance -->
<div class="d-flex align-items-center mb-1" style="font-size: 0.875rem; color: #667eea;">
<i class="fa fa-certificate me-1"></i>
<span class="fw-semibold">Certified Strength & Conditioning Coach</span>
</div>
<!-- Location -->
<div class="d-flex align-items-center text-muted" style="font-size: 0.875rem;">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="me-1 flex-shrink-0">
<path d="M20 10c0 4.993-5.539 10.193-7.399 11.799a1 1 0 0 1-1.202 0C9.539 20.193 4 14.993 4 10a8 8 0 0 1 16 0"></path>
<circle cx="12" cy="10" r="3"></circle>
</svg>
<span class="text-truncate">Ghassan Yusuf</span>
</div>
</div>
<div class="row g-2 text-center mb-3" style="font-size: 0.75rem;">
<div class="col-4">
<div class="p-2 rounded" style="background-color: rgba(102, 126, 234, 0.05);">
<i class="fa-solid fa-calendar mb-1" style="color: #6b7280; font-size: 1rem;"></i>
<p class="fw-semibold mb-0" style="color: #1f2937;">13</p>
<p class="text-muted mb-0">Years Exp.</p>
</div>
</div>
<div class="col-4">
<div class="p-2 rounded" style="background-color: rgba(102, 126, 234, 0.05);">
<i class="fa-solid fa-certificate mb-1" style="color: #6b7280; font-size: 1rem;"></i>
<p class="fw-semibold mb-0" style="color: #1f2937;">NASM</p>
<p class="text-muted mb-0">Packages</p>
</div>
</div>
<div class="col-4">
<div class="p-2 rounded" style="background-color: rgba(102, 126, 234, 0.05);">
<i class="fa-solid fa-star"></i>
<p class="fw-semibold mb-0" style="color: #1f2937;">5.0</p>
<p class="text-muted mb-0">Rating</p>
</div>
</div>
</div>
<!-- Action Buttons -->
<div class="d-flex gap-2">
<button class="btn btn-primary flex-fill fw-semibold" style="font-size: 0.875rem;">
<i class="fa-solid fa-calendar-plus me-1"></i>Book Session
</button>
<button class="btn btn-outline-primary flex-fill fw-semibold" style="font-size: 0.875rem;">
View Details
</button>
</div>
</div>
</div>
`;
container.appendChild(trainerCard);
trainerAdded = true;
}
if (clubs.length === 0 && !trainerAdded) {
noResultsContainer.style.display = 'flex';
return;
}
noResultsContainer.style.display = 'none';
clubs.forEach(club => {
const card = document.createElement('div');
card.className = 'col';
// Prepare cover image
let coverImageHtml = '';
if (club.cover_image) {
coverImageHtml = `<img src="/storage/${club.cover_image}" alt="${club.club_name}" loading="lazy" class="w-100 h-100 club-cover-img" style="object-fit: cover; transition: transform 0.3s ease;">`;
} else {
coverImageHtml = `<div class="w-100 h-100 d-flex align-items-center justify-content-center" style="background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);">
<i class="bi bi-image text-white" style="font-size: 3rem; opacity: 0.3;"></i>
</div>`;
}
// Prepare logo
let logoHtml = '';
if (club.logo) {
logoHtml = `<img src="/storage/${club.logo}" alt="${club.club_name} logo" loading="lazy" class="w-100 h-100 rounded-circle" style="object-fit: contain;">`;
} else {
logoHtml = `<div class="w-100 h-100 rounded-circle bg-primary d-flex align-items-center justify-content-center">
<span class="text-white fw-bold fs-4">${club.club_name.charAt(0)}</span>
</div>`;
}
card.innerHTML = `
<div class="card border shadow-sm overflow-hidden club-card" style="border-radius: 0; cursor: pointer; transition: all 0.3s ease;">
<!-- Cover Image -->
<div class="position-relative overflow-hidden" style="height: 192px;">
${coverImageHtml}
<!-- Club Logo - Bottom Left -->
<div class="position-absolute" style="bottom: 8px; left: 8px;">
<div class="bg-white shadow border p-0.5" style="width: 80px; height: 80px; border-radius: 50%; border-color: rgba(0,0,0,0.1) !important;">
${logoHtml}
</div>
</div>
<!-- Sports Club Badge - Top Left -->
<div class="position-absolute" style="top: 8px; left: 8px;">
<span class="badge text-white px-3 py-1" style="background-color: #dc3545; border-radius: 9999px; font-size: 0.75rem; font-weight: 600;"><i class="fa-solid fa-building me-1"></i>Sports Club</span>
</div>
</div>
<!-- Card Body -->
<div class="p-4" style="background-color: white;">
<div class="mb-3">
<!-- Club Name -->
<h3 class="fw-semibold mb-2 club-title" style="font-size: 1.125rem; color: #1f2937; transition: color 0.3s ease;">${club.club_name}</h3>
<!-- Distance -->
<div class="d-flex align-items-center mb-1" style="font-size: 0.875rem; color: #667eea;">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="me-1 flex-shrink-0">
<polygon points="3 11 22 2 13 21 11 13 3 11"></polygon>
</svg>
<span class="fw-semibold">${club.distance ? club.distance + ' km away' : 'Location available'}</span>
</div>
<!-- Address/Owner -->
<div class="d-flex align-items-center text-muted" style="font-size: 0.875rem;">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="me-1 flex-shrink-0">
<path d="M20 10c0 4.993-5.539 10.193-7.399 11.799a1 1 0 0 1-1.202 0C9.539 20.193 4 14.993 4 10a8 8 0 0 1 16 0"></path>
<circle cx="12" cy="10" r="3"></circle>
</svg>
<span class="text-truncate">${club.owner_name || 'N/A'}</span>
</div>
</div>
<!-- Stats Grid -->
<div class="row g-2 text-center mb-3" style="font-size: 0.75rem;">
<div class="col-4">
<div class="p-2 rounded" style="background-color: rgba(102, 126, 234, 0.05);">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="mx-auto mb-1" style="color: #667eea;">
<path d="M16 21v-2a4 4 0 0 0-4-4H6a4 4 0 0 0-4 4v2"></path>
<circle cx="9" cy="7" r="4"></circle>
<path d="M22 21v-2a4 4 0 0 0-3-3.87"></path>
<path d="M16 3.13a4 4 0 0 1 0 7.75"></path>
</svg>
<p class="fw-semibold mb-0" style="color: #1f2937;">13</p>
<p class="text-muted mb-0">Members</p>
</div>
</div>
<div class="col-4">
<div class="p-2 rounded" style="background-color: rgba(102, 126, 234, 0.05);">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="mx-auto mb-1" style="color: #667eea;">
<path d="M14.4 14.4 9.6 9.6"></path>
<path d="M18.657 21.485a2 2 0 1 1-2.829-2.828l-1.767 1.768a2 2 0 1 1-2.829-2.829l6.364-6.364a2 2 0 1 1 2.829 2.829l-1.768 1.767a2 2 0 1 1 2.828 2.829z"></path>
<path d="m21.5 21.5-1.4-1.4"></path>
<path d="M3.9 3.9 2.5 2.5"></path>
<path d="M6.404 12.768a2 2 0 1 1-2.829-2.829l1.768-1.767a2 2 0 1 1-2.828-2.829l2.828-2.828a2 2 0 1 1 2.829 2.828l1.767-1.768a2 2 0 1 1 2.829 2.829z"></path>
</svg>
<p class="fw-semibold mb-0" style="color: #1f2937;">0</p>
<p class="text-muted mb-0">Packages</p>
</div>
</div>
<div class="col-4">
<div class="p-2 rounded" style="background-color: rgba(102, 126, 234, 0.05);">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="mx-auto mb-1" style="color: #667eea;">
<path d="M11.525 2.295a.53.53 0 0 1 .95 0l2.31 4.679a2.123 2.123 0 0 0 1.595 1.16l5.166.756a.53.53 0 0 1 .294.904l-3.736 3.638a2.123 2.123 0 0 0-.611 1.878l.882 5.14a.53.53 0 0 1-.771.56l-4.618-2.428a2.122 2.122 0 0 0-1.973 0L6.396 21.01a.53.53 0 0 1-.77-.56l.881-5.139a2.122 2.122 0 0 0-.611-1.879L2.16 9.795a.53.53 0 0 1 .294-.906l5.165-.755a2.122 2.122 0 0 0 1.597-1.16z"></path>
</svg>
<p class="fw-semibold mb-0" style="color: #1f2937;">0</p>
<p class="text-muted mb-0">Trainers</p>
</div>
</div>
</div>
<!-- Action Buttons -->
<div class="d-flex gap-2">
<button class="btn btn-primary flex-fill fw-semibold" style="font-size: 0.875rem;">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="me-1">
<path d="M16 21v-2a4 4 0 0 0-4-4H6a4 4 0 0 0-4 4v2"></path>
<circle cx="9" cy="7" r="4"></circle>
<line x1="19" x2="19" y1="8" y2="14"></line>
<line x1="22" x2="16" y1="11" y2="11"></line>
</svg>
Join Club
</button>
<button class="btn btn-outline-primary flex-fill fw-semibold" style="font-size: 0.875rem;">View Details</button>
</div>
</div>
</div>
`;
container.appendChild(card);
});
}
// Filter clubs
function filterClubs() {
const searchTerm = document.getElementById('searchInput').value.toLowerCase();
let filtered = allClubs.filter(club => {
const matchesSearch = club.club_name.toLowerCase().includes(searchTerm) ||
(club.owner_name && club.owner_name.toLowerCase().includes(searchTerm));
// Add category filtering logic here when categories are available in data
return matchesSearch;
});
displayClubs(filtered);
}
// Show alert
function showAlert(message, type = 'danger') {
const alert = document.getElementById('locationAlert');
const messageSpan = document.getElementById('locationMessage');
alert.style.display = 'block';
alert.className = `alert alert-${type} alert-dismissible fade show`;
messageSpan.textContent = message;
}
// Reverse geocode to get address
async function reverseGeocode(lat, lng) {
try {
const response = await fetch(`https://nominatim.openstreetmap.org/reverse?format=json&lat=${lat}&lon=${lng}&zoom=18`);
const data = await response.json();
return data.address || null;
} catch (error) {
console.error('Reverse geocoding error:', error);
return null;
}
}
// Get country info from address
function getCountryInfo(address) {
if (!address || !countriesData.length) return null;
const iso2 = address.country_code?.toUpperCase();
if (!iso2) return null;
const country = countriesData.find(c => c.iso2 === iso2);
if (country) {
const flag = iso2.split('').map(char => String.fromCodePoint(127397 + char.charCodeAt(0))).join('');
return { flag, name: country.name, iso3: country.iso3 };
}
return null;
}
// Update modal location display with country and area
async function updateModalLocation(lat, lng) {
const address = await reverseGeocode(lat, lng);
const info = getCountryInfo(address);
const coords = `Latitude: ${lat.toFixed(6)}, Longitude: ${lng.toFixed(6)}`;
const area = address?.suburb || address?.town || address?.city || address?.state || address?.county || '';
if (info) {
document.getElementById('modalLocationCoordinates').innerHTML = `${info.name}${area ? ', ' + area : ''} - ${coords}`;
} else {
document.getElementById('modalLocationCoordinates').textContent = `${area ? area + ' - ' : ''}${coords}`;
}
}
</script>
@endpush
@endsection