- Implemented dashboard view with vehicle stats, active services, recent activity, and upcoming appointments. - Created estimates view with filtering options and a list of service estimates. - Developed invoices view to manage service invoices and payment history with filtering. - Added vehicles view to display registered vehicles and their details. - Built work orders view to track the progress of vehicle services with filtering and detailed information.
246 lines
15 KiB
PHP
246 lines
15 KiB
PHP
<div>
|
|
<!-- Dashboard Header -->
|
|
<div class="mb-8">
|
|
<div class="flex justify-between items-center">
|
|
<div>
|
|
<h1 class="text-2xl font-bold text-gray-900">Welcome back, {{ Auth::user()->name }}!</h1>
|
|
<p class="text-gray-600">Here's an overview of your vehicle services and appointments.</p>
|
|
</div>
|
|
<button wire:click="refreshDashboard"
|
|
class="inline-flex items-center px-3 py-2 border border-gray-300 shadow-sm text-sm leading-4 font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500">
|
|
<svg class="h-4 w-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"></path>
|
|
</svg>
|
|
Refresh
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Stats Grid -->
|
|
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6 mb-8">
|
|
<div class="bg-white rounded-lg shadow p-6">
|
|
<div class="flex items-center">
|
|
<div class="flex-shrink-0">
|
|
<svg class="h-8 w-8 text-blue-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19.428 15.428a2 2 0 00-1.022-.547l-2.387-.477a6 6 0 00-3.86.517l-.318.158a6 6 0 01-3.86.517L6.05 15.21a2 2 0 00-1.806.547M8 4h8l-1 1v5.172a2 2 0 00.586 1.414l5 5c1.26 1.26.367 3.414-1.415 3.414H4.828c-1.782 0-2.674-2.154-1.414-3.414l5-5A2 2 0 009 11.172V5l-1-1z"></path>
|
|
</svg>
|
|
</div>
|
|
<div class="ml-5 w-0 flex-1">
|
|
<dl>
|
|
<dt class="text-sm font-medium text-gray-500 truncate">Total Vehicles</dt>
|
|
<dd class="text-lg font-medium text-gray-900">{{ $stats['total_vehicles'] }}</dd>
|
|
</dl>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="bg-white rounded-lg shadow p-6">
|
|
<div class="flex items-center">
|
|
<div class="flex-shrink-0">
|
|
<svg class="h-8 w-8 text-yellow-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"></path>
|
|
</svg>
|
|
</div>
|
|
<div class="ml-5 w-0 flex-1">
|
|
<dl>
|
|
<dt class="text-sm font-medium text-gray-500 truncate">Active Services</dt>
|
|
<dd class="text-lg font-medium text-gray-900">{{ $stats['active_jobs'] }}</dd>
|
|
</dl>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="bg-white rounded-lg shadow p-6">
|
|
<div class="flex items-center">
|
|
<div class="flex-shrink-0">
|
|
<svg class="h-8 w-8 text-orange-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"></path>
|
|
</svg>
|
|
</div>
|
|
<div class="ml-5 w-0 flex-1">
|
|
<dl>
|
|
<dt class="text-sm font-medium text-gray-500 truncate">Pending Estimates</dt>
|
|
<dd class="text-lg font-medium text-gray-900">{{ $stats['pending_estimates'] }}</dd>
|
|
</dl>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="bg-white rounded-lg shadow p-6">
|
|
<div class="flex items-center">
|
|
<div class="flex-shrink-0">
|
|
<svg class="h-8 w-8 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"></path>
|
|
</svg>
|
|
</div>
|
|
<div class="ml-5 w-0 flex-1">
|
|
<dl>
|
|
<dt class="text-sm font-medium text-gray-500 truncate">Completed Services</dt>
|
|
<dd class="text-lg font-medium text-gray-900">{{ $stats['completed_services'] }}</dd>
|
|
</dl>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
|
|
<!-- Active Job Cards -->
|
|
<div class="bg-white rounded-lg shadow">
|
|
<div class="px-6 py-4 border-b border-gray-200">
|
|
<h2 class="text-lg font-semibold text-gray-900">Active Services</h2>
|
|
</div>
|
|
<div class="p-6">
|
|
@if($activeJobCards->count() > 0)
|
|
<div class="space-y-4">
|
|
@foreach($activeJobCards as $jobCard)
|
|
<div class="border rounded-lg p-4 hover:bg-gray-50">
|
|
<div class="flex justify-between items-start">
|
|
<div>
|
|
<h3 class="font-medium text-gray-900">Job Card #{{ $jobCard->id }}</h3>
|
|
<p class="text-sm text-gray-600">
|
|
{{ $jobCard->vehicle->year ?? '' }} {{ $jobCard->vehicle->make ?? '' }} {{ $jobCard->vehicle->model ?? '' }}
|
|
</p>
|
|
@if($jobCard->serviceAdvisor)
|
|
<p class="text-xs text-gray-500">Advisor: {{ $jobCard->serviceAdvisor->name }}</p>
|
|
@endif
|
|
</div>
|
|
<div class="text-right">
|
|
<span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium
|
|
@switch($jobCard->status)
|
|
@case('pending')
|
|
bg-yellow-100 text-yellow-800
|
|
@break
|
|
@case('in_progress')
|
|
bg-blue-100 text-blue-800
|
|
@break
|
|
@case('estimate_sent')
|
|
bg-orange-100 text-orange-800
|
|
@break
|
|
@default
|
|
bg-gray-100 text-gray-800
|
|
@endswitch
|
|
">
|
|
{{ ucfirst(str_replace('_', ' ', $jobCard->status)) }}
|
|
</span>
|
|
<a href="{{ route('customer-portal.status', $jobCard) }}"
|
|
class="block mt-2 text-sm text-blue-600 hover:text-blue-800">
|
|
View Details
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
<div class="mt-4 text-center">
|
|
<a href="{{ route('customer-portal.work-orders') }}"
|
|
class="text-blue-600 hover:text-blue-800 text-sm font-medium">
|
|
View All Work Orders →
|
|
</a>
|
|
</div>
|
|
@else
|
|
<div class="text-center py-8">
|
|
<svg class="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19.428 15.428a2 2 0 00-1.022-.547l-2.387-.477a6 6 0 00-3.86.517l-.318.158a6 6 0 01-3.86.517L6.05 15.21a2 2 0 00-1.806.547M8 4h8l-1 1v5.172a2 2 0 00.586 1.414l5 5c1.26 1.26.367 3.414-1.415 3.414H4.828c-1.782 0-2.674-2.154-1.414-3.414l5-5A2 2 0 009 11.172V5l-1-1z"></path>
|
|
</svg>
|
|
<h3 class="mt-2 text-sm font-medium text-gray-900">No active services</h3>
|
|
<p class="mt-1 text-sm text-gray-500">You don't have any vehicles currently being serviced.</p>
|
|
</div>
|
|
@endif
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Recent Activity -->
|
|
<div class="bg-white rounded-lg shadow">
|
|
<div class="px-6 py-4 border-b border-gray-200">
|
|
<h2 class="text-lg font-semibold text-gray-900">Recent Activity</h2>
|
|
</div>
|
|
<div class="p-6">
|
|
@if($recentActivity->count() > 0)
|
|
<div class="space-y-4">
|
|
@foreach($recentActivity as $activity)
|
|
<div class="flex space-x-3">
|
|
<div class="flex-shrink-0">
|
|
@if($activity['type'] === 'job_card')
|
|
<div class="h-8 w-8 rounded-full bg-blue-100 flex items-center justify-center">
|
|
<svg class="h-4 w-4 text-blue-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19.428 15.428a2 2 0 00-1.022-.547l-2.387-.477a6 6 0 00-3.86.517l-.318.158a6 6 0 01-3.86.517L6.05 15.21a2 2 0 00-1.806.547M8 4h8l-1 1v5.172a2 2 0 00.586 1.414l5 5c1.26 1.26.367 3.414-1.415 3.414H4.828c-1.782 0-2.674-2.154-1.414-3.414l5-5A2 2 0 009 11.172V5l-1-1z"></path>
|
|
</svg>
|
|
</div>
|
|
@else
|
|
<div class="h-8 w-8 rounded-full bg-green-100 flex items-center justify-center">
|
|
<svg class="h-4 w-4 text-green-600" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"></path>
|
|
</svg>
|
|
</div>
|
|
@endif
|
|
</div>
|
|
<div class="min-w-0 flex-1">
|
|
<p class="text-sm font-medium text-gray-900">
|
|
<a href="{{ $activity['url'] }}" class="hover:underline">
|
|
{{ $activity['title'] }}
|
|
</a>
|
|
</p>
|
|
<p class="text-sm text-gray-500">{{ $activity['description'] }}</p>
|
|
<p class="text-xs text-gray-400">{{ $activity['date']->diffForHumans() }}</p>
|
|
</div>
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
@else
|
|
<div class="text-center py-8">
|
|
<svg class="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"></path>
|
|
</svg>
|
|
<h3 class="mt-2 text-sm font-medium text-gray-900">No recent activity</h3>
|
|
<p class="mt-1 text-sm text-gray-500">Your activity will appear here once you start using our services.</p>
|
|
</div>
|
|
@endif
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Upcoming Appointments -->
|
|
@if($upcomingAppointments->count() > 0)
|
|
<div class="mt-8 bg-white rounded-lg shadow">
|
|
<div class="px-6 py-4 border-b border-gray-200">
|
|
<h2 class="text-lg font-semibold text-gray-900">Upcoming Appointments</h2>
|
|
</div>
|
|
<div class="p-6">
|
|
<div class="space-y-4">
|
|
@foreach($upcomingAppointments as $appointment)
|
|
<div class="flex justify-between items-center border-l-4 border-blue-400 pl-4">
|
|
<div>
|
|
<h3 class="font-medium text-gray-900">{{ $appointment->service_type }}</h3>
|
|
<p class="text-sm text-gray-600">{{ $appointment->scheduled_datetime->format('M j, Y g:i A') }}</p>
|
|
@if($appointment->notes)
|
|
<p class="text-sm text-gray-500">{{ $appointment->notes }}</p>
|
|
@endif
|
|
</div>
|
|
<span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium
|
|
@switch($appointment->status)
|
|
@case('confirmed')
|
|
bg-green-100 text-green-800
|
|
@break
|
|
@case('pending')
|
|
bg-yellow-100 text-yellow-800
|
|
@break
|
|
@default
|
|
bg-gray-100 text-gray-800
|
|
@endswitch
|
|
">
|
|
{{ ucfirst($appointment->status) }}
|
|
</span>
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
<div class="mt-4 text-center">
|
|
<a href="{{ route('customer-portal.appointments') }}"
|
|
class="text-blue-600 hover:text-blue-800 text-sm font-medium">
|
|
View All Appointments →
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
@endif
|
|
</div>
|