sackey cbae4564b9
Some checks failed
linter / quality (push) Has been cancelled
tests / ci (push) Has been cancelled
Add customer portal views for dashboard, estimates, invoices, vehicles, and work orders
- 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.
2025-08-08 09:56:26 +00:00

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>