takeone/resources/views/family/show.blade.php
2026-01-27 16:34:46 +03:00

2529 lines
144 KiB
PHP

@extends('layouts.app')
@php
function calculateTimeDifference($date1, $date2) {
$diff = $date1->diff($date2);
$parts = [];
if ($diff->y > 0) $parts[] = $diff->y . ' year' . ($diff->y > 1 ? 's' : '');
if ($diff->m > 0) $parts[] = $diff->m . ' month' . ($diff->m > 1 ? 's' : '');
if ($diff->d > 0) $parts[] = $diff->d . ' day' . ($diff->d > 1 ? 's' : '');
return implode(' ', $parts) ?: 'Same day';
}
@endphp
@section('content')
<div class="container py-4">
<!-- Flash Messages -->
@if(session('success'))
<div class="alert alert-success alert-dismissible fade show" role="alert">
{{ session('success') }}
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
</div>
@endif
@if(session('error'))
<div class="alert alert-danger alert-dismissible fade show" role="alert">
{{ session('error') }}
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
</div>
@endif
<!-- Header -->
<div class="d-flex justify-content-between align-items-center mb-4">
<div>
<h2 class="fw-bold mb-1">Member Profile</h2>
<p class="text-muted mb-0">Comprehensive member information and analytics</p>
</div>
</div>
<!-- Profile Card -->
<div class="card shadow-sm border-0 mb-4">
<div class="d-flex">
<!-- Profile Picture -->
<div style="width: 180px; min-height: 250px; border-radius: 0.375rem 0 0 0.375rem;">
@if($relationship->dependent->profile_picture)
<img src="{{ asset('storage/' . $relationship->dependent->profile_picture) }}" alt="{{ $relationship->dependent->full_name }}" class="w-100 h-100" style="object-fit: cover; border-radius: 0.375rem 0 0 0.375rem;">
@else
<div class="w-100 h-100 d-flex align-items-center justify-content-center text-white fw-bold" style="font-size: 3rem; background: linear-gradient(135deg, {{ $relationship->dependent->gender == 'm' ? '#0d6efd 0%, #0a58ca 100%' : '#d63384 0%, #a61e4d 100%' }}); border-radius: 0.375rem 0 0 0.375rem;">
{{ strtoupper(substr($relationship->dependent->full_name, 0, 1)) }}
</div>
@endif
</div>
<!-- Profile Info -->
<div class="flex-grow-1 p-4">
<div class="d-flex justify-content-between align-items-start mb-2">
<h3 class="fw-bold mb-0">{{ $relationship->dependent->full_name }}</h3>
@if($relationship->relationship_type == 'self' || Auth::id() == $relationship->guardian_user_id)
<div>
<div class="dropdown">
<button class="btn btn-primary rounded-pill dropdown-toggle" type="button" id="actionDropdown" data-bs-toggle="dropdown" aria-expanded="false">
<i class="bi bi-lightning me-1"></i>Action
</button>
<ul class="dropdown-menu dropdown-menu-end" aria-labelledby="actionDropdown">
<li><a class="dropdown-item" href="#"><i class="bi bi-trophy me-2"></i>Add Achievement</a></li>
<li><a class="dropdown-item" href="#"><i class="bi bi-calendar-check me-2"></i>Add Attendance Record</a></li>
<li><a class="dropdown-item" href="#"><i class="bi bi-calendar-event me-2"></i>Add Event Participation</a></li>
<li><a class="dropdown-item" href="#" data-bs-target="#healthUpdateModal"><i class="bi bi-heart-pulse me-2"></i>Add Health Update</a></li>
<li><a class="dropdown-item" href="#" data-bs-toggle="modal" data-bs-target="#tournamentParticipationModal"><i class="bi bi-award me-2"></i>Add Tournament Participation</a></li>
<li><a class="dropdown-item" href="@if($relationship->relationship_type == 'self'){{ route('profile.edit') }}@else{{ route('family.edit', $relationship->dependent->id) }}@endif">
<i class="bi bi-pencil me-2"></i>Edit Info
</a></li>
<li><a class="dropdown-item" href="#"><i class="bi bi-bullseye me-2"></i>Set a Goal</a></li>
</ul>
</div>
</div>
@else
<button class="btn btn-primary btn-sm rounded-pill">
<i class="bi bi-person-plus me-1"></i>Follow
</button>
@endif
</div>
@if($relationship->dependent->motto)
<p class="text-muted fst-italic mb-3">"{{ $relationship->dependent->motto }}"</p>
@endif
<!-- Achievement Badges -->
<div class="d-flex gap-2 mb-3 flex-wrap">
<a href="#" class="border bg-white rounded px-2 py-1 text-decoration-none achievement-badge" style="font-size: 1rem;" data-medal-type="special" onclick="filterTournamentsByMedal('special')">🏆 <span class="fw-semibold text-dark">{{ $awardCounts['special'] }}</span></a>
<a href="#" class="border bg-white rounded px-2 py-1 text-decoration-none achievement-badge" style="font-size: 1rem;" data-medal-type="1st" onclick="filterTournamentsByMedal('1st')">🥇 <span class="fw-semibold text-dark">{{ $awardCounts['1st'] }}</span></a>
<a href="#" class="border bg-white rounded px-2 py-1 text-decoration-none achievement-badge" style="font-size: 1rem;" data-medal-type="2nd" onclick="filterTournamentsByMedal('2nd')">🥈 <span class="fw-semibold text-dark">{{ $awardCounts['2nd'] }}</span></a>
<a href="#" class="border bg-white rounded px-2 py-1 text-decoration-none achievement-badge" style="font-size: 1rem;" data-medal-type="3rd" onclick="filterTournamentsByMedal('3rd')">🥉 <span class="fw-semibold text-dark">{{ $awardCounts['3rd'] }}</span></a>
<a href="#goals" class="border bg-white rounded px-2 py-1 text-decoration-none" style="font-size: 1rem;" onclick="document.getElementById('goals-tab').click();">🎯 <span class="fw-semibold text-dark">{{ $activeGoalsCount + $completedGoalsCount }}</span></a>
<a href="#" class="border bg-white rounded px-2 py-1 text-decoration-none" style="font-size: 1rem;"> <span class="fw-semibold text-dark">{{ $totalAffiliations }}</span></a>
</div>
<!-- Status Badges -->
<div class="d-flex gap-3 mb-3 align-items-center flex-wrap">
<span class="text-muted small">
<span class="fw-semibold text-dark nationality-display" data-iso3="{{ $relationship->dependent->nationality }}">{{ $relationship->dependent->nationality }}</span>
</span>
<span class="text-muted small">
<i class="bi bi-{{ $relationship->dependent->gender == 'm' ? 'gender-male' : 'gender-female' }} me-1"></i>
<span class="fw-semibold text-dark">{{ $relationship->dependent->gender == 'm' ? 'Male' : 'Female' }}</span>
</span>
<span class="text-muted small">
<i class="bi bi-calendar-event me-1"></i>
Age <span class="fw-semibold text-dark">{{ $relationship->dependent->age }}</span>
</span>
<span class="text-muted small">
@php
$horoscopeSymbols = [
'Aries' => '♈',
'Taurus' => '♉',
'Gemini' => '♊',
'Cancer' => '♋',
'Leo' => '♌',
'Virgo' => '♍',
'Libra' => '♎',
'Scorpio' => '♏',
'Sagittarius' => '♐',
'Capricorn' => '♑',
'Aquarius' => '♒',
'Pisces' => '♓'
];
$horoscope = $relationship->dependent->horoscope ?? 'N/A';
$symbol = $horoscopeSymbols[$horoscope] ?? '';
@endphp
{{ $symbol }} <span class="fw-semibold text-dark">{{ $horoscope }}</span>
</span>
<span class="text-muted small">
<i class="bi bi-check-circle-fill text-success me-1"></i>
<span class="fw-semibold text-success">Active</span>
</span>
<span class="text-muted small">
<i class="bi bi-calendar-check me-1"></i>
Joined <span class="fw-semibold text-dark">{{ $relationship->dependent->created_at->format('F Y') }}</span>
</span>
</div>
<!-- Social Media Icons -->
@if($relationship->dependent->social_links && count($relationship->dependent->social_links) > 0)
<div class="d-flex gap-2 flex-wrap">
@php
$socialLinks = $relationship->dependent->social_links;
ksort($socialLinks); // Sort by platform name
$socialIcons = [
'facebook' => 'bi-facebook',
'twitter' => 'X', // Special case for Twitter/X
'instagram' => 'bi-instagram',
'linkedin' => 'bi-linkedin',
'youtube' => 'bi-youtube',
'tiktok' => 'bi-tiktok',
'snapchat' => 'bi-snapchat',
'whatsapp' => 'bi-whatsapp',
'telegram' => 'bi-telegram',
'discord' => 'bi-discord',
'reddit' => 'bi-reddit',
'pinterest' => 'bi-pinterest',
'twitch' => 'bi-twitch',
'github' => 'bi-github',
'spotify' => 'bi-spotify',
'skype' => 'bi-skype',
'slack' => 'bi-slack',
'medium' => 'bi-medium',
'vimeo' => 'bi-vimeo',
'messenger' => 'bi-messenger',
'wechat' => 'bi-wechat',
'line' => 'bi-line',
];
$socialTitles = [
'facebook' => 'Facebook',
'twitter' => 'Twitter/X',
'instagram' => 'Instagram',
'linkedin' => 'LinkedIn',
'youtube' => 'YouTube',
'tiktok' => 'TikTok',
'snapchat' => 'Snapchat',
'whatsapp' => 'WhatsApp',
'telegram' => 'Telegram',
'discord' => 'Discord',
'reddit' => 'Reddit',
'pinterest' => 'Pinterest',
'twitch' => 'Twitch',
'github' => 'GitHub',
'spotify' => 'Spotify',
'skype' => 'Skype',
'slack' => 'Slack',
'medium' => 'Medium',
'vimeo' => 'Vimeo',
'messenger' => 'Messenger',
'wechat' => 'WeChat',
'line' => 'Line',
];
@endphp
@foreach($socialLinks as $platform => $url)
@if(!empty($url) && isset($socialIcons[$platform]))
<a href="{{ $url }}" target="_blank" rel="noopener noreferrer" class="btn btn-sm btn-outline-secondary rounded-circle" style="width: 32px; height: 32px; padding: 0; display: flex; align-items: center; justify-content: center;" title="{{ $socialTitles[$platform] ?? ucfirst($platform) }}">
@if($platform === 'twitter')
<span style="font-weight: bold; font-size: 1.2rem;">{{ $socialIcons[$platform] }}</span>
@else
<i class="bi {{ $socialIcons[$platform] }}"></i>
@endif
</a>
@endif
@endforeach
</div>
@endif
</div>
</div>
</div>
<!-- Navigation Tabs -->
<ul class="nav nav-tabs nav-fill mb-4" id="profileTabs" role="tablist">
<li class="nav-item" role="presentation">
<button class="nav-link active text-dark" id="overview-tab" data-bs-toggle="tab" data-bs-target="#overview" type="button" role="tab">
<i class="bi bi-eye me-2"></i>Overview
</button>
</li>
<li class="nav-item" role="presentation">
<button class="nav-link text-dark" id="attendance-tab" data-bs-toggle="tab" data-bs-target="#attendance" type="button" role="tab">
<i class="bi bi-calendar-check me-2"></i>Attendance
</button>
</li>
<li class="nav-item" role="presentation">
<button class="nav-link text-dark" id="health-tab" data-bs-toggle="tab" data-bs-target="#health" type="button" role="tab">
<i class="bi bi-heart-pulse me-2"></i>Health
</button>
</li>
<li class="nav-item" role="presentation">
<button class="nav-link text-dark" id="goals-tab" data-bs-toggle="tab" data-bs-target="#goals" type="button" role="tab">
<i class="bi bi-bullseye me-2"></i>Goals
</button>
</li>
<li class="nav-item" role="presentation">
<button class="nav-link text-dark" id="affiliations-tab" data-bs-toggle="tab" data-bs-target="#affiliations" type="button" role="tab">
<i class="bi bi-diagram-3 me-2"></i>Affiliations
</button>
</li>
<li class="nav-item" role="presentation">
<button class="nav-link text-dark" id="tournaments-tab" data-bs-toggle="tab" data-bs-target="#tournaments" type="button" role="tab">
<i class="bi bi-award me-2"></i>Tournaments
</button>
</li>
<li class="nav-item" role="presentation">
<button class="nav-link text-dark" id="events-tab" data-bs-toggle="tab" data-bs-target="#events" type="button" role="tab">
<i class="bi bi-calendar-event me-2"></i>Events
</button>
</li>
</ul>
<!-- Tab Content -->
<div class="tab-content" id="profileTabsContent">
<!-- Overview Tab -->
<div class="tab-pane fade show active" id="overview" role="tabpanel">
<!-- Profile Statistics and Revenue Chart Row -->
<div class="row mb-4">
<!-- Profile Statistics -->
<div class="col-lg-6">
<div class="card shadow-sm border-0 h-100">
<div class="card-body p-4">
<div class="d-flex align-items-center mb-2">
<i class="bi bi-bar-chart-line text-primary me-2"></i>
<h5 class="mb-0 fw-bold">Profile Statistics</h5>
</div>
<p class="text-muted small mb-4">Key performance metrics and milestones</p>
<div class="row g-3">
<!-- Total Sessions -->
<div class="col-md-6">
<div class="d-flex align-items-center gap-3 p-3 bg-light rounded">
<div class="rounded-circle d-flex align-items-center justify-content-center" style="width: 48px; height: 48px; background-color: #6f42c1;">
<i class="bi bi-people-fill text-white"></i>
</div>
<div class="flex-grow-1">
<div class="small text-muted mb-1">Total Sessions</div>
<div class="h4 fw-bold mb-2">127</div>
<div class="progress" style="height: 4px; background-color: #e9ecef;">
<div class="progress-bar" role="progressbar" style="width: 85%; background: linear-gradient(90deg, #6f42c1 0%, #8b5cf6 100%);"></div>
</div>
<small class="text-muted" style="font-size: 0.75rem;">Sessions completed this year</small>
</div>
</div>
</div>
<!-- Attendance Rate -->
<div class="col-md-6">
<div class="d-flex align-items-center gap-3 p-3 bg-light rounded">
<div class="rounded-circle d-flex align-items-center justify-content-center" style="width: 48px; height: 48px; background-color: #3b82f6;">
<i class="bi bi-graph-up-arrow text-white"></i>
</div>
<div class="flex-grow-1">
<div class="small text-muted mb-1">Attendance Rate</div>
<div class="h4 fw-bold mb-2">85%</div>
<div class="progress" style="height: 4px; background-color: #e9ecef;">
<div class="progress-bar" role="progressbar" style="width: 85%; background: linear-gradient(90deg, #6f42c1 0%, #10b981 100%);"></div>
</div>
<small class="text-muted" style="font-size: 0.75rem;">Average session attendance</small>
</div>
</div>
</div>
<!-- Achievements -->
<div class="col-md-6">
<div class="d-flex align-items-center gap-3 p-3 bg-light rounded">
<div class="rounded-circle d-flex align-items-center justify-content-center" style="width: 48px; height: 48px; background-color: #8b5cf6;">
<i class="bi bi-trophy-fill text-white"></i>
</div>
<div class="flex-grow-1">
<div class="small text-muted mb-1">Achievements</div>
<div class="h4 fw-bold mb-2">8</div>
<div class="progress" style="height: 4px; background-color: #e9ecef;">
<div class="progress-bar" role="progressbar" style="width: 40%; background: linear-gradient(90deg, #6f42c1 0%, #10b981 100%);"></div>
</div>
<small class="text-muted" style="font-size: 0.75rem;">Total badges earned</small>
</div>
</div>
</div>
<!-- Goal Completion -->
<div class="col-md-6">
<div class="d-flex align-items-center gap-3 p-3 bg-light rounded">
<div class="rounded-circle d-flex align-items-center justify-content-center" style="width: 48px; height: 48px; background-color: #10b981;">
<i class="bi bi-check-circle-fill text-white"></i>
</div>
<div class="flex-grow-1">
<div class="small text-muted mb-1">Goal Completion</div>
<div class="h4 fw-bold mb-2">75%</div>
<div class="progress" style="height: 4px; background-color: #e9ecef;">
<div class="progress-bar" role="progressbar" style="width: 75%; background: linear-gradient(90deg, #6f42c1 0%, #10b981 100%);"></div>
</div>
<small class="text-muted" style="font-size: 0.75rem;">Current goals achieved</small>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Self Investment Chart -->
<div class="col-lg-6">
<div class="card shadow-sm border-0 h-100">
<div class="card-body p-4">
<div class="d-flex align-items-center mb-2">
<i class="bi bi-bar-chart-line text-primary me-2"></i>
<h5 class="mb-0 fw-bold">Self Investment Chart</h5>
</div>
<p class="text-muted small mb-4">Self investment analytics over time</p>
<div class="d-flex align-items-center justify-content-center" style="min-height: 300px;">
<div class="text-center">
<i class="bi bi-graph-up text-muted" style="font-size: 3rem;"></i>
<p class="text-muted mt-3 mb-1">Revenue chart visualization coming soon...</p>
<small class="text-muted">Chart will display revenue trends over time</small>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Complete Payment & Revenue History -->
<div class="card shadow-sm border-0 mb-4">
<div class="card-body p-4">
<div class="d-flex align-items-center mb-2">
<i class="bi bi-receipt text-primary me-2"></i>
<h5 class="mb-0 fw-bold">Complete Payment & Revenue History</h5>
</div>
<p class="text-muted small mb-4">All package payments and revenue transactions in one view</p>
<div class="table-responsive">
<table class="table table-hover align-middle">
<thead class="table-light">
<tr>
<th class="text-muted small fw-semibold">Date</th>
<th class="text-muted small fw-semibold">Transaction Type</th>
<th class="text-muted small fw-semibold">Package/Item</th>
<th class="text-muted small fw-semibold">Duration</th>
<th class="text-muted small fw-semibold">Sessions</th>
<th class="text-muted small fw-semibold">Amount</th>
<th class="text-muted small fw-semibold">Status</th>
<th class="text-muted small fw-semibold">Method</th>
<th class="text-muted small fw-semibold">Evidence</th>
</tr>
</thead>
<tbody>
@forelse($invoices as $invoice)
<tr>
<td class="small">{{ $invoice->created_at->format('Y-m-d') }}</td>
<td class="small text-primary">Invoice</td>
<td class="small">{{ $invoice->tenant->club_name ?? 'N/A' }}</td>
<td class="small text-muted">-</td>
<td class="small">-</td>
<td class="small fw-semibold" style="color: {{ $invoice->status == 'paid' ? '#10b981' : '#f59e0b' }};">{{ $invoice->amount }} BHD</td>
<td>
@if($invoice->status == 'paid')
<span class="badge bg-success-subtle text-success small"> Paid</span>
@elseif($invoice->status == 'due')
<span class="badge bg-warning-subtle text-warning small"> Due</span>
@else
<span class="badge bg-secondary-subtle text-secondary small">{{ ucfirst($invoice->status) }}</span>
@endif
</td>
<td class="small">-</td>
<td class="small">
<a href="{{ route('bills.receipt', $invoice->id) }}" target="_blank" title="View Receipt"><i class="bi bi-file-earmark-text text-primary"></i></a>
<a href="{{ route('bills.receipt', $invoice->id) }}?download=1" download title="Download Receipt"><i class="bi bi-download text-secondary ms-1"></i></a>
</td>
</tr>
@empty
<tr>
<td colspan="9" class="text-center text-muted small">No invoices found</td>
</tr>
@endforelse
</tbody>
</table>
</div>
</div>
</div>
</div>
<!-- Attendance Tab -->
<div class="tab-pane fade" id="attendance" role="tabpanel">
<div class="card shadow-sm border-0">
<div class="card-body p-4">
<div class="d-flex align-items-center justify-content-between mb-4">
<div>
<h1 class="h3 fw-bold">Member Attendance</h1>
<p class="text-muted">Track your gym session attendance and performance</p>
</div>
</div>
<!-- Summary Cards -->
<div class="row g-4 mb-4">
<div class="col-md-4">
<div class="card shadow-sm bg-light">
<div class="card-body text-center">
<div class="display-4 fw-bold text-success mb-2">{{ $sessionsCompleted }}</div>
<h6 class="card-title text-muted">Sessions Completed</h6>
</div>
</div>
</div>
<div class="col-md-4">
<div class="card shadow-sm bg-light">
<div class="card-body text-center">
<div class="display-4 fw-bold text-danger mb-2">{{ $noShows }}</div>
<h6 class="card-title text-muted">No Shows</h6>
</div>
</div>
</div>
<div class="col-md-4">
<div class="card shadow-sm bg-light">
<div class="card-body text-center">
<div class="display-4 fw-bold text-primary mb-2">{{ $attendanceRate }}%</div>
<h6 class="card-title text-muted">Attendance Rate</h6>
</div>
</div>
</div>
</div>
<!-- Attendance Table -->
<div class="card shadow-sm">
<div class="card-header bg-light">
<h6 class="card-title mb-0">Session History</h6>
</div>
<div class="card-body p-0">
<div class="table-responsive">
<table class="table table-hover mb-0">
<thead class="table-light">
<tr>
<th class="border-0 fw-semibold">Date & Time</th>
<th class="border-0 fw-semibold">Session Type</th>
<th class="border-0 fw-semibold">Trainer Name</th>
<th class="border-0 fw-semibold">Status</th>
<th class="border-0 fw-semibold">Notes</th>
</tr>
</thead>
<tbody>
@forelse($attendanceRecords as $record)
<tr>
<td class="align-middle">
<div class="fw-semibold">{{ $record->session_datetime->format('M j, Y') }}</div>
<small class="text-muted">{{ $record->session_datetime->format('g:i A') }}</small>
</td>
<td class="align-middle">{{ $record->session_type }}</td>
<td class="align-middle">{{ $record->trainer_name }}</td>
<td class="align-middle">
@if($record->status === 'completed')
<span class="badge bg-success">Completed</span>
@else
<span class="badge bg-danger">No Show</span>
@endif
</td>
<td class="align-middle">
<small class="text-muted">{{ $record->notes ?: '-' }}</small>
</td>
</tr>
@empty
<tr>
<td colspan="5" class="text-center py-4">
<i class="bi bi-calendar-check text-muted" style="font-size: 2rem;"></i>
<p class="text-muted mt-2 mb-0">No attendance records found</p>
</td>
</tr>
@endforelse
</tbody>
</table>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Health Tab -->
<div class="tab-pane fade" id="health" role="tabpanel">
<!-- Health Tracking Header -->
<div class="card shadow-sm border-0 mb-4">
<div class="card-body p-4">
<div class="d-flex align-items-center mb-2">
<i class="bi bi-heart-pulse text-danger me-2"></i>
<h5 class="mb-0 fw-bold">Health Metrics Overview</h5>
</div>
<div class="d-flex justify-content-between align-items-center mb-4">
<p class="text-muted small mb-0">Monitor health metrics and progress over time</p>
@if($latestHealthRecord)
@php
$latestDate = $latestHealthRecord->recorded_at;
$now = \Carbon\Carbon::now();
$diff = $latestDate->diff($now);
@endphp
<div class="d-flex align-items-center gap-3">
<div class="d-flex align-items-center gap-2">
<i class="bi bi-calendar-event text-primary"></i>
<span class="fw-semibold">Snapshot Date:</span>
<span class="text-muted">{{ $latestDate->format('F j, Y') }}</span>
</div>
<div class="vr"></div>
<div class="d-flex align-items-center gap-2">
<i class="bi bi-clock-history text-primary"></i>
<span class="fw-semibold">Time Since:</span>
<span class="text-muted">
@if($diff->y > 0)
{{ $diff->y }} {{ $diff->y == 1 ? 'year' : 'years' }}
@endif
@if($diff->m > 0)
{{ $diff->m }} {{ $diff->m == 1 ? 'month' : 'months' }}
@endif
@if($diff->d > 0)
{{ $diff->d }} {{ $diff->d == 1 ? 'day' : 'days' }}
@endif
ago
</span>
</div>
</div>
@else
<div class="text-muted small">No health records available</div>
@endif
</div>
<!-- Health Metrics Cards -->
<div class="row g-3">
@if($latestHealthRecord)
<!-- Weight -->
<div class="col-md-2">
<div class="text-center p-3 bg-light rounded">
<i class="bi bi-speedometer2 text-purple mb-2" style="font-size: 1.5rem; color: #8b5cf6;"></i>
<div class="h4 fw-bold mb-0">{{ $latestHealthRecord->weight ?? 'N/A' }}</div>
<small class="text-muted">Weight (kg)</small>
</div>
</div>
<!-- Body Fat -->
<div class="col-md-2">
<div class="text-center p-3 bg-light rounded">
<i class="bi bi-activity text-warning mb-2" style="font-size: 1.5rem;"></i>
<div class="h4 fw-bold mb-0">{{ $latestHealthRecord->body_fat_percentage ?? 'N/A' }}%</div>
<small class="text-muted">Body Fat</small>
</div>
</div>
<!-- Body Water -->
<div class="col-md-2">
<div class="text-center p-3 bg-light rounded">
<i class="bi bi-droplet text-info mb-2" style="font-size: 1.5rem;"></i>
<div class="h4 fw-bold mb-0">{{ $latestHealthRecord->body_water_percentage ?? 'N/A' }}%</div>
<small class="text-muted">Body Water</small>
</div>
</div>
<!-- Muscle Mass -->
<div class="col-md-2">
<div class="text-center p-3 bg-light rounded">
<i class="bi bi-heart text-success mb-2" style="font-size: 1.5rem;"></i>
<div class="h4 fw-bold mb-0">{{ $latestHealthRecord->muscle_mass ?? 'N/A' }}</div>
<small class="text-muted">Muscle Mass</small>
</div>
</div>
<!-- Bone Mass -->
<div class="col-md-2">
<div class="text-center p-3 bg-light rounded">
<i class="bi bi-capsule text-secondary mb-2" style="font-size: 1.5rem;"></i>
<div class="h4 fw-bold mb-0">{{ $latestHealthRecord->bone_mass ?? 'N/A' }}</div>
<small class="text-muted">Bone Mass</small>
</div>
</div>
<!-- BMR -->
<div class="col-md-2">
<div class="text-center p-3 bg-light rounded">
<i class="bi bi-lightning text-danger mb-2" style="font-size: 1.5rem;"></i>
<div class="h4 fw-bold mb-0">{{ $latestHealthRecord->bmr ?? 'N/A' }}</div>
<small class="text-muted">BMR (cal)</small>
</div>
</div>
@else
<div class="col-12">
<div class="text-center py-4">
<i class="bi bi-heart-pulse text-muted" style="font-size: 3rem;"></i>
<p class="text-muted mt-3">No health metrics available</p>
</div>
</div>
@endif
</div>
</div>
</div>
<!-- Body Composition Analysis & Compare Row -->
<div class="row mb-4">
<!-- Body Composition Analysis -->
<div class="col-lg-7">
<div class="card shadow-sm border-0 h-100">
<div class="card-body p-4">
<h5 class="fw-bold mb-4"><i class="bi bi-activity me-2"></i>Body Composition Analysis</h5>
<div class="chart-container" style="position: relative; height: 500px; width: 100%;">
<canvas id="radarChart" data-current='@json($comparisonRecords->first())' data-previous='@json($comparisonRecords->skip(1)->first())'></canvas>
</div>
</div>
</div>
</div>
<!-- Compare -->
<div class="col-lg-5">
<div class="card shadow-sm border-0 h-100">
<div class="card-body p-4">
<h5 class="fw-bold mb-4"><i class="bi bi-bar-chart-line me-2"></i>Compare</h5>
@if($comparisonRecords->count() >= 2)
@php
$current = $comparisonRecords->first();
$previous = $comparisonRecords->skip(1)->first();
@endphp
<div class="mb-3">
<div class="row g-2">
<div class="col-6 text-center">
<label class="form-label fw-bold">From</label>
<select class="form-select form-select-sm" id="currentDate">
@foreach($healthRecords as $record)
<option value="{{ $record->id }}" {{ $record->id == $current->id ? 'selected' : '' }}>
{{ $record->recorded_at->format('M j, Y') }}
</option>
@endforeach
</select>
</div>
<div class="col-6 text-center">
<label class="form-label fw-bold">To</label>
<select class="form-select form-select-sm" id="previousDate">
@foreach($healthRecords as $record)
<option value="{{ $record->id }}" {{ $record->id == $previous->id ? 'selected' : '' }}>
{{ $record->recorded_at->format('M j, Y') }}
</option>
@endforeach
</select>
</div>
</div>
<div class="mt-3">
<div class="alert alert-secondary text-center py-2" id="timeDifference">
@if($current && $previous)
<strong>Time between records:</strong> {{ calculateTimeDifference($current->recorded_at, $previous->recorded_at) }}
@else
Select dates to see time difference
@endif
</div>
</div>
</div>
<div class="table-responsive">
<table class="table table-sm align-middle">
<thead>
<tr class="border-bottom">
<th class="text-muted small fw-semibold">Metric</th>
<th class="text-muted small fw-semibold text-end">Current</th>
<th class="text-muted small fw-semibold text-end">Previous</th>
<th class="text-muted small fw-semibold text-center">Change</th>
</tr>
</thead>
<tbody>
@php
function getChangeIcon($current, $previous) {
if ($current > $previous) return '<i class="bi bi-arrow-up text-success"></i>';
if ($current < $previous) return '<i class="bi bi-arrow-down text-danger"></i>';
return '<i class="bi bi-dash text-muted"></i>';
}
@endphp
<tr data-metric="height">
<td class="small"><i class="bi bi-rulers me-2"></i>Height</td>
<td class="small text-end fw-semibold text-primary">{{ $current->height ?? 'N/A' }}cm</td>
<td class="small text-end text-danger">{{ $previous->height ?? 'N/A' }}cm</td>
<td class="text-center">{!! $current->height && $previous->height ? getChangeIcon($current->height, $previous->height) : '-' !!}</td>
</tr>
<tr data-metric="weight">
<td class="small"><i class="bi bi-speedometer2 me-2"></i>Weight</td>
<td class="small text-end fw-semibold text-primary">{{ $current->weight ?? 'N/A' }}kg</td>
<td class="small text-end text-danger">{{ $previous->weight ?? 'N/A' }}kg</td>
<td class="text-center">{!! $current->weight && $previous->weight ? getChangeIcon($current->weight, $previous->weight) : '-' !!}</td>
</tr>
<tr data-metric="body_fat">
<td class="small"><i class="bi bi-activity me-2"></i>Body Fat</td>
<td class="small text-end fw-semibold text-primary">{{ $current->body_fat_percentage ?? 'N/A' }}%</td>
<td class="small text-end text-danger">{{ $previous->body_fat_percentage ?? 'N/A' }}%</td>
<td class="text-center">{!! $current->body_fat_percentage && $previous->body_fat_percentage ? getChangeIcon($current->body_fat_percentage, $previous->body_fat_percentage) : '-' !!}</td>
</tr>
<tr data-metric="bmi">
<td class="small"><i class="bi bi-calculator me-2"></i>BMI</td>
<td class="small text-end fw-semibold text-primary">{{ $current->bmi ?? 'N/A' }}</td>
<td class="small text-end text-danger">{{ $previous->bmi ?? 'N/A' }}</td>
<td class="text-center">{!! $current->bmi && $previous->bmi ? getChangeIcon($current->bmi, $previous->bmi) : '-' !!}</td>
</tr>
<tr data-metric="body_water">
<td class="small"><i class="bi bi-droplet me-2"></i>Body Water</td>
<td class="small text-end fw-semibold text-primary">{{ $current->body_water_percentage ?? 'N/A' }}%</td>
<td class="small text-end text-danger">{{ $previous->body_water_percentage ?? 'N/A' }}%</td>
<td class="text-center">{!! $current->body_water_percentage && $previous->body_water_percentage ? getChangeIcon($current->body_water_percentage, $previous->body_water_percentage) : '-' !!}</td>
</tr>
<tr data-metric="muscle_mass">
<td class="small"><i class="bi bi-heart me-2"></i>Muscle Mass</td>
<td class="small text-end fw-semibold text-primary">{{ $current->muscle_mass ?? 'N/A' }}kg</td>
<td class="small text-end text-danger">{{ $previous->muscle_mass ?? 'N/A' }}kg</td>
<td class="text-center">{!! $current->muscle_mass && $previous->muscle_mass ? getChangeIcon($current->muscle_mass, $previous->muscle_mass) : '-' !!}</td>
</tr>
<tr data-metric="bone_mass">
<td class="small"><i class="bi bi-capsule me-2"></i>Bone Mass</td>
<td class="small text-end fw-semibold text-primary">{{ $current->bone_mass ?? 'N/A' }}kg</td>
<td class="small text-end text-danger">{{ $previous->bone_mass ?? 'N/A' }}kg</td>
<td class="text-center">{!! $current->bone_mass && $previous->bone_mass ? getChangeIcon($current->bone_mass, $previous->bone_mass) : '-' !!}</td>
</tr>
<tr data-metric="visceral_fat">
<td class="small"><i class="bi bi-activity me-2"></i>Visceral Fat</td>
<td class="small text-end fw-semibold text-primary">{{ $current->visceral_fat ?? 'N/A' }}</td>
<td class="small text-end text-danger">{{ $previous->visceral_fat ?? 'N/A' }}</td>
<td class="text-center">{!! $current->visceral_fat && $previous->visceral_fat ? getChangeIcon($current->visceral_fat, $previous->visceral_fat) : '-' !!}</td>
</tr>
<tr data-metric="bmr">
<td class="small"><i class="bi bi-lightning me-2"></i>BMR</td>
<td class="small text-end fw-semibold text-primary">{{ $current->bmr ?? 'N/A' }}cal</td>
<td class="small text-end text-danger">{{ $previous->bmr ?? 'N/A' }}cal</td>
<td class="text-center">{!! $current->bmr && $previous->bmr ? getChangeIcon($current->bmr, $previous->bmr) : '-' !!}</td>
</tr>
<tr data-metric="protein">
<td class="small"><i class="bi bi-heart-pulse me-2"></i>Protein</td>
<td class="small text-end fw-semibold text-primary">{{ $current->protein_percentage ?? 'N/A' }}%</td>
<td class="small text-end text-danger">{{ $previous->protein_percentage ?? 'N/A' }}%</td>
<td class="text-center">{!! $current->protein_percentage && $previous->protein_percentage ? getChangeIcon($current->protein_percentage, $previous->protein_percentage) : '-' !!}</td>
</tr>
<tr data-metric="body_age">
<td class="small"><i class="bi bi-calendar-heart me-2"></i>Body Age</td>
<td class="small text-end fw-semibold text-primary">{{ $current->body_age ?? 'N/A' }}yrs</td>
<td class="small text-end text-danger">{{ $previous->body_age ?? 'N/A' }}yrs</td>
<td class="text-center">{!! $current->body_age && $previous->body_age ? getChangeIcon($current->body_age, $previous->body_age) : '-' !!}</td>
</tr>
</tbody>
</table>
</div>
@else
<div class="text-center py-4">
<i class="bi bi-bar-chart-line text-muted" style="font-size: 3rem;"></i>
<p class="text-muted mt-3">Need at least 2 health records to compare</p>
</div>
@endif
</div>
</div>
</div>
</div>
<!-- Health Tracking History -->
<div class="card shadow-sm border-0">
<div class="card-body p-4">
<h5 class="fw-bold mb-4"><i class="bi bi-heart-pulse me-2"></i>Health Tracking</h5>
@if($healthRecords->count() > 0)
<div class="table-responsive">
<table class="table table-hover align-middle">
<thead class="table-light">
<tr>
<th class="text-muted small fw-semibold">Date</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-rulers me-1"></i>Height (cm)</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-speedometer2 me-1"></i>Weight (kg)</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-activity me-1"></i>Body Fat %</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-calculator me-1"></i>BMI</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-droplet me-1"></i>Body Water %</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-heart me-1"></i>Muscle Mass (kg)</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-capsule me-1"></i>Bone Mass (kg)</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-activity me-1"></i>Visceral Fat</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-lightning me-1"></i>BMR</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-heart-pulse me-1"></i>Protein %</th>
<th class="text-muted small fw-semibold text-center"><i class="bi bi-calendar-heart me-1"></i>Body Age</th>
</tr>
</thead>
<tbody>
@foreach($healthRecords as $record)
<tr data-record-id="{{ $record->id }}" class="position-relative history-row">
<td class="small fw-semibold">{{ $record->recorded_at->format('M j, Y') }}</td>
<td class="small text-center">{{ $record->height ?? '-' }}</td>
<td class="small text-center">{{ $record->weight ?? '-' }}</td>
<td class="small text-center">{{ $record->body_fat_percentage ?? '-' }}</td>
<td class="small text-center">{{ $record->bmi ?? '-' }}</td>
<td class="small text-center">{{ $record->body_water_percentage ?? '-' }}</td>
<td class="small text-center">{{ $record->muscle_mass ?? '-' }}</td>
<td class="small text-center">{{ $record->bone_mass ?? '-' }}</td>
<td class="small text-center">{{ $record->visceral_fat ?? '-' }}</td>
<td class="small text-center">{{ $record->bmr ?? '-' }}</td>
<td class="small text-center">{{ $record->protein_percentage ?? '-' }}</td>
<td class="small text-center">{{ $record->body_age ?? '-' }}</td>
<td class="position-absolute top-50 end-0 translate-middle-y opacity-0 edit-record-btn" style="cursor: pointer; right: 10px;">
<i class="bi bi-pencil text-primary" style="font-size: 1.2rem;"></i>
</td>
</tr>
@endforeach
</tbody>
</table>
</div>
<!-- Pagination -->
<div class="d-flex justify-content-center mt-4">
{{ $healthRecords->links() }}
</div>
@else
<div class="text-center py-5">
<i class="bi bi-clipboard-data text-muted" style="font-size: 3rem;"></i>
<p class="text-muted mt-3">No health records found</p>
<small class="text-muted">Health tracking data will appear here once records are added</small>
</div>
@endif
</div>
</div>
</div>
<!-- Goals Tab -->
<div class="tab-pane fade" id="goals" role="tabpanel">
<div class="card shadow-sm border-0">
<div class="card-body p-4">
<!-- Section Title & Subtitle -->
<div class="d-flex justify-content-between align-items-center mb-4">
<div>
<h5 class="fw-bold mb-1"><i class="bi bi-bullseye me-2"></i>Goal Tracking</h5>
<p class="text-muted small mb-0">Set, track, and achieve your fitness objectives.</p>
</div>
</div>
<!-- Summary Cards -->
<div class="row g-3 mb-4">
<!-- Active Goals -->
<div class="col-md-4">
<div class="card shadow-sm border-0 text-center h-100 goal-filter-card" style="background: linear-gradient(135deg, #8b5cf6 0%, #a855f7 100%); min-height: 120px; cursor: pointer;" data-filter="active">
<div class="card-body p-3 d-flex flex-column justify-content-center align-items-center h-100">
<i class="bi bi-bullseye text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1">{{ $activeGoalsCount }}</h4>
<small class="text-white-50">Active Goals</small>
</div>
</div>
</div>
<!-- Completed Goals -->
<div class="col-md-4">
<div class="card shadow-sm border-0 text-center h-100 goal-filter-card" style="background: linear-gradient(135deg, #10b981 0%, #34d399 100%); min-height: 120px; cursor: pointer;" data-filter="completed">
<div class="card-body p-3 d-flex flex-column justify-content-center align-items-center h-100">
<i class="bi bi-check-circle-fill text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1">{{ $completedGoalsCount }}</h4>
<small class="text-white-50">Completed Goals</small>
</div>
</div>
</div>
<!-- Success Rate -->
<div class="col-md-4">
<div class="card shadow-sm border-0 text-center h-100" style="background: linear-gradient(135deg, #f59e0b 0%, #fbbf24 100%); min-height: 120px;">
<div class="card-body p-3 d-flex flex-column justify-content-center align-items-center h-100">
<i class="bi bi-graph-up-arrow text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1">{{ $successRate }}%</h4>
<small class="text-white-50">Success Rate</small>
</div>
</div>
</div>
</div>
<!-- Current Goals List -->
@if($goals->count() > 0)
<div class="row g-4">
@foreach($goals as $goal)
<div class="col-lg-6">
<div class="card shadow-sm border-0 h-100 position-relative">
<!-- Edit Button (only for active goals and authorized users) -->
@if($goal->status == 'active' && ($relationship->relationship_type == 'self' || Auth::id() == $relationship->guardian_user_id))
<button class="btn btn-sm btn-outline-primary rounded-circle position-absolute top-0 end-0 mt-2 me-2 edit-goal-btn" style="width: 32px; height: 32px; padding: 0;" data-goal-id="{{ $goal->id }}" title="Edit Goal">
<i class="bi bi-pencil"></i>
</button>
@endif
<div class="card-body p-4">
<!-- Title & Icon -->
<div class="d-flex align-items-center mb-3">
<div class="rounded-circle d-flex align-items-center justify-content-center me-3" style="width: 48px; height: 48px; background-color: #8b5cf6;">
@if($goal->icon_type == 'dumbbell')
<i class="bi bi-dumbbell text-white"></i>
@elseif($goal->icon_type == 'clock')
<i class="bi bi-clock text-white"></i>
@else
<i class="bi bi-bullseye text-white"></i>
@endif
</div>
<div class="flex-grow-1">
<h6 class="fw-bold mb-1">{{ $goal->title }}</h6>
@if($goal->description)
<p class="text-muted small mb-0">{{ $goal->description }}</p>
@endif
</div>
</div>
<!-- Progress Indicator -->
<div class="mb-3">
<div class="d-flex justify-content-between align-items-center mb-2">
<small class="text-muted">Progress: {{ number_format($goal->current_progress_value, 1) }} / {{ number_format($goal->target_value, 1) }} {{ $goal->unit }}</small>
<small class="fw-semibold">{{ number_format($goal->progress_percentage, 1) }}%</small>
</div>
<div class="progress" style="height: 8px;">
<div class="progress-bar" role="progressbar" style="width: {{ $goal->progress_percentage }}%; background: linear-gradient(90deg, #8b5cf6 0%, #10b981 100%);" aria-valuenow="{{ $goal->progress_percentage }}" aria-valuemin="0" aria-valuemax="100"></div>
</div>
</div>
<!-- Dates & Status -->
<div class="row g-2 mb-3">
<div class="col-6">
<small class="text-muted d-block">Started:</small>
<small class="fw-semibold">{{ $goal->start_date->format('M d, Y') }}</small>
</div>
<div class="col-6 text-end">
<small class="text-muted d-block">Target:</small>
<small class="fw-semibold">{{ $goal->target_date->format('M d, Y') }}</small>
</div>
</div>
<!-- Status Badges -->
<div class="d-flex gap-2 flex-wrap">
<span class="badge {{ $goal->status == 'active' ? 'bg-primary' : 'bg-success' }} small">
{{ ucfirst($goal->status) }}
</span>
<span class="badge {{ $goal->priority_level == 'high' ? 'bg-danger' : ($goal->priority_level == 'medium' ? 'bg-warning text-dark' : 'bg-secondary') }} small">
{{ ucfirst($goal->priority_level) }}
</span>
</div>
</div>
</div>
</div>
@endforeach
</div>
@else
<div class="text-center py-4">
<i class="bi bi-bullseye text-muted" style="font-size: 3rem;"></i>
<h5 class="text-muted mt-3 mb-2">No Goals Set Yet</h5>
<p class="text-muted mb-0">Start your fitness journey by setting your first goal!</p>
</div>
@endif
</div>
</div>
</div>
<!-- Affiliations Tab -->
<div class="tab-pane fade" id="affiliations" role="tabpanel">
@include('family.partials.affiliations-enhanced')
</div>
<!-- Tournaments Tab -->
<div class="tab-pane fade" id="tournaments" role="tabpanel">
<!-- Tournament & Event Participation Card -->
<div class="card shadow-sm border-0 mb-4">
<div class="card-body p-4">
<!-- Section Title & Subtitle -->
<div class="d-flex justify-content-between align-items-center mb-4">
<div>
<h5 class="fw-bold mb-1"><i class="bi bi-trophy-fill text-warning me-2"></i>Tournament & Event Participation</h5>
<p class="text-muted small mb-0">Proven champion with multiple championship wins and prestigious awards.</p>
</div>
<!-- Filter Section -->
<div class="d-flex align-items-center">
<label for="sportFilter" class="form-label me-2 mb-0 fw-semibold">Filter by Sport:</label>
<select class="form-select form-select-sm" id="sportFilter" style="width: 150px;">
<option value="all">All Sports</option>
@foreach($sports as $sport)
<option value="{{ $sport }}">{{ $sport }}</option>
@endforeach
</select>
</div>
</div>
<!-- Award Summary Cards -->
<div class="row g-3" id="awardCards">
<div class="col-md-3">
<div class="card shadow-sm border-0 text-center" style="background: linear-gradient(135deg, #FFD700 0%, #FFA500 100%);">
<div class="card-body p-3">
<i class="bi bi-trophy-fill text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1" id="specialCount">{{ $awardCounts['special'] }}</h4>
<small class="text-white-50">Special Award</small>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card shadow-sm border-0 text-center" style="background: linear-gradient(135deg, #FFD700 0%, #FFA500 100%);">
<div class="card-body p-3">
<i class="bi bi-award-fill text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1" id="firstCount">{{ $awardCounts['1st'] }}</h4>
<small class="text-white-50">1st Place</small>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card shadow-sm border-0 text-center" style="background: linear-gradient(135deg, #C0C0C0 0%, #A8A8A8 100%);">
<div class="card-body p-3">
<i class="bi bi-award-fill text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1" id="secondCount">{{ $awardCounts['2nd'] }}</h4>
<small class="text-white-50">2nd Place</small>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card shadow-sm border-0 text-center" style="background: linear-gradient(135deg, #CD7F32 0%, #A0522D 100%);">
<div class="card-body p-3">
<i class="bi bi-award-fill text-white mb-2" style="font-size: 2rem;"></i>
<h4 class="text-white fw-bold mb-1" id="thirdCount">{{ $awardCounts['3rd'] }}</h4>
<small class="text-white-50">3rd Place</small>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Tournament & Championships History Card -->
<div class="card shadow-sm border-0">
<div class="card-body p-4">
<h6 class="fw-bold mb-3"><i class="bi bi-list-ul me-2"></i>Tournament & Championships History</h6>
@if($tournamentEvents->count() > 0)
<div class="table-responsive">
<table class="table table-hover align-middle" id="tournamentsTable">
<thead class="table-light">
<tr>
<th class="text-muted small fw-semibold">Tournament Details</th>
<th class="text-muted small fw-semibold">Club Affiliation</th>
<th class="text-muted small fw-semibold">Performance & Result</th>
<th class="text-muted small fw-semibold">Notes & Media</th>
</tr>
</thead>
<tbody>
@foreach($tournamentEvents as $event)
<tr data-sport="{{ $event->sport }}">
<td>
<div class="fw-bold">{{ $event->title }}</div>
<div class="d-flex gap-2 mt-1 flex-wrap">
<span class="badge bg-{{ $event->type == 'championship' ? 'primary' : 'secondary' }} small">{{ ucfirst($event->type) }}</span>
<span class="badge bg-info small">{{ $event->sport }}</span>
</div>
<div class="text-muted small mt-1">
<i class="bi bi-calendar-event me-1"></i>{{ $event->date->format('M j, Y') }}
@if($event->time)
<i class="bi bi-clock me-1 ms-2"></i>{{ $event->time->format('H:i') }}
@endif
@if($event->location)
<i class="bi bi-geo-alt me-1 ms-2"></i>{{ $event->location }}
@endif
@if($event->participants_count)
<i class="bi bi-people me-1 ms-2"></i>{{ $event->participants_count }} participants
@endif
</div>
</td>
<td>
@if($event->clubAffiliation)
<div>
<div class="small fw-semibold">{{ $event->clubAffiliation->club_name }}</div>
<div class="text-muted small">{{ $event->clubAffiliation->location }}</div>
</div>
@else
<span class="text-muted small">Individual</span>
@endif
</td>
<td>
@if($event->performanceResults->count() > 0)
@foreach($event->performanceResults as $result)
<div class="d-flex align-items-center gap-2 mb-1">
@if($result->medal_type == '1st')
<i class="bi bi-award-fill text-warning"></i>
<span class="badge bg-warning text-dark small">1st Place</span>
@elseif($result->medal_type == '2nd')
<i class="bi bi-award-fill text-secondary"></i>
<span class="badge bg-secondary small">2nd Place</span>
@elseif($result->medal_type == '3rd')
<i class="bi bi-award-fill" style="color: #CD7F32;"></i>
<span class="badge" style="background-color: #CD7F32; color: white;" small>3rd Place</span>
@elseif($result->medal_type == 'special')
<i class="bi bi-trophy-fill text-warning"></i>
<span class="badge bg-warning text-dark small">Special Award</span>
@endif
@if($result->points)
<small class="text-muted">{{ $result->points }} pts</small>
@endif
</div>
@if($result->description)
<small class="text-muted">{{ $result->description }}</small>
@endif
@endforeach
@else
<span class="text-muted small">No results recorded</span>
@endif
</td>
<td>
@if($event->notesMedia->count() > 0)
@foreach($event->notesMedia as $note)
@if($note->note_text)
<p class="mb-1 small">{{ $note->note_text }}</p>
@endif
@if($note->media_link)
<a href="{{ $note->media_link }}" target="_blank" class="btn btn-sm btn-outline-primary small">
<i class="bi bi-image me-1"></i>View Media
</a>
@endif
@endforeach
@else
<span class="text-muted small">No notes available</span>
@endif
</td>
</tr>
@endforeach
</tbody>
</table>
</div>
@else
<div class="text-center py-5">
<i class="bi bi-trophy text-muted" style="font-size: 3rem;"></i>
<p class="text-muted mt-3">No tournament records found</p>
<small class="text-muted">Tournament participation will appear here once records are added</small>
</div>
@endif
</div>
</div>
</div>
<!-- Events Tab -->
<div class="tab-pane fade" id="events" role="tabpanel">
<div class="card shadow-sm border-0">
<div class="card-body p-4">
<h5 class="fw-bold mb-3"><i class="bi bi-calendar-event me-2"></i>Event Participation</h5>
<p class="text-muted">Event history coming soon...</p>
</div>
</div>
</div>
</div>
</div>
<!-- Goal Edit Modal -->
<div class="modal fade" id="goalEditModal" tabindex="-1" aria-labelledby="goalEditModalLabel" aria-hidden="true">
<div class="modal-dialog modal-lg">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="goalEditModalLabel">Edit Goal Progress</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<form id="goalEditForm" method="POST">
@csrf
@method('PUT')
<div class="modal-body">
<div class="row g-3">
<div class="col-12">
<div class="d-flex align-items-center mb-3">
<div class="rounded-circle d-flex align-items-center justify-content-center me-3" id="goalIconDisplay" style="width: 48px; height: 48px; background-color: #8b5cf6;">
<i class="bi bi-bullseye text-white"></i>
</div>
<div>
<h6 class="fw-bold mb-1" id="goalTitleDisplay">Goal Title</h6>
<p class="text-muted small mb-0" id="goalDescriptionDisplay">Goal description</p>
</div>
</div>
</div>
<div class="col-md-6">
<label for="current_progress_value" class="form-label">Current Progress <span class="text-danger">*</span></label>
<div class="input-group">
<input type="number" step="0.1" class="form-control" id="current_progress_value" name="current_progress_value" required>
<span class="input-group-text" id="goalUnitDisplay">lbs</span>
</div>
<div class="form-text">Target: <span id="goalTargetDisplay">170.0 lbs</span></div>
</div>
<div class="col-md-6">
<label for="goal_status" class="form-label">Status</label>
<select class="form-select" id="goal_status" name="status">
<option value="active">Active</option>
<option value="completed">Completed</option>
</select>
</div>
<div class="col-12">
<div class="progress" style="height: 8px;">
<div class="progress-bar" role="progressbar" id="progressPreview" style="width: 0%; background: linear-gradient(90deg, #8b5cf6 0%, #10b981 100%);"></div>
</div>
<small class="text-muted mt-1 d-block" id="progressTextPreview">Progress: 0.0 / 170.0 lbs (0.0%)</small>
</div>
</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
<button type="submit" class="btn btn-primary">Update Goal</button>
</div>
</form>
</div>
</div>
</div>
<!-- Health Update Modal -->
<div class="modal fade" id="healthUpdateModal" tabindex="-1" aria-labelledby="healthUpdateModalLabel" aria-hidden="true">
<div class="modal-dialog modal-lg">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="healthUpdateModalLabel">Add Health Update</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<form id="healthUpdateForm" method="POST" action="{{ route('family.store-health', $relationship->dependent->id) }}">
@csrf
<div class="modal-body">
<div class="row g-3">
<div class="col-md-6">
<label for="recorded_at" class="form-label">Date <span class="text-danger">*</span></label>
<input type="date" class="form-control" id="recorded_at" name="recorded_at" value="{{ \Carbon\Carbon::now()->format('Y-m-d') }}" required>
</div>
<div class="col-md-6">
<label for="height" class="form-label">Height (cm)</label>
<input type="number" step="0.1" class="form-control" id="height" name="height">
</div>
<div class="col-md-6">
<label for="weight" class="form-label">Weight (kg)</label>
<input type="number" step="0.1" class="form-control" id="weight" name="weight">
</div>
<div class="col-md-6">
<label for="body_fat_percentage" class="form-label">Body Fat (%)</label>
<input type="number" step="0.1" class="form-control" id="body_fat_percentage" name="body_fat_percentage">
</div>
<div class="col-md-6">
<label for="bmi" class="form-label">BMI</label>
<input type="number" step="0.1" class="form-control" id="bmi" name="bmi">
</div>
<div class="col-md-6">
<label for="body_water_percentage" class="form-label">Body Water (%)</label>
<input type="number" step="0.1" class="form-control" id="body_water_percentage" name="body_water_percentage">
</div>
<div class="col-md-6">
<label for="muscle_mass" class="form-label">Muscle Mass (kg)</label>
<input type="number" step="0.1" class="form-control" id="muscle_mass" name="muscle_mass">
</div>
<div class="col-md-6">
<label for="bone_mass" class="form-label">Bone Mass (kg)</label>
<input type="number" step="0.1" class="form-control" id="bone_mass" name="bone_mass">
</div>
<div class="col-md-6">
<label for="visceral_fat" class="form-label">Visceral Fat</label>
<input type="number" class="form-control" id="visceral_fat" name="visceral_fat">
</div>
<div class="col-md-6">
<label for="bmr" class="form-label">BMR (cal)</label>
<input type="number" class="form-control" id="bmr" name="bmr">
</div>
<div class="col-md-6">
<label for="protein_percentage" class="form-label">Protein (%)</label>
<input type="number" step="0.1" class="form-control" id="protein_percentage" name="protein_percentage">
</div>
<div class="col-md-6">
<label for="body_age" class="form-label">Body Age (years)</label>
<input type="number" class="form-control" id="body_age" name="body_age">
</div>
</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
<button type="submit" class="btn btn-primary">Save Health Update</button>
</div>
</form>
</div>
</div>
</div>
<!-- Tournament Participation Modal -->
<div class="modal fade" id="tournamentParticipationModal" tabindex="-1" aria-labelledby="tournamentParticipationModalLabel" aria-hidden="true">
<div class="modal-dialog modal-xl">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="tournamentParticipationModalLabel">Add Tournament Participation</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<form id="tournamentParticipationForm" method="POST" action="{{ route('family.store-tournament', $relationship->dependent->id) }}">
@csrf
<div class="modal-body">
<div class="row g-3">
<!-- Tournament Details -->
<div class="col-md-6">
<label for="tournament_title" class="form-label">Tournament Title <span class="text-danger">*</span></label>
<input type="text" class="form-control" id="tournament_title" name="title" required>
</div>
<div class="col-md-6">
<label for="tournament_type" class="form-label">Type <span class="text-danger">*</span></label>
<select class="form-select" id="tournament_type" name="type" required>
<option value="">Select Type</option>
<option value="championship">Championship</option>
<option value="tournament">Tournament</option>
<option value="competition">Competition</option>
<option value="exhibition">Exhibition</option>
</select>
</div>
<div class="col-md-6">
<label for="tournament_sport" class="form-label">Sport <span class="text-danger">*</span></label>
<select class="form-select" id="tournament_sport" name="sport" required>
<option value="">Select Sport</option>
<option value="Boxing">Boxing</option>
<option value="Taekwondo">Taekwondo</option>
<option value="Karate">Karate</option>
<option value="Martial Arts">Martial Arts</option>
<option value="Fitness">Fitness</option>
<option value="Weightlifting">Weightlifting</option>
<option value="Other">Other</option>
</select>
</div>
<div class="col-md-6">
<x-birthdate-dropdown
name="date"
id="tournament_date"
label="Date"
:required="true"
:min-year="2000"
:max-year="date('Y')"
:error="$errors->first('date')" />
</div>
<div class="col-md-6">
<label for="tournament_time" class="form-label">Time</label>
<input type="time" class="form-control" id="tournament_time" name="time">
</div>
<div class="col-md-6">
<label for="tournament_location" class="form-label">Location</label>
<input type="text" class="form-control" id="tournament_location" name="location" placeholder="Venue name or address">
</div>
<div class="col-md-6">
<label for="participants_count" class="form-label">Number of Participants</label>
<input type="number" class="form-control" id="participants_count" name="participants_count" min="1">
</div>
<div class="col-md-6">
<label for="club_affiliation_id" class="form-label">Club Affiliation</label>
<select class="form-select" id="club_affiliation_id" name="club_affiliation_id">
<option value="">Select Club (Optional)</option>
@foreach($clubAffiliations ?? [] as $affiliation)
<option value="{{ $affiliation->id }}">{{ $affiliation->club_name }}</option>
@endforeach
</select>
</div>
<!-- Performance Results Section -->
<div class="col-12">
<hr>
<h6 class="mb-3">Performance Results</h6>
<div id="performanceResultsContainer">
<div class="performance-result-item mb-3 p-3 border rounded">
<div class="row g-2">
<div class="col-md-4">
<label class="form-label">Medal Type</label>
<select class="form-select medal-type" name="performance_results[0][medal_type]">
<option value="">Select Medal</option>
<option value="special">Special Award</option>
<option value="1st">1st Place</option>
<option value="2nd">2nd Place</option>
<option value="3rd">3rd Place</option>
</select>
</div>
<div class="col-md-3">
<label class="form-label">Points</label>
<input type="number" class="form-control" name="performance_results[0][points]" min="0" step="0.1">
</div>
<div class="col-md-4">
<label class="form-label">Description</label>
<input type="text" class="form-control" name="performance_results[0][description]" placeholder="Optional description">
</div>
<div class="col-md-1 d-flex align-items-end">
<button type="button" class="btn btn-outline-danger btn-sm remove-result" style="display: none;">
<i class="bi bi-trash"></i>
</button>
</div>
</div>
</div>
</div>
<button type="button" class="btn btn-outline-primary btn-sm" id="addPerformanceResult">
<i class="bi bi-plus me-1"></i>Add Another Result
</button>
</div>
<!-- Notes & Media Section -->
<div class="col-12">
<hr>
<h6 class="mb-3">Notes & Media</h6>
<div id="notesMediaContainer">
<div class="notes-media-item mb-3 p-3 border rounded">
<div class="row g-2">
<div class="col-md-6">
<label class="form-label">Note Text</label>
<textarea class="form-control" name="notes_media[0][note_text]" rows="2" placeholder="Optional notes about the tournament"></textarea>
</div>
<div class="col-md-5">
<label class="form-label">Media Link</label>
<input type="url" class="form-control" name="notes_media[0][media_link]" placeholder="https://example.com/photo.jpg">
</div>
<div class="col-md-1 d-flex align-items-end">
<button type="button" class="btn btn-outline-danger btn-sm remove-note" style="display: none;">
<i class="bi bi-trash"></i>
</button>
</div>
</div>
</div>
</div>
<button type="button" class="btn btn-outline-primary btn-sm" id="addNotesMedia">
<i class="bi bi-plus me-1"></i>Add Another Note/Media
</button>
</div>
</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
<button type="submit" class="btn btn-primary">Save Tournament Record</button>
</div>
</form>
</div>
</div>
</div>
<style>
.history-row:hover .edit-record-btn {
opacity: 1 !important;
}
/* Timeline Styles */
.timeline {
position: relative;
padding-left: 30px;
}
.timeline::before {
content: '';
position: absolute;
left: 15px;
top: 0;
bottom: 0;
width: 2px;
background: #e9ecef;
}
.timeline-item {
position: relative;
margin-bottom: 20px;
}
.timeline-marker {
position: absolute;
left: -22px;
top: 20px;
width: 12px;
height: 12px;
border-radius: 50%;
background: #6c757d;
border: 2px solid #fff;
z-index: 1;
}
.timeline-marker.bg-primary {
background: #0d6efd;
}
.timeline-content {
background: #fff;
border-radius: 8px;
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
transition: all 0.3s ease;
}
.timeline-content:hover {
box-shadow: 0 4px 8px rgba(0,0,0,0.15);
transform: translateY(-2px);
}
.affiliation-card {
transition: all 0.3s ease;
}
.affiliation-card:hover {
box-shadow: 0 4px 12px rgba(0,0,0,0.15) !important;
transform: translateY(-2px);
}
.affiliation-card.border-primary {
border-color: #0d6efd !important;
box-shadow: 0 0 0 0.2rem rgba(13, 110, 253, 0.25) !important;
}
</style>
<script>
document.addEventListener('DOMContentLoaded', function() {
// Load countries from JSON file
fetch('/data/countries.json')
.then(response => response.json())
.then(countries => {
// Convert all nationality displays from ISO3 to country name with flag
document.querySelectorAll('.nationality-display').forEach(element => {
const iso3Code = element.getAttribute('data-iso3');
if (!iso3Code) return;
const country = countries.find(c => c.iso3 === iso3Code);
if (country) {
// Get flag emoji from ISO2 code
const flagEmoji = country.iso2
.toUpperCase()
.split('')
.map(char => String.fromCodePoint(127397 + char.charCodeAt(0)))
.join('');
element.textContent = `${flagEmoji} ${country.name}`;
}
});
})
.catch(error => console.error('Error loading countries:', error));
// Handle Add Health Update click
document.querySelector('a[href="#"][data-bs-target="#healthUpdateModal"]').addEventListener('click', function(e) {
e.preventDefault();
resetHealthModal();
const modal = new bootstrap.Modal(document.getElementById('healthUpdateModal'));
modal.show();
});
// Handle Edit Health Record click
document.addEventListener('click', function(e) {
if (e.target.closest('.edit-record-btn')) {
e.preventDefault();
const recordId = e.target.closest('tr').getAttribute('data-record-id');
populateHealthModalForEdit(recordId);
const modal = new bootstrap.Modal(document.getElementById('healthUpdateModal'));
modal.show();
}
});
// Activate health tab if URL has #health
if (window.location.hash === '#health') {
const healthTab = document.querySelector('#health-tab');
if (healthTab) {
const tab = new bootstrap.Tab(healthTab);
tab.show();
}
}
// Store health records data for dynamic comparison
const healthRecordsData = @json($healthRecords->items());
// Radar chart variables
let radarChart = null;
const metricLabels = ['Height', 'Weight', 'Body Fat', 'BMI', 'Body Water', 'Muscle Mass', 'Bone Mass', 'Visceral Fat', 'BMR', 'Protein', 'Body Age'];
const metricKeys = ['height', 'weight', 'body_fat_percentage', 'bmi', 'body_water_percentage', 'muscle_mass', 'bone_mass', 'visceral_fat', 'bmr', 'protein_percentage', 'body_age'];
// Function to create/update radar chart
function updateRadarChart(currentRecord, previousRecord) {
console.log('updateRadarChart called', currentRecord, previousRecord);
const ctx = document.getElementById('radarChart').getContext('2d');
// Extract data for current and previous records
const currentData = metricKeys.map(key => currentRecord ? (currentRecord[key] || 0) : 0);
const previousData = metricKeys.map(key => previousRecord ? (previousRecord[key] || 0) : 0);
if (radarChart) {
radarChart.destroy();
}
radarChart = new window.Chart(ctx, {
type: 'radar',
data: {
labels: metricLabels,
datasets: [
{
label: 'Current Reading',
data: currentData,
fill: true,
backgroundColor: 'rgba(54, 162, 235, 0.2)',
borderColor: 'rgb(54, 162, 235)',
pointBackgroundColor: 'rgb(54, 162, 235)',
pointBorderColor: '#fff',
pointHoverBackgroundColor: '#fff',
pointHoverBorderColor: 'rgb(54, 162, 235)'
},
{
label: 'Previous Reading',
data: previousData,
fill: false,
borderColor: 'rgb(255, 99, 132)',
pointBackgroundColor: 'rgb(255, 99, 132)',
pointBorderColor: '#fff',
pointHoverBackgroundColor: '#fff',
pointHoverBorderColor: 'rgb(255, 99, 132)'
}
]
},
options: {
responsive: true,
maintainAspectRatio: false,
elements: {
line: {
borderWidth: 3
}
},
plugins: {
legend: {
position: 'top',
},
tooltip: {
callbacks: {
label: function(context) {
let label = context.dataset.label || '';
if (label) {
label += ': ';
}
label += context.parsed.r;
return label;
}
}
}
},
scales: {
r: {
beginAtZero: true,
ticks: {
display: false
}
}
}
}
});
}
// Function to calculate BMI
function calculateBMI() {
const height = parseFloat(document.getElementById('height').value);
const weight = parseFloat(document.getElementById('weight').value);
const bmiField = document.getElementById('bmi');
if (height > 0 && weight > 0) {
const heightInMeters = height / 100;
const bmi = weight / (heightInMeters * heightInMeters);
bmiField.value = bmi.toFixed(1);
} else {
bmiField.value = '';
}
}
// Add event listeners for BMI calculation
document.getElementById('height').addEventListener('input', calculateBMI);
document.getElementById('weight').addEventListener('input', calculateBMI);
// Function to reset modal for adding new record
function resetHealthModal() {
document.getElementById('healthUpdateModalLabel').textContent = 'Add Health Update';
document.getElementById('healthUpdateForm').action = '{{ route("family.store-health", $relationship->dependent->id) }}';
document.getElementById('healthUpdateForm').method = 'POST';
document.getElementById('recorded_at').value = '{{ \Carbon\Carbon::now()->format("Y-m-d") }}';
document.getElementById('height').value = '';
document.getElementById('weight').value = '';
document.getElementById('body_fat_percentage').value = '';
document.getElementById('bmi').value = '';
document.getElementById('body_water_percentage').value = '';
document.getElementById('muscle_mass').value = '';
document.getElementById('bone_mass').value = '';
document.getElementById('visceral_fat').value = '';
document.getElementById('bmr').value = '';
document.getElementById('protein_percentage').value = '';
document.getElementById('body_age').value = '';
document.querySelector('#healthUpdateForm button[type="submit"]').textContent = 'Save Health Update';
}
// Function to populate modal for editing
function populateHealthModalForEdit(recordId) {
const record = healthRecordsData.find(r => r.id == recordId);
if (!record) return;
document.getElementById('healthUpdateModalLabel').textContent = 'Edit Health Update';
document.getElementById('healthUpdateForm').action = '{{ route("family.update-health", ["id" => $relationship->dependent->id, "recordId" => "__RECORD_ID__"]) }}'.replace('__RECORD_ID__', recordId);
document.getElementById('healthUpdateForm').method = 'POST';
// Add method spoofing for PUT
let methodInput = document.querySelector('#healthUpdateForm input[name="_method"]');
if (!methodInput) {
methodInput = document.createElement('input');
methodInput.type = 'hidden';
methodInput.name = '_method';
document.getElementById('healthUpdateForm').appendChild(methodInput);
}
methodInput.value = 'PUT';
document.getElementById('recorded_at').value = record.recorded_at.split('T')[0];
document.getElementById('height').value = record.height || '';
document.getElementById('weight').value = record.weight || '';
document.getElementById('body_fat_percentage').value = record.body_fat_percentage || '';
document.getElementById('bmi').value = record.bmi || '';
document.getElementById('body_water_percentage').value = record.body_water_percentage || '';
document.getElementById('muscle_mass').value = record.muscle_mass || '';
document.getElementById('bone_mass').value = record.bone_mass || '';
document.getElementById('visceral_fat').value = record.visceral_fat || '';
document.getElementById('bmr').value = record.bmr || '';
document.getElementById('protein_percentage').value = record.protein_percentage || '';
document.getElementById('body_age').value = record.body_age || '';
document.querySelector('#healthUpdateForm button[type="submit"]').textContent = 'Update Health Update';
}
// Handle comparison dropdown changes
const currentDateSelect = document.getElementById('currentDate');
const previousDateSelect = document.getElementById('previousDate');
if (currentDateSelect && previousDateSelect) {
function updateComparisonTable() {
const currentId = currentDateSelect.value;
const previousId = previousDateSelect.value;
if (!currentId || !previousId) {
document.getElementById('timeDifference').innerHTML = 'Select dates to see time difference';
return;
}
const currentRecord = healthRecordsData.find(r => r.id == currentId);
const previousRecord = healthRecordsData.find(r => r.id == previousId);
if (!currentRecord || !previousRecord) {
document.getElementById('timeDifference').innerHTML = 'Select dates to see time difference';
return;
}
// Update time difference
const timeDiff = calculateTimeDifference(currentRecord.recorded_at, previousRecord.recorded_at);
document.getElementById('timeDifference').innerHTML = `<strong>Time between records:</strong> ${timeDiff}`;
// Update the table rows
updateTableRow('height', currentRecord.height, previousRecord.height);
updateTableRow('weight', currentRecord.weight, previousRecord.weight);
updateTableRow('body_fat', currentRecord.body_fat_percentage, previousRecord.body_fat_percentage);
updateTableRow('bmi', currentRecord.bmi, previousRecord.bmi);
updateTableRow('body_water', currentRecord.body_water_percentage, previousRecord.body_water_percentage);
updateTableRow('muscle_mass', currentRecord.muscle_mass, previousRecord.muscle_mass);
updateTableRow('bone_mass', currentRecord.bone_mass, previousRecord.bone_mass);
updateTableRow('visceral_fat', currentRecord.visceral_fat, previousRecord.visceral_fat);
updateTableRow('bmr', currentRecord.bmr, previousRecord.bmr);
updateTableRow('protein', currentRecord.protein_percentage, previousRecord.protein_percentage);
updateTableRow('body_age', currentRecord.body_age, previousRecord.body_age);
// Update the radar chart
updateRadarChart(currentRecord, previousRecord);
}
function calculateTimeDifference(date1, date2) {
const d1 = new Date(date1);
const d2 = new Date(date2);
const diff = Math.abs(d1 - d2);
const years = Math.floor(diff / (1000 * 60 * 60 * 24 * 365));
const months = Math.floor((diff % (1000 * 60 * 60 * 24 * 365)) / (1000 * 60 * 60 * 24 * 30));
const days = Math.floor((diff % (1000 * 60 * 60 * 24 * 30)) / (1000 * 60 * 60 * 24));
const parts = [];
if (years > 0) parts.push(`${years} year${years > 1 ? 's' : ''}`);
if (months > 0) parts.push(`${months} month${months > 1 ? 's' : ''}`);
if (days > 0) parts.push(`${days} day${days > 1 ? 's' : ''}`);
return parts.length > 0 ? parts.join(' ') : 'Same day';
}
function updateTableRow(metric, currentValue, previousValue) {
const row = document.querySelector(`tr[data-metric="${metric}"]`);
if (!row) return;
const cells = row.querySelectorAll('td');
if (cells.length >= 4) {
// Update current value
if (metric === 'height') {
cells[1].textContent = currentValue ? `${currentValue}cm` : 'N/A';
} else if (metric === 'weight' || metric === 'muscle_mass' || metric === 'bone_mass') {
cells[1].textContent = currentValue ? `${currentValue}kg` : 'N/A';
} else if (metric === 'body_fat' || metric === 'body_water' || metric === 'protein') {
cells[1].textContent = currentValue ? `${currentValue}%` : 'N/A';
} else if (metric === 'bmr') {
cells[1].textContent = currentValue ? `${currentValue}cal` : 'N/A';
} else if (metric === 'body_age') {
cells[1].textContent = currentValue ? `${currentValue}yrs` : 'N/A';
} else {
cells[1].textContent = currentValue || 'N/A';
}
// Update previous value
if (metric === 'height') {
cells[2].textContent = previousValue ? `${previousValue}cm` : 'N/A';
} else if (metric === 'weight' || metric === 'muscle_mass' || metric === 'bone_mass') {
cells[2].textContent = previousValue ? `${previousValue}kg` : 'N/A';
} else if (metric === 'body_fat' || metric === 'body_water' || metric === 'protein') {
cells[2].textContent = previousValue ? `${previousValue}%` : 'N/A';
} else if (metric === 'bmr') {
cells[2].textContent = previousValue ? `${previousValue}cal` : 'N/A';
} else if (metric === 'body_age') {
cells[2].textContent = previousValue ? `${previousValue}yrs` : 'N/A';
} else {
cells[2].textContent = previousValue || 'N/A';
}
// Update change cell
const changeCell = cells[3];
if (currentValue && previousValue) {
const change = currentValue - previousValue;
let arrow = '';
let colorClass = 'text-muted';
if (change > 0) {
arrow = '↑';
colorClass = 'text-danger';
} else if (change < 0) {
arrow = '↓';
colorClass = 'text-success';
} else {
arrow = '—';
colorClass = 'text-muted';
}
changeCell.innerHTML = `<span class="${colorClass}">${arrow} ${Math.abs(change).toFixed(1)}</span>`;
} else {
changeCell.textContent = '-';
}
}
}
currentDateSelect.addEventListener('change', updateComparisonTable);
previousDateSelect.addEventListener('change', updateComparisonTable);
// Initialize chart with default values
updateComparisonTable();
}
// Initialize radar chart with data from data attributes after a delay to ensure tab is shown
setTimeout(() => {
const radarCanvas = document.getElementById('radarChart');
if (radarCanvas) {
const currentData = radarCanvas.dataset.current ? JSON.parse(radarCanvas.dataset.current) : null;
const previousData = radarCanvas.dataset.previous ? JSON.parse(radarCanvas.dataset.previous) : null;
if (currentData && previousData) {
updateRadarChart(currentData, previousData);
} else {
// Initialize with empty chart or message
updateRadarChart(null, null);
}
}
}, 1000);
// Tournament filtering functionality
const sportFilter = document.getElementById('sportFilter');
const tournamentsTable = document.getElementById('tournamentsTable');
const awardCards = document.getElementById('awardCards');
// Global variables for current filters
let currentSportFilter = 'all';
let currentMedalFilter = 'all';
function applyTournamentFilters() {
const rows = tournamentsTable.querySelectorAll('tbody tr');
let visibleRows = 0;
let specialCount = 0, firstCount = 0, secondCount = 0, thirdCount = 0;
rows.forEach(row => {
const sport = row.getAttribute('data-sport');
const performanceCell = row.querySelector('td:nth-child(3)');
let hasMatchingMedal = false;
if (performanceCell) {
const badges = performanceCell.querySelectorAll('.badge');
badges.forEach(badge => {
if (currentMedalFilter === 'all') {
hasMatchingMedal = true;
} else if (currentMedalFilter === 'special' && badge.textContent.includes('Special Award')) {
hasMatchingMedal = true;
} else if (currentMedalFilter === '1st' && badge.textContent.includes('1st Place')) {
hasMatchingMedal = true;
} else if (currentMedalFilter === '2nd' && badge.textContent.includes('2nd Place')) {
hasMatchingMedal = true;
} else if (currentMedalFilter === '3rd' && badge.textContent.includes('3rd Place')) {
hasMatchingMedal = true;
}
});
}
const sportMatch = currentSportFilter === 'all' || sport === currentSportFilter;
const medalMatch = currentMedalFilter === 'all' || hasMatchingMedal;
if (sportMatch && medalMatch) {
row.style.display = '';
visibleRows++;
// Count awards in visible rows
if (performanceCell) {
const badges = performanceCell.querySelectorAll('.badge');
badges.forEach(badge => {
if (badge.textContent.includes('Special Award')) specialCount++;
else if (badge.textContent.includes('1st Place')) firstCount++;
else if (badge.textContent.includes('2nd Place')) secondCount++;
else if (badge.textContent.includes('3rd Place')) thirdCount++;
});
}
} else {
row.style.display = 'none';
}
});
// Update award counts
document.getElementById('specialCount').textContent = specialCount;
document.getElementById('firstCount').textContent = firstCount;
document.getElementById('secondCount').textContent = secondCount;
document.getElementById('thirdCount').textContent = thirdCount;
// Show/hide award cards based on visible rows
if (visibleRows === 0) {
awardCards.style.display = 'none';
} else {
awardCards.style.display = '';
}
}
if (sportFilter && tournamentsTable) {
sportFilter.addEventListener('change', function() {
currentSportFilter = this.value;
applyTournamentFilters();
});
}
// Function to filter tournaments by medal type (called from achievement badges)
window.filterTournamentsByMedal = function(medalType) {
// Switch to tournaments tab
const tournamentsTab = document.getElementById('tournaments-tab');
if (tournamentsTab) {
const tab = new bootstrap.Tab(tournamentsTab);
tab.show();
}
// Set medal filter
currentMedalFilter = medalType;
currentSportFilter = 'all'; // Reset sport filter
// Reset sport filter dropdown
if (sportFilter) {
sportFilter.value = 'all';
}
// Apply filters
applyTournamentFilters();
};
// Goals filtering functionality
const goalFilterCards = document.querySelectorAll('.goal-filter-card');
const goalsContainer = document.querySelector('.row.g-4'); // Container with goal cards
const goalsTitle = document.querySelector('h5.fw-bold'); // The "Goal Tracking" title
if (goalFilterCards.length > 0 && goalsContainer) {
// Add click event to filter cards
goalFilterCards.forEach(card => {
card.addEventListener('click', function() {
const filterType = this.getAttribute('data-filter');
filterGoals(filterType);
// Update card styles to show active filter
goalFilterCards.forEach(c => c.classList.remove('border', 'border-primary', 'shadow-lg'));
this.classList.add('border', 'border-primary', 'shadow-lg');
});
});
// Add click event to title to show all goals
if (goalsTitle) {
goalsTitle.style.cursor = 'pointer';
goalsTitle.addEventListener('click', function() {
filterGoals('all');
// Remove active styles from filter cards
goalFilterCards.forEach(c => c.classList.remove('border', 'border-primary', 'shadow-lg'));
});
}
}
function filterGoals(filterType) {
const goalCards = goalsContainer.querySelectorAll('.col-lg-6');
goalCards.forEach(card => {
const statusBadge = card.querySelector('.badge');
if (!statusBadge) return;
const statusText = statusBadge.textContent.toLowerCase();
if (filterType === 'all') {
card.style.display = '';
} else if (filterType === 'active' && statusText === 'active') {
card.style.display = '';
} else if (filterType === 'completed' && statusText === 'completed') {
card.style.display = '';
} else {
card.style.display = 'none';
}
});
// Update title to show current filter
const titleElement = document.querySelector('h5.fw-bold');
if (titleElement) {
const baseTitle = 'Goal Tracking';
if (filterType === 'all') {
titleElement.innerHTML = `<i class="bi bi-bullseye me-2"></i>${baseTitle}`;
} else {
const filterLabel = filterType === 'active' ? 'Active' : 'Completed';
titleElement.innerHTML = `<i class="bi bi-bullseye me-2"></i>${baseTitle} - ${filterLabel} Goals`;
}
}
}
// Goal editing functionality
const editGoalButtons = document.querySelectorAll('.edit-goal-btn');
const goalEditModal = new bootstrap.Modal(document.getElementById('goalEditModal'));
const goalEditForm = document.getElementById('goalEditForm');
let currentGoalId = null;
// Store goals data for editing
const goalsData = @json($goals);
if (editGoalButtons.length > 0) {
editGoalButtons.forEach(button => {
button.addEventListener('click', function(e) {
e.preventDefault();
const goalId = this.getAttribute('data-goal-id');
populateGoalEditModal(goalId);
goalEditModal.show();
});
});
}
function populateGoalEditModal(goalId) {
const goal = goalsData.find(g => g.id == goalId);
if (!goal) return;
currentGoalId = goalId;
// Update form action
goalEditForm.action = `/family/goal/${goalId}`;
// Populate modal fields
document.getElementById('goalTitleDisplay').textContent = goal.title;
document.getElementById('goalDescriptionDisplay').textContent = goal.description || 'No description';
document.getElementById('current_progress_value').value = goal.current_progress_value;
document.getElementById('goal_status').value = goal.status;
document.getElementById('goalUnitDisplay').textContent = goal.unit;
document.getElementById('goalTargetDisplay').textContent = `${goal.target_value} ${goal.unit}`;
// Update icon
const iconElement = document.getElementById('goalIconDisplay').querySelector('i');
if (goal.icon_type === 'dumbbell') {
iconElement.className = 'bi bi-dumbbell text-white';
} else if (goal.icon_type === 'clock') {
iconElement.className = 'bi bi-clock text-white';
} else {
iconElement.className = 'bi bi-bullseye text-white';
}
// Update progress preview
updateProgressPreview();
}
function updateProgressPreview() {
const currentValue = parseFloat(document.getElementById('current_progress_value').value) || 0;
const goal = goalsData.find(g => g.id == currentGoalId);
if (!goal) return;
const targetValue = goal.target_value;
const percentage = targetValue > 0 ? Math.min(100, (currentValue / targetValue) * 100) : 0;
document.getElementById('progressPreview').style.width = `${percentage}%`;
document.getElementById('progressTextPreview').textContent = `Progress: ${currentValue.toFixed(1)} / ${targetValue.toFixed(1)} ${goal.unit} (${percentage.toFixed(1)}%)`;
}
// Update progress preview on input change
document.getElementById('current_progress_value').addEventListener('input', updateProgressPreview);
// Handle form submission
goalEditForm.addEventListener('submit', function(e) {
e.preventDefault();
const formData = new FormData(this);
formData.append('_method', 'PUT');
fetch(this.action, {
method: 'POST',
body: formData,
headers: {
'X-CSRF-TOKEN': document.querySelector('meta[name="csrf-token"]').getAttribute('content')
}
})
.then(response => response.json())
.then(data => {
if (data.success) {
goalEditModal.hide();
// Reload the page to show updated data
window.location.reload();
} else {
alert('Error updating goal: ' + (data.message || 'Unknown error'));
}
})
.catch(error => {
console.error('Error:', error);
alert('Error updating goal. Please try again.');
});
});
});
</script>
<!-- Chart.js for Skills Wheel -->
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<!-- Affiliations Tab JavaScript -->
<script>
document.addEventListener('DOMContentLoaded', function() {
// Affiliations data
const affiliationsData = @json($clubAffiliations);
let skillsChart = null;
let selectedAffiliationId = null;
// Initialize affiliations functionality
function initAffiliations() {
// Set up timeline click handlers
document.querySelectorAll('.affiliation-card').forEach(card => {
card.addEventListener('click', function() {
const affiliationId = this.getAttribute('data-affiliation-id');
selectAffiliation(affiliationId);
// Update visual selection
document.querySelectorAll('.affiliation-card').forEach(c => {
c.classList.remove('border-primary');
});
this.classList.add('border-primary');
});
});
// Select first affiliation by default if available
if (affiliationsData.length > 0) {
selectAffiliation(affiliationsData[0].id);
}
}
function selectAffiliation(affiliationId) {
selectedAffiliationId = affiliationId;
const affiliation = affiliationsData.find(a => a.id == affiliationId);
if (!affiliation) return;
// Update skills chart
updateSkillsChart(affiliation.skill_acquisitions || []);
// Update affiliation details
updateAffiliationDetails(affiliation);
}
function updateSkillsChart(skills) {
const ctx = document.getElementById('skillsChart').getContext('2d');
const noSkillsMessage = document.getElementById('noSkillsMessage');
if (skills.length === 0) {
if (skillsChart) {
skillsChart.destroy();
skillsChart = null;
}
document.getElementById('skillsChart').style.display = 'none';
noSkillsMessage.classList.remove('d-none');
return;
}
document.getElementById('skillsChart').style.display = 'block';
noSkillsMessage.classList.add('d-none');
// Prepare data for polar area chart
const labels = skills.map(skill => skill.skill_name);
const data = skills.map(skill => skill.duration_months);
const backgroundColors = [
'rgba(54, 162, 235, 0.8)',
'rgba(255, 99, 132, 0.8)',
'rgba(255, 205, 86, 0.8)',
'rgba(75, 192, 192, 0.8)',
'rgba(153, 102, 255, 0.8)',
'rgba(255, 159, 64, 0.8)',
'rgba(199, 199, 199, 0.8)',
'rgba(83, 102, 255, 0.8)',
'rgba(255, 99, 255, 0.8)',
'rgba(99, 255, 132, 0.8)'
];
if (skillsChart) {
skillsChart.destroy();
}
skillsChart = new Chart(ctx, {
type: 'polarArea',
data: {
labels: labels,
datasets: [{
data: data,
backgroundColor: backgroundColors.slice(0, skills.length),
borderWidth: 2,
borderColor: backgroundColors.slice(0, skills.length).map(color => color.replace('0.8', '1')),
}]
},
options: {
responsive: true,
maintainAspectRatio: false,
plugins: {
legend: {
position: 'bottom',
labels: {
padding: 20,
usePointStyle: true
}
},
tooltip: {
callbacks: {
label: function(context) {
const skill = skills[context.dataIndex];
const duration = skill.formatted_duration || `${skill.duration_months} months`;
return `${context.label}: ${duration}`;
}
}
}
},
scales: {
r: {
beginAtZero: true,
ticks: {
display: false
}
}
},
animation: {
animateScale: true,
animateRotate: true
}
}
});
}
function updateAffiliationDetails(affiliation) {
const detailsContainer = document.getElementById('affiliationDetails');
let html = `
<div class="d-flex align-items-center mb-3">
${affiliation.logo ?
`<img src="${affiliation.logo}" alt="${affiliation.club_name}" class="me-3 rounded" style="width: 50px; height: 50px; object-fit: cover;">` :
`<div class="bg-primary text-white rounded d-flex align-items-center justify-content-center me-3" style="width: 50px; height: 50px;">
<i class="bi bi-building"></i>
</div>`
}
<div>
<h5 class="mb-1">${affiliation.club_name}</h5>
<p class="text-muted mb-0">${affiliation.date_range}</p>
<span class="badge bg-info text-dark small">${affiliation.formatted_duration}</span>
</div>
</div>
`;
if (affiliation.location) {
html += `<p class="mb-2"><i class="bi bi-geo-alt me-2"></i><strong>Location:</strong> ${affiliation.location}</p>`;
}
if (affiliation.description) {
html += `<p class="mb-2"><strong>Description:</strong> ${affiliation.description}</p>`;
}
if (affiliation.coaches && affiliation.coaches.length > 0) {
html += `<p class="mb-2"><strong>Coaches:</strong> ${affiliation.coaches.join(', ')}</p>`;
}
if (affiliation.affiliation_media && affiliation.affiliation_media.length > 0) {
html += `<div class="mt-3"><strong>Media & Certificates:</strong></div>`;
html += `<div class="row g-2 mt-1">`;
affiliation.affiliation_media.forEach(media => {
const iconClass = media.icon_class || 'bi-file';
html += `
<div class="col-6">
<a href="${media.full_url}" target="_blank" class="btn btn-outline-secondary btn-sm w-100">
<i class="bi ${iconClass} me-1"></i>${media.title || media.media_type}
</a>
</div>
`;
});
html += `</div>`;
}
detailsContainer.innerHTML = html;
}
// Initialize when affiliations tab is shown
const affiliationsTab = document.getElementById('affiliations-tab');
if (affiliationsTab) {
affiliationsTab.addEventListener('shown.bs.tab', function() {
initAffiliations();
});
}
// Initialize immediately if affiliations tab is active
if (document.getElementById('affiliations').classList.contains('show')) {
initAffiliations();
}
});
// Tournament Participation Modal Functionality
document.addEventListener('DOMContentLoaded', function() {
let performanceResultIndex = 1;
let notesMediaIndex = 1;
// Add Performance Result
document.getElementById('addPerformanceResult').addEventListener('click', function() {
const container = document.getElementById('performanceResultsContainer');
const newItem = document.createElement('div');
newItem.className = 'performance-result-item mb-3 p-3 border rounded';
newItem.innerHTML = `
<div class="row g-2">
<div class="col-md-4">
<label class="form-label">Medal Type</label>
<select class="form-select medal-type" name="performance_results[${performanceResultIndex}][medal_type]">
<option value="">Select Medal</option>
<option value="special">Special Award</option>
<option value="1st">1st Place</option>
<option value="2nd">2nd Place</option>
<option value="3rd">3rd Place</option>
</select>
</div>
<div class="col-md-3">
<label class="form-label">Points</label>
<input type="number" class="form-control" name="performance_results[${performanceResultIndex}][points]" min="0" step="0.1">
</div>
<div class="col-md-4">
<label class="form-label">Description</label>
<input type="text" class="form-control" name="performance_results[${performanceResultIndex}][description]" placeholder="Optional description">
</div>
<div class="col-md-1 d-flex align-items-end">
<button type="button" class="btn btn-outline-danger btn-sm remove-result">
<i class="bi bi-trash"></i>
</button>
</div>
</div>
`;
container.appendChild(newItem);
performanceResultIndex++;
// Show remove buttons if more than one result
updateRemoveButtons('performance-result-item', 'remove-result');
});
// Add Notes & Media
document.getElementById('addNotesMedia').addEventListener('click', function() {
const container = document.getElementById('notesMediaContainer');
const newItem = document.createElement('div');
newItem.className = 'notes-media-item mb-3 p-3 border rounded';
newItem.innerHTML = `
<div class="row g-2">
<div class="col-md-6">
<label class="form-label">Note Text</label>
<textarea class="form-control" name="notes_media[${notesMediaIndex}][note_text]" rows="2" placeholder="Optional notes about the tournament"></textarea>
</div>
<div class="col-md-5">
<label class="form-label">Media Link</label>
<input type="url" class="form-control" name="notes_media[${notesMediaIndex}][media_link]" placeholder="https://example.com/photo.jpg">
</div>
<div class="col-md-1 d-flex align-items-end">
<button type="button" class="btn btn-outline-danger btn-sm remove-note">
<i class="bi bi-trash"></i>
</button>
</div>
</div>
`;
container.appendChild(newItem);
notesMediaIndex++;
// Show remove buttons if more than one note
updateRemoveButtons('notes-media-item', 'remove-note');
});
// Remove Performance Result
document.addEventListener('click', function(e) {
if (e.target.closest('.remove-result')) {
e.target.closest('.performance-result-item').remove();
updateRemoveButtons('performance-result-item', 'remove-result');
}
});
// Remove Notes & Media
document.addEventListener('click', function(e) {
if (e.target.closest('.remove-note')) {
e.target.closest('.notes-media-item').remove();
updateRemoveButtons('notes-media-item', 'remove-note');
}
});
function updateRemoveButtons(itemClass, buttonClass) {
const items = document.querySelectorAll('.' + itemClass);
const buttons = document.querySelectorAll('.' + buttonClass);
if (items.length > 1) {
buttons.forEach(button => button.style.display = 'block');
} else {
buttons.forEach(button => button.style.display = 'none');
}
}
// Reset modal when opened
document.getElementById('tournamentParticipationModal').addEventListener('show.bs.modal', function() {
// Reset form
document.getElementById('tournamentParticipationForm').reset();
// Reset dynamic content
const performanceContainer = document.getElementById('performanceResultsContainer');
const notesContainer = document.getElementById('notesMediaContainer');
// Keep only the first item in each container
const performanceItems = performanceContainer.querySelectorAll('.performance-result-item');
const notesItems = notesContainer.querySelectorAll('.notes-media-item');
for (let i = 1; i < performanceItems.length; i++) {
performanceItems[i].remove();
}
for (let i = 1; i < notesItems.length; i++) {
notesItems[i].remove();
}
// Reset indices
performanceResultIndex = 1;
notesMediaIndex = 1;
// Hide remove buttons
document.querySelectorAll('.remove-result, .remove-note').forEach(button => {
button.style.display = 'none';
});
});
// Handle form submission
document.getElementById('tournamentParticipationForm').addEventListener('submit', function(e) {
e.preventDefault();
const formData = new FormData(this);
fetch(this.action, {
method: 'POST',
body: formData,
headers: {
'X-CSRF-TOKEN': document.querySelector('meta[name="csrf-token"]').getAttribute('content')
}
})
.then(response => response.json())
.then(data => {
if (data.success) {
// Close modal
const modal = bootstrap.Modal.getInstance(document.getElementById('tournamentParticipationModal'));
modal.hide();
// Show success message
showAlert('Tournament record added successfully!', 'success');
// Reload page to show new data
setTimeout(() => {
window.location.reload();
}, 1500);
} else {
showAlert('Error adding tournament record: ' + (data.message || 'Unknown error'), 'danger');
}
})
.catch(error => {
console.error('Error:', error);
showAlert('Error adding tournament record. Please try again.', 'danger');
});
});
function showAlert(message, type) {
const alertDiv = document.createElement('div');
alertDiv.className = `alert alert-${type} alert-dismissible fade show position-fixed`;
alertDiv.style.cssText = 'top: 20px; right: 20px; z-index: 9999; min-width: 300px;';
alertDiv.innerHTML = `
${message}
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
`;
document.body.appendChild(alertDiv);
// Auto remove after 5 seconds
setTimeout(() => {
if (alertDiv.parentNode) {
alertDiv.remove();
}
}, 5000);
}
});
</script>
@endsection