takeone/resources/views/member/partials/affiliations-enhanced.blade.php

917 lines
58 KiB
PHP

@php
// Helper function to calculate age at a specific date with detailed format
function calculateAgeAtDate($birthdate, $date) {
if (!$birthdate || !$date) return null;
$birth = \Carbon\Carbon::parse($birthdate);
$targetDate = \Carbon\Carbon::parse($date);
$diff = $birth->diff($targetDate);
$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
<div class="card shadow-sm border-0">
<div class="card-body p-4">
<!-- Header with Filter -->
<div class="d-flex justify-content-between align-items-center mb-4">
<div>
<h5 class="fw-bold mb-1"><i class="bi bi-diagram-3 me-2"></i>Club Affiliations & Skills Journey</h5>
<p class="text-muted small mb-0">Complete history of club memberships, skills acquired, and instructors</p>
</div>
<div class="d-flex gap-2">
<select class="form-select form-select-sm" id="skillFilter" style="width: 200px;">
<option value="all">All Skills</option>
@foreach($allSkills ?? [] as $skill)
<option value="{{ $skill }}">{{ $skill }}</option>
@endforeach
</select>
<button class="btn btn-sm btn-outline-secondary" id="resetFilters">
<i class="bi bi-arrow-clockwise"></i> Reset
</button>
</div>
</div>
@if($clubAffiliations->count() > 0)
<!-- Summary Stats -->
<div class="row g-3 mb-4">
<div class="col-md-3">
<div class="card shadow-sm h-100" style="background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); border: none;">
<div class="card-body text-center text-white p-3">
<i class="bi bi-building display-5 mb-2"></i>
<h3 class="fw-bold mb-1">{{ $totalAffiliations }}</h3>
<small class="opacity-75">Total Clubs</small>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card shadow-sm h-100" style="background: linear-gradient(135deg, #f093fb 0%, #f5576c 100%); border: none;">
<div class="card-body text-center text-white p-3">
<i class="bi bi-star-fill display-5 mb-2"></i>
<h3 class="fw-bold mb-1">{{ $distinctSkills }}</h3>
<small class="opacity-75">Unique Skills</small>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card shadow-sm h-100" style="background: linear-gradient(135deg, #4facfe 0%, #00f2fe 100%); border: none;">
<div class="card-body text-center text-white p-3">
<i class="bi bi-calendar-check display-5 mb-2"></i>
<h3 class="fw-bold mb-1">{{ floor($totalMembershipDuration / 12) }}y {{ $totalMembershipDuration % 12 }}m</h3>
<small class="opacity-75">Total Training</small>
</div>
</div>
</div>
<div class="col-md-3">
<div class="card shadow-sm h-100" style="background: linear-gradient(135deg, #fa709a 0%, #fee140 100%); border: none;">
<div class="card-body text-center text-white p-3">
<i class="bi bi-people-fill display-5 mb-2"></i>
<h3 class="fw-bold mb-1">{{ $totalInstructors ?? 0 }}</h3>
<small class="opacity-75">Instructors</small>
</div>
</div>
</div>
</div>
<!-- Timeline -->
<div class="row">
<div class="col-12">
<div class="card shadow-sm border-0">
<div class="card-header" style="background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); border: none;">
<h6 class="card-title mb-0 text-white">
<i class="bi bi-clock-history me-2"></i>Membership Timeline
</h6>
</div>
<div class="card-body p-4" style="max-height: 800px; overflow-y: auto;">
<div class="timeline-enhanced">
@foreach($clubAffiliations as $index => $affiliation)
@php
$ageAtStart = calculateAgeAtDate($user->birthdate, $affiliation->start_date);
$ageAtEnd = $affiliation->end_date ? calculateAgeAtDate($user->birthdate, $affiliation->end_date) : null;
$isOngoing = !$affiliation->end_date;
// Get all skills for this affiliation
$affiliationSkills = $affiliation->skillAcquisitions ?? collect();
$skillNames = $affiliationSkills->pluck('skill_name')->unique()->implode(',');
@endphp
<div class="timeline-item-enhanced mb-4" data-affiliation-id="{{ $affiliation->id }}" data-skills="{{ $skillNames }}">
<!-- Timeline Marker -->
<div class="timeline-marker-enhanced {{ $isOngoing ? 'pulse' : '' }}"></div>
<!-- Affiliation Card -->
<div class="affiliation-card-enhanced card border-0 shadow-sm">
<!-- Card Header with Gradient -->
<div class="card-header border-0 p-3" style="background: linear-gradient(135deg, {{ $index % 4 == 0 ? '#667eea 0%, #764ba2' : ($index % 4 == 1 ? '#f093fb 0%, #f5576c' : ($index % 4 == 2 ? '#4facfe 0%, #00f2fe' : '#fa709a 0%, #fee140')) }} 100%);">
<div class="d-flex align-items-center">
@if($affiliation->logo)
<img src="{{ asset('storage/' . $affiliation->logo) }}" alt="{{ $affiliation->club_name }}" class="rounded-circle me-3" style="width: 50px; height: 50px; object-fit: cover; border: 3px solid white;">
@else
<div class="rounded-circle bg-white d-flex align-items-center justify-content-center me-3" style="width: 50px; height: 50px;">
<i class="bi bi-building" style="font-size: 1.5rem; color: #667eea;"></i>
</div>
@endif
<div class="flex-grow-1 text-white">
<h5 class="mb-1 fw-bold">{{ $affiliation->club_name }}</h5>
<div class="d-flex gap-3 flex-wrap">
@if($affiliation->start_date)
<small class="opacity-90">
<i class="bi bi-calendar-event me-1"></i>{{ $affiliation->start_date->format('M Y') }} - {{ $isOngoing ? 'Present' : ($affiliation->end_date ? $affiliation->end_date->format('M Y') : 'N/A') }}
</small>
@endif
@if($affiliation->formatted_duration)
<small class="opacity-90">
<i class="bi bi-hourglass-split me-1"></i>{{ $affiliation->formatted_duration }}
</small>
@endif
@if($ageAtStart)
<small class="opacity-90">
<i class="bi bi-person me-1"></i>Age: {{ $ageAtStart }}{{ $ageAtEnd && $ageAtEnd != $ageAtStart ? " to $ageAtEnd" : '' }}
</small>
@endif
</div>
</div>
@if($isOngoing)
<span class="badge bg-success">
<i class="bi bi-circle-fill me-1" style="font-size: 0.5rem;"></i>Active
</span>
@endif
</div>
</div>
<!-- Card Body -->
<div class="card-body p-3">
@if($affiliation->location)
<div class="mb-3">
<i class="bi bi-geo-alt text-primary me-2"></i>
<span class="text-muted">{{ $affiliation->location }}</span>
</div>
@endif
<!-- Skills Acquired as Badges -->
@if($affiliationSkills->count() > 0)
<div class="mb-3">
<h6 class="fw-bold mb-2">
<i class="bi bi-star-fill me-2 text-warning"></i>Skills Acquired ({{ $affiliationSkills->count() }})
</h6>
<div class="d-flex gap-2 flex-wrap">
@foreach($affiliationSkills as $skill)
<span class="badge skill-badge bg-{{ $skill->proficiency_level == 'expert' ? 'danger' : ($skill->proficiency_level == 'advanced' ? 'warning' : ($skill->proficiency_level == 'intermediate' ? 'info' : 'secondary')) }}"
data-bs-toggle="tooltip"
data-bs-placement="top"
data-bs-html="true"
title="<strong>{{ $skill->skill_name }}</strong><br>
Proficiency: {{ ucfirst($skill->proficiency_level) }}<br>
Duration: {{ $skill->formatted_duration }}<br>
@if($skill->instructor)Instructor: {{ $skill->instructor->user->full_name ?? 'Unknown' }}<br>@endif
@if($skill->start_date)Started: {{ $skill->start_date->format('M Y') }}@endif">
<i class="bi bi-star-fill me-1"></i>{{ $skill->skill_name }}
<span class="badge bg-white text-dark ms-1" style="font-size: 0.65rem;">{{ ucfirst($skill->proficiency_level) }}</span>
</span>
@endforeach
</div>
</div>
@endif
<!-- Training Packages -->
@if($affiliation->subscriptions && $affiliation->subscriptions->count() > 0)
<div class="mb-3">
<h6 class="fw-bold mb-2">
<i class="bi bi-box-seam me-2 text-primary"></i>Training Packages ({{ $affiliation->subscriptions->count() }})
</h6>
<div class="d-flex gap-2 flex-wrap">
@foreach($affiliation->subscriptions as $subIndex => $subscription)
@if($subscription->package)
<button type="button" class="btn btn-sm btn-outline-primary package-card-btn"
data-bs-toggle="modal"
data-bs-target="#packageModal_{{ $affiliation->id }}_{{ $subscription->id }}">
<i class="bi bi-box me-1"></i>{{ $subscription->package->name }}
</button>
@endif
@endforeach
</div>
</div>
@endif
<!-- Instructors -->
@php
$instructors = $affiliationSkills->pluck('instructor')->filter()->unique('id');
@endphp
@if($instructors->count() > 0)
<div class="mb-2">
<h6 class="fw-bold mb-2">
<i class="bi bi-people-fill me-2 text-success"></i>Instructors ({{ $instructors->count() }})
</h6>
<div class="d-flex gap-2 flex-wrap">
@foreach($instructors as $instructor)
<div class="instructor-badge" role="button"
data-bs-toggle="modal"
data-bs-target="#instructorModal_{{ $instructor->id }}">
<div class="d-flex align-items-center gap-2 p-2 bg-light rounded">
<div class="rounded-circle bg-success text-white d-flex align-items-center justify-content-center" style="width: 32px; height: 32px; font-size: 0.8rem;">
{{ strtoupper(substr($instructor->user->full_name ?? 'I', 0, 1)) }}
</div>
<div>
<div class="fw-semibold small">{{ $instructor->user->full_name ?? 'Unknown' }}</div>
<div class="text-muted" style="font-size: 0.7rem;">{{ $instructor->role ?? 'Instructor' }}</div>
</div>
</div>
</div>
@endforeach
</div>
</div>
@endif
</div>
</div>
</div>
@endforeach
</div>
<!-- Instructor Modals -->
@php
$allInstructors = collect();
foreach($clubAffiliations as $aff) {
$affInstructors = $aff->skillAcquisitions->pluck('instructor')->filter()->unique('id');
$allInstructors = $allInstructors->merge($affInstructors);
}
$allInstructors = $allInstructors->unique('id');
@endphp
@foreach($allInstructors as $instructor)
<div class="modal fade" id="instructorModal_{{ $instructor->id }}" tabindex="-1" aria-hidden="true">
<div class="modal-dialog modal-dialog-centered modal-dialog-scrollable" style="max-width: 600px;">
<div class="modal-content">
<div class="modal-header" style="background: linear-gradient(135deg, #11998e 0%, #38ef7d 100%);">
<h5 class="modal-title text-white">
<i class="bi bi-person-badge me-2"></i>Instructor Profile
</h5>
<button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body" style="max-height: 70vh; overflow-y: auto;">
<div class="text-center mb-4">
<!-- Profile Picture -->
<div class="mb-3">
@if($instructor->user->profile_picture)
<img src="{{ asset('storage/' . $instructor->user->profile_picture) }}"
alt="{{ $instructor->user->full_name }}"
class="rounded-circle"
style="width: 100px; height: 100px; object-fit: cover; border: 4px solid #11998e;">
@else
<div class="rounded-circle mx-auto d-flex align-items-center justify-content-center text-white"
style="width: 100px; height: 100px; font-size: 2.5rem; background: linear-gradient(135deg, #11998e 0%, #38ef7d 100%);">
{{ strtoupper(substr($instructor->user->full_name ?? 'I', 0, 1)) }}
</div>
@endif
</div>
<!-- Name & Role -->
<h5 class="fw-bold mb-1">{{ $instructor->user->full_name ?? 'Unknown Instructor' }}</h5>
<p class="text-muted mb-2">{{ $instructor->role ?? 'Instructor' }}</p>
<!-- Average Rating -->
@php
$avgRating = $instructor->reviews()->avg('rating') ?? 0;
$reviewCount = $instructor->reviews()->count();
@endphp
<div class="mb-3">
<div class="d-flex justify-content-center align-items-center gap-2">
<div class="stars-display">
@for($i = 1; $i <= 5; $i++)
<i class="bi bi-star{{ $i <= round($avgRating) ? '-fill' : '' }} text-warning"></i>
@endfor
</div>
<span class="text-muted small">({{ number_format($avgRating, 1) }} / {{ $reviewCount }} {{ $reviewCount == 1 ? 'review' : 'reviews' }})</span>
</div>
</div>
</div>
<!-- Quick Stats -->
<div class="row g-2 mb-3">
@php
$instructorSkills = \App\Models\SkillAcquisition::where('instructor_id', $instructor->id)->get();
$studentsCount = $instructorSkills->pluck('clubAffiliation.member_id')->unique()->count();
$skillsTaught = $instructorSkills->pluck('skill_name')->unique();
@endphp
<div class="col-6">
<div class="card bg-light border-0">
<div class="card-body p-2">
<div class="h4 mb-0 text-primary">{{ $studentsCount }}</div>
<small class="text-muted">Students</small>
</div>
</div>
</div>
<div class="col-6">
<div class="card bg-light border-0">
<div class="card-body p-2">
<div class="h4 mb-0 text-success">{{ $skillsTaught->count() }}</div>
<small class="text-muted">Skills</small>
</div>
</div>
</div>
</div>
<!-- Skills Taught -->
@if($skillsTaught->count() > 0)
<div class="mb-3 text-start">
<label class="text-muted small fw-semibold mb-2">Specializes In:</label>
<div class="d-flex gap-1 flex-wrap">
@foreach($skillsTaught as $skill)
<span class="badge bg-success">{{ $skill }}</span>
@endforeach
</div>
</div>
@endif
<!-- Contact Info -->
@if($instructor->user->email)
<div class="mb-2 text-start">
<small class="text-muted">
<i class="bi bi-envelope me-1"></i>{{ $instructor->user->email }}
</small>
</div>
@endif
@if($instructor->user->mobile)
<div class="mb-3 text-start">
<small class="text-muted">
<i class="bi bi-phone me-1"></i>{{ $instructor->user->mobile }}
</small>
</div>
@endif
<!-- Reviews Section -->
<div class="mt-4">
<h6 class="fw-bold mb-3">
<i class="bi bi-chat-left-text me-2"></i>Reviews
</h6>
<!-- Add/Edit Review Form -->
@php
$userReview = $instructor->reviews()->where('reviewer_user_id', auth()->id())->first();
@endphp
<div class="card bg-light mb-3" id="reviewForm_{{ $instructor->id }}">
<div class="card-body">
<form class="instructor-review-form" data-instructor-id="{{ $instructor->id }}" data-review-id="{{ $userReview->id ?? '' }}">
@csrf
<div class="mb-3">
<label class="form-label small fw-semibold">Your Rating</label>
<div class="star-rating" data-rating="{{ $userReview->rating ?? 0 }}">
@for($i = 1; $i <= 5; $i++)
<i class="bi bi-star{{ $userReview && $i <= $userReview->rating ? '-fill' : '' }} star-input" data-value="{{ $i }}"></i>
@endfor
</div>
<input type="hidden" name="rating" value="{{ $userReview->rating ?? 0 }}" required>
</div>
<div class="mb-3">
<label class="form-label small fw-semibold">Your Review</label>
<textarea name="comment" class="form-control form-control-sm" rows="3" placeholder="Share your experience...">{{ $userReview->comment ?? '' }}</textarea>
</div>
<button type="submit" class="btn btn-success btn-sm w-100">
<i class="bi bi-{{ $userReview ? 'pencil' : 'plus-circle' }} me-1"></i>
{{ $userReview ? 'Update Review' : 'Submit Review' }}
</button>
</form>
</div>
</div>
<!-- Reviews List -->
<div class="reviews-list" id="reviewsList_{{ $instructor->id }}">
@foreach($instructor->reviews()->with('reviewer')->latest()->get() as $review)
<div class="card mb-2">
<div class="card-body p-3">
<div class="d-flex align-items-start mb-2">
<div class="flex-grow-1">
<div class="fw-semibold small">{{ $review->reviewer->full_name }}</div>
<div class="stars-display small">
@for($i = 1; $i <= 5; $i++)
<i class="bi bi-star{{ $i <= $review->rating ? '-fill' : '' }} text-warning"></i>
@endfor
</div>
</div>
<small class="text-muted">
{{ $review->wasUpdated() ? 'Updated ' : '' }}{{ $review->wasUpdated() ? $review->updated_at->diffForHumans() : $review->reviewed_at->diffForHumans() }}
</small>
</div>
@if($review->comment)
<p class="mb-0 small text-muted">{{ $review->comment }}</p>
@endif
</div>
</div>
@endforeach
</div>
</div>
</div>
<div class="modal-footer">
<a href="{{ route('family.show', $instructor->user_id) }}" class="btn btn-primary btn-sm">
<i class="bi bi-person-lines-fill me-1"></i>View Full Profile
</a>
<button type="button" class="btn btn-secondary btn-sm" data-bs-dismiss="modal">Close</button>
</div>
</div>
</div>
</div>
@endforeach
<!-- Package Modals (Outside Timeline Loop) -->
@foreach($clubAffiliations as $affiliation)
@foreach($affiliation->subscriptions as $subIndex => $subscription)
@if($subscription->package)
<div class="modal fade" id="packageModal_{{ $affiliation->id }}_{{ $subscription->id }}" tabindex="-1" aria-hidden="true">
<div class="modal-dialog modal-dialog-centered modal-dialog-scrollable">
<div class="modal-content">
<div class="modal-header" style="background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);">
<h5 class="modal-title text-white">
<i class="bi bi-box-seam me-2"></i>{{ $subscription->package->name }}
</h5>
<button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
<div class="mb-3">
<label class="text-muted small fw-semibold">Subscription Period</label>
<div>
<i class="bi bi-calendar-range me-2 text-primary"></i>
{{ $subscription->start_date ? $subscription->start_date->format('M d, Y') : 'N/A' }} - {{ $subscription->end_date ? $subscription->end_date->format('M d, Y') : 'N/A' }}
</div>
@php
$durationText = 'N/A';
if ($subscription->start_date && $subscription->end_date) {
$duration = $subscription->start_date->diff($subscription->end_date);
$durationParts = [];
if ($duration->y > 0) $durationParts[] = $duration->y . ' year' . ($duration->y > 1 ? 's' : '');
if ($duration->m > 0) $durationParts[] = $duration->m . ' month' . ($duration->m > 1 ? 's' : '');
if ($duration->d > 0) $durationParts[] = $duration->d . ' day' . ($duration->d > 1 ? 's' : '');
$durationText = implode(' ', $durationParts) ?: 'Same day';
}
@endphp
<small class="text-muted">
<i class="bi bi-hourglass-split me-1"></i>Duration: {{ $durationText }}
</small>
</div>
@if($subscription->package->description)
<div class="mb-3">
<label class="text-muted small fw-semibold">Description</label>
<p class="mb-0">{{ $subscription->package->description }}</p>
</div>
@endif
@if($subscription->package->price)
<div class="mb-3">
<label class="text-muted small fw-semibold">Price</label>
<div class="h5 mb-0 text-success">
<i class="bi bi-currency-dollar"></i>{{ number_format($subscription->package->price, 2) }}
</div>
</div>
@endif
@if($subscription->package->packageActivities && $subscription->package->packageActivities->count() > 0)
<div class="mb-3">
<label class="text-muted small fw-semibold">Activities & Skills Included</label>
<div class="list-group">
@foreach($subscription->package->packageActivities as $pkgActivity)
@if($pkgActivity->activity)
<div class="list-group-item">
<div class="d-flex align-items-start mb-2">
<i class="bi bi-check-circle-fill text-success me-2 mt-1"></i>
<div class="flex-grow-1">
<div class="fw-semibold">{{ $pkgActivity->activity->name }}</div>
@if($pkgActivity->activity->description)
<small class="text-muted d-block mb-2">{{ $pkgActivity->activity->description }}</small>
@endif
@php
// Get skills taught in this activity
$activitySkills = \App\Models\SkillAcquisition::where('activity_id', $pkgActivity->activity_id)
->where('club_affiliation_id', $affiliation->id)
->get();
@endphp
@if($activitySkills->count() > 0)
<div class="mb-2">
<small class="text-muted d-block mb-1">Skills Practiced:</small>
<div class="d-flex gap-1 flex-wrap">
@foreach($activitySkills as $actSkill)
<span class="badge bg-{{ $actSkill->proficiency_level == 'expert' ? 'danger' : ($actSkill->proficiency_level == 'advanced' ? 'warning' : ($actSkill->proficiency_level == 'intermediate' ? 'info' : 'secondary')) }}" style="font-size: 0.7rem;">
<i class="bi bi-star-fill me-1"></i>{{ $actSkill->skill_name }}
</span>
@endforeach
</div>
</div>
@endif
</div>
@if($pkgActivity->instructor && $pkgActivity->instructor->user)
<div class="text-end">
<small class="text-muted">
<i class="bi bi-person-badge"></i>
{{ $pkgActivity->instructor->user->full_name }}
</small>
</div>
@endif
</div>
</div>
@endif
@endforeach
</div>
</div>
@endif
@php
// Check if this package was subscribed to multiple times
$samePackageSubscriptions = $affiliation->subscriptions
->where('package_id', $subscription->package_id)
->where('id', '!=', $subscription->id);
@endphp
@if($samePackageSubscriptions->count() > 0)
<div class="mb-3">
<label class="text-muted small fw-semibold">
<i class="bi bi-arrow-repeat me-1"></i>Other Subscriptions to This Package
</label>
<div class="alert alert-info mb-0" style="font-size: 0.85rem;">
<div class="fw-semibold mb-1">You subscribed to this package {{ $samePackageSubscriptions->count() + 1 }} times:</div>
<ul class="mb-0 ps-3">
<li class="text-primary fw-semibold">
{{ $subscription->start_date ? $subscription->start_date->format('M d, Y') : 'N/A' }} - {{ $subscription->end_date ? $subscription->end_date->format('M d, Y') : 'N/A' }} (Current)
</li>
@foreach($samePackageSubscriptions as $otherSub)
<li>
{{ $otherSub->start_date ? $otherSub->start_date->format('M d, Y') : 'N/A' }} - {{ $otherSub->end_date ? $otherSub->end_date->format('M d, Y') : 'N/A' }}
@php
$gap = 0;
if ($subscription->start_date && $otherSub->start_date) {
$gap = $subscription->start_date->diffInMonths($otherSub->start_date);
}
@endphp
@if($gap > 0)
<small class="text-muted">({{ abs($gap) }} months {{ $subscription->start_date->gt($otherSub->start_date) ? 'before' : 'after' }} current)</small>
@endif
</li>
@endforeach
</ul>
</div>
</div>
@endif
<div class="mb-0">
<label class="text-muted small fw-semibold">Status</label>
<div>
<span class="badge bg-{{ $subscription->status == 'active' ? 'success' : 'secondary' }}">
{{ ucfirst($subscription->status) }}
</span>
<span class="badge bg-{{ $subscription->payment_status == 'paid' ? 'success' : 'warning' }} ms-2">
Payment: {{ ucfirst($subscription->payment_status) }}
</span>
</div>
</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Close</button>
</div>
</div>
</div>
</div>
@endif
@endforeach
@endforeach
</div>
</div>
</div>
</div>
@else
<div class="text-center py-5">
<i class="bi bi-diagram-3 text-muted" style="font-size: 3rem;"></i>
<h5 class="text-muted mt-3 mb-2">No Affiliations Yet</h5>
<p class="text-muted mb-0">Club affiliations and skills will appear here once added</p>
</div>
@endif
</div>
</div>
<style>
/* Enhanced Timeline Styles */
.timeline-enhanced {
position: relative;
padding-left: 40px;
}
.timeline-enhanced::before {
content: '';
position: absolute;
left: 20px;
top: 0;
bottom: 0;
width: 3px;
background: linear-gradient(180deg, #667eea 0%, #764ba2 100%);
}
.timeline-item-enhanced {
position: relative;
animation: fadeInUp 0.5s ease-out;
}
@keyframes fadeInUp {
from {
opacity: 0;
transform: translateY(20px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
.timeline-marker-enhanced {
position: absolute;
left: -28px;
top: 25px;
width: 16px;
height: 16px;
border-radius: 50%;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
border: 3px solid #fff;
box-shadow: 0 0 0 3px rgba(102, 126, 234, 0.2);
z-index: 1;
}
.timeline-marker-enhanced.pulse {
animation: pulse 2s infinite;
}
@keyframes pulse {
0%, 100% {
box-shadow: 0 0 0 3px rgba(102, 126, 234, 0.2);
}
50% {
box-shadow: 0 0 0 8px rgba(102, 126, 234, 0.4);
}
}
.affiliation-card-enhanced {
transition: all 0.3s ease;
margin-left: 10px;
}
.affiliation-card-enhanced:hover {
transform: translateX(5px);
box-shadow: 0 8px 16px rgba(0,0,0,0.15) !important;
}
.skill-badge {
padding: 0.5rem 0.75rem;
font-size: 0.85rem;
font-weight: 500;
transition: all 0.2s ease;
cursor: pointer;
}
.skill-badge:hover {
transform: translateY(-2px);
box-shadow: 0 4px 8px rgba(0,0,0,0.2);
}
.package-card-btn {
transition: all 0.2s ease;
}
.package-card-btn:hover {
transform: translateY(-2px);
box-shadow: 0 4px 8px rgba(0,0,0,0.15);
}
.modal-dialog {
max-width: 600px;
}
.modal-body {
max-height: 60vh;
overflow-y: auto;
}
.instructor-badge {
transition: all 0.2s ease;
cursor: pointer;
}
.instructor-badge:hover {
transform: scale(1.05);
box-shadow: 0 4px 8px rgba(0,0,0,0.15);
}
.instructor-badge:active {
transform: scale(0.98);
}
/* Star Rating Styles */
.star-rating {
font-size: 1.5rem;
cursor: pointer;
}
.star-rating .star-input {
color: #ddd;
transition: color 0.2s ease;
}
.star-rating .star-input:hover,
.star-rating .star-input.active {
color: #ffc107;
}
.star-rating .bi-star-fill {
color: #ffc107;
}
.stars-display {
font-size: 1rem;
}
.stars-display .bi-star-fill {
color: #ffc107;
}
.stars-display .bi-star {
color: #ddd;
}
/* Filter transition */
.timeline-item-enhanced.filtered-out {
display: none;
animation: fadeOut 0.3s ease-out;
}
@keyframes fadeOut {
from {
opacity: 1;
transform: translateY(0);
}
to {
opacity: 0;
transform: translateY(-20px);
}
}
</style>
<script>
document.addEventListener('DOMContentLoaded', function() {
const skillFilter = document.getElementById('skillFilter');
const resetButton = document.getElementById('resetFilters');
const timelineItems = document.querySelectorAll('.timeline-item-enhanced');
// Initialize Bootstrap tooltips
const tooltipTriggerList = [].slice.call(document.querySelectorAll('[data-bs-toggle="tooltip"]'));
tooltipTriggerList.map(function (tooltipTriggerEl) {
return new bootstrap.Tooltip(tooltipTriggerEl);
});
// Star Rating Functionality
document.querySelectorAll('.star-rating').forEach(ratingContainer => {
const stars = ratingContainer.querySelectorAll('.star-input');
const ratingInput = ratingContainer.parentElement.querySelector('input[name="rating"]');
stars.forEach((star, index) => {
star.addEventListener('click', function() {
const value = this.getAttribute('data-value');
ratingInput.value = value;
// Update star display
stars.forEach((s, i) => {
if (i < value) {
s.classList.remove('bi-star');
s.classList.add('bi-star-fill');
} else {
s.classList.remove('bi-star-fill');
s.classList.add('bi-star');
}
});
});
star.addEventListener('mouseenter', function() {
const value = this.getAttribute('data-value');
stars.forEach((s, i) => {
if (i < value) {
s.classList.add('active');
} else {
s.classList.remove('active');
}
});
});
});
ratingContainer.addEventListener('mouseleave', function() {
stars.forEach(s => s.classList.remove('active'));
});
});
// Review Form Submission
document.querySelectorAll('.instructor-review-form').forEach(form => {
form.addEventListener('submit', function(e) {
e.preventDefault();
const instructorId = this.getAttribute('data-instructor-id');
const reviewId = this.getAttribute('data-review-id');
const formData = new FormData(this);
const data = {
rating: formData.get('rating'),
comment: formData.get('comment')
};
// Validate rating
if (!data.rating || data.rating == 0) {
alert('Please select a rating');
return;
}
const url = reviewId
? `/instructor/reviews/${reviewId}`
: `/instructor/${instructorId}/reviews`;
const method = reviewId ? 'PUT' : 'POST';
fetch(url, {
method: method,
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': document.querySelector('meta[name="csrf-token"]').getAttribute('content')
},
body: JSON.stringify(data)
})
.then(response => response.json())
.then(result => {
if (result.success) {
// Show success message
showAlert(result.message, 'success');
// Reload the page to show updated reviews
setTimeout(() => {
window.location.reload();
}, 1000);
} else {
showAlert(result.message || 'Error submitting review', 'danger');
}
})
.catch(error => {
console.error('Error:', error);
showAlert('Error submitting review. 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);
setTimeout(() => {
if (alertDiv.parentNode) {
alertDiv.remove();
}
}, 5000);
}
if (skillFilter) {
skillFilter.addEventListener('change', function() {
const selectedSkill = this.value;
timelineItems.forEach(item => {
const itemSkills = item.getAttribute('data-skills');
if (selectedSkill === 'all') {
item.classList.remove('filtered-out');
item.style.display = '';
} else {
if (itemSkills && itemSkills.includes(selectedSkill)) {
item.classList.remove('filtered-out');
item.style.display = '';
} else {
item.classList.add('filtered-out');
setTimeout(() => {
item.style.display = 'none';
}, 300);
}
}
});
});
}
if (resetButton) {
resetButton.addEventListener('click', function() {
if (skillFilter) {
skillFilter.value = 'all';
skillFilter.dispatchEvent(new Event('change'));
}
});
}
});
</script>