sackey e3b2b220d2
Some checks are pending
linter / quality (push) Waiting to run
tests / ci (push) Waiting to run
Enhance UI and functionality across various components
- Increased icon sizes in service items, service orders, users, and technician management for better visibility.
- Added custom loading indicators with appropriate icons in search fields for vehicles, work orders, and technicians.
- Introduced invoice management routes for better organization and access control.
- Created a new test for the estimate PDF functionality to ensure proper rendering and data integrity.
2025-08-16 14:36:58 +00:00

287 lines
18 KiB
PHP

<div class="max-w-7xl mx-auto py-6 px-4 sm:px-6 lg:px-8">
<!-- Page Header -->
<div class="mb-8">
<div class="flex items-center justify-between">
<div>
<h1 class="text-3xl font-bold tracking-tight text-zinc-900 dark:text-zinc-100">Service Items Management</h1>
<p class="mt-2 text-sm text-zinc-600 dark:text-zinc-400">
Manage labor operations and service items for diagnosis and repairs
</p>
</div>
<button wire:click="toggleForm"
class="inline-flex items-center px-4 py-2 bg-blue-600 hover:bg-blue-700 text-white font-medium rounded-lg transition-colors">
<svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4"/>
</svg>
{{ $showForm ? 'Cancel' : 'Add Service Item' }}
</button>
</div>
</div>
<!-- Flash Messages -->
@if (session()->has('message'))
<div class="mb-6 bg-green-50 dark:bg-green-900/20 border border-green-200 dark:border-green-800 rounded-lg p-4">
<div class="flex">
<svg class="h-5 w-5 text-green-400" 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"/>
</svg>
<div class="ml-3">
<p class="text-sm font-medium text-green-800 dark:text-green-200">{{ session('message') }}</p>
</div>
</div>
</div>
@endif
<!-- Add/Edit Form -->
@if($showForm)
<div class="mb-8 bg-white dark:bg-zinc-800 rounded-xl border border-zinc-200 dark:border-zinc-700 shadow-sm">
<div class="px-6 py-4 border-b border-zinc-200 dark:border-zinc-700">
<h2 class="text-lg font-semibold text-zinc-900 dark:text-zinc-100">
{{ $editingId ? 'Edit Service Item' : 'Add New Service Item' }}
</h2>
</div>
<form wire:submit.prevent="save" class="p-6">
<div class="grid grid-cols-1 md:grid-cols-2 gap-6">
<!-- Service Name -->
<div>
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Service Name *
</label>
<input type="text"
wire:model="service_name"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500"
placeholder="e.g., Oil Change, Brake Inspection">
@error('service_name')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
<!-- Category -->
<div>
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Category *
</label>
<select wire:model="category"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500">
<option value="">Select Category</option>
@foreach($categories as $key => $label)
<option value="{{ $key }}">{{ $label }}</option>
@endforeach
</select>
@error('category')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
<!-- Labor Rate -->
<div>
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Labor Rate ($/hour) *
</label>
<div class="relative">
<span class="absolute left-3 top-1/2 transform -translate-y-1/2 text-zinc-500">$</span>
<input type="number"
wire:model="labor_rate"
step="0.01"
min="0"
max="500"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500 pl-8"
placeholder="85.00">
</div>
@error('labor_rate')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
<!-- Estimated Hours -->
<div>
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Estimated Hours *
</label>
<input type="number"
wire:model="estimated_hours"
step="0.25"
min="0.25"
max="40"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500"
placeholder="1.0">
@error('estimated_hours')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
<!-- Status -->
<div>
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Status *
</label>
<select wire:model="status"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500">
<option value="active">Active</option>
<option value="inactive">Inactive</option>
</select>
@error('status')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
<!-- Description -->
<div class="md:col-span-2">
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Description
</label>
<textarea wire:model="description"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500"
rows="3"
placeholder="Detailed description of the service..."></textarea>
@error('description')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
<!-- Technician Notes -->
<div class="md:col-span-2">
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">
Technician Notes
</label>
<textarea wire:model="technician_notes"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500"
rows="2"
placeholder="Special instructions or notes for technicians..."></textarea>
@error('technician_notes')
<p class="mt-1 text-sm text-red-600 dark:text-red-400">{{ $message }}</p>
@enderror
</div>
</div>
<!-- Form Actions -->
<div class="mt-6 flex items-center justify-end space-x-3">
<button type="button"
wire:click="toggleForm"
class="px-4 py-2 text-sm font-medium text-zinc-700 dark:text-zinc-300 bg-white dark:bg-zinc-800 border border-zinc-300 dark:border-zinc-600 rounded-lg hover:bg-zinc-50 dark:hover:bg-zinc-700 transition-colors">
Cancel
</button>
<button type="submit"
class="px-4 py-2 text-sm font-medium text-white bg-blue-600 hover:bg-blue-700 rounded-lg transition-colors">
{{ $editingId ? 'Update Service Item' : 'Create Service Item' }}
</button>
</div>
</form>
</div>
@endif
<!-- Search and Filters -->
<div class="mb-6 bg-white dark:bg-zinc-800 rounded-xl border border-zinc-200 dark:border-zinc-700 shadow-sm p-6">
<div class="grid grid-cols-1 md:grid-cols-3 gap-4">
<div class="md:col-span-2">
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">Search Service Items</label>
<input type="text"
wire:model.live.debounce.300ms="searchTerm"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500"
placeholder="Search by service name or description...">
</div>
<div>
<label class="block text-sm font-medium text-zinc-700 dark:text-zinc-300 mb-2">Filter by Category</label>
<select wire:model.live="categoryFilter"
class="w-full rounded-lg border-zinc-300 dark:border-zinc-600 bg-white dark:bg-zinc-900 text-zinc-900 dark:text-zinc-100 shadow-sm focus:border-blue-500 focus:ring-blue-500">
<option value="">All Categories</option>
@foreach($categories as $key => $label)
<option value="{{ $key }}">{{ $label }}</option>
@endforeach
</select>
</div>
</div>
</div>
<!-- Service Items Table -->
<div class="bg-white dark:bg-zinc-800 rounded-xl border border-zinc-200 dark:border-zinc-700 shadow-sm overflow-hidden">
<div class="px-6 py-4 border-b border-zinc-200 dark:border-zinc-700">
<h3 class="text-lg font-semibold text-zinc-900 dark:text-zinc-100">Service Items</h3>
</div>
@if($serviceItems->count() > 0)
<div class="overflow-x-auto">
<table class="min-w-full divide-y divide-zinc-200 dark:divide-zinc-700">
<thead class="bg-zinc-50 dark:bg-zinc-900">
<tr>
<th class="px-6 py-3 text-left text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Service</th>
<th class="px-6 py-3 text-left text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Category</th>
<th class="px-6 py-3 text-left text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Labor Rate</th>
<th class="px-6 py-3 text-left text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Est. Hours</th>
<th class="px-6 py-3 text-left text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Est. Cost</th>
<th class="px-6 py-3 text-left text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Status</th>
<th class="px-6 py-3 text-right text-xs font-medium text-zinc-500 dark:text-zinc-400 uppercase tracking-wider">Actions</th>
</tr>
</thead>
<tbody class="bg-white dark:bg-zinc-800 divide-y divide-zinc-200 dark:divide-zinc-700">
@foreach($serviceItems as $item)
<tr class="hover:bg-zinc-50 dark:hover:bg-zinc-700/50">
<td class="px-6 py-4">
<div>
<div class="text-sm font-medium text-zinc-900 dark:text-zinc-100">{{ $item->service_name }}</div>
@if($item->description)
<div class="text-sm text-zinc-500 dark:text-zinc-400">{{ Str::limit($item->description, 60) }}</div>
@endif
</div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<span class="inline-flex px-2 py-1 text-xs font-semibold rounded-full bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200">
{{ $item->category }}
</span>
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-zinc-900 dark:text-zinc-100">
${{ number_format($item->labor_rate, 2) }}/hr
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-zinc-900 dark:text-zinc-100">
{{ $item->estimated_hours }}h
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium text-green-600 dark:text-green-400">
${{ number_format($item->estimated_hours * $item->labor_rate, 2) }}
</td>
<td class="px-6 py-4 whitespace-nowrap">
<span class="inline-flex px-2 py-1 text-xs font-semibold rounded-full
{{ $item->status === 'active' ? 'bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200' : 'bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-200' }}">
{{ ucfirst($item->status) }}
</span>
</td>
<td class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
<div class="flex items-center justify-end space-x-2">
<button wire:click="edit({{ $item->id }})"
class="text-blue-600 hover:text-blue-900 dark:text-blue-400 dark:hover:text-blue-300">
Edit
</button>
<button wire:click="delete({{ $item->id }})"
wire:confirm="Are you sure you want to delete this service item?"
class="text-red-600 hover:text-red-900 dark:text-red-400 dark:hover:text-red-300">
Delete
</button>
</div>
</td>
</tr>
@endforeach
</tbody>
</table>
</div>
<!-- Pagination -->
<div class="px-6 py-4 border-t border-zinc-200 dark:border-zinc-700">
{{ $serviceItems->links() }}
</div>
@else
<div class="px-6 py-12 text-center">
<svg class="mx-auto h-12 w-12 text-zinc-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 10.172V5L8 4z"/>
</svg>
<h3 class="mt-2 text-sm font-medium text-zinc-900 dark:text-zinc-100">No service items found</h3>
<p class="mt-1 text-sm text-zinc-500 dark:text-zinc-400">
@if($searchTerm || $categoryFilter)
Try adjusting your search criteria.
@else
Get started by creating your first service item.
@endif
</p>
</div>
@endif
</div>
</div>