Files
bhoza-shift-manager/src/app/Models/Shift.php
Mulanga 961f288d97 feat: add Shift Planning module with full CRUD, attendance, and reporting
- Add shifts, shift_staff, and shift_attendance migrations
- Add Shift, ShiftStaff, ShiftAttendance Eloquent models with auditing
- Add ShiftService with business logic (create, start, complete, assign staff, mark attendance, reports, timesheets)
- Add ShiftResource with list, create, edit, and attendance management pages
- Add staff and attendance relation managers
- Add standalone pages: ActiveShifts, StaffManagement, ShiftReports, Timesheets
- Add dashboard widgets: ShiftOverview stats, TodaysShifts table
- Add ShiftPolicy for role-based authorization
- Add 10 shift permissions and manager role to RolePermissionSeeder
- Update User model with shift relationships
- Fix audits table migration with required columns for owen-it/laravel-auditing
- Update DatabaseSeeder to create admin user and call RolePermissionSeeder
- Switch docker-compose to MariaDB 10.11 for Docker Desktop compatibility
2026-03-10 09:44:17 +02:00

140 lines
3.2 KiB
PHP

<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
use Illuminate\Database\Eloquent\Relations\HasMany;
use OwenIt\Auditing\Contracts\Auditable;
class Shift extends Model implements Auditable
{
use HasFactory;
use \OwenIt\Auditing\Auditable;
protected $fillable = [
'title',
'shift_date',
'planned_start_time',
'planned_end_time',
'actual_start_time',
'actual_end_time',
'status',
'notes',
'created_by',
'started_by',
];
protected function casts(): array
{
return [
'shift_date' => 'date',
'actual_start_time' => 'datetime',
'actual_end_time' => 'datetime',
];
}
// --- Relationships ---
public function creator(): BelongsTo
{
return $this->belongsTo(User::class, 'created_by');
}
public function starter(): BelongsTo
{
return $this->belongsTo(User::class, 'started_by');
}
public function staff(): BelongsToMany
{
return $this->belongsToMany(User::class, 'shift_staff')
->withPivot('assigned_by')
->withTimestamps();
}
public function shiftStaff(): HasMany
{
return $this->hasMany(ShiftStaff::class);
}
public function attendances(): HasMany
{
return $this->hasMany(ShiftAttendance::class);
}
// --- State Checks ---
public function isPlanned(): bool
{
return $this->status === 'planned';
}
public function isInProgress(): bool
{
return $this->status === 'in_progress';
}
public function isCompleted(): bool
{
return $this->status === 'completed';
}
// --- Computed Attributes ---
public function getPlannedDurationHoursAttribute(): float
{
$start = \Carbon\Carbon::parse($this->planned_start_time);
$end = \Carbon\Carbon::parse($this->planned_end_time);
if ($end->lt($start)) {
$end->addDay();
}
return round($start->diffInMinutes($end) / 60, 2);
}
public function getActualDurationHoursAttribute(): ?float
{
if (!$this->actual_start_time || !$this->actual_end_time) {
return null;
}
return round($this->actual_start_time->diffInMinutes($this->actual_end_time) / 60, 2);
}
// --- Scopes ---
public function scopePlanned($query)
{
return $query->where('status', 'planned');
}
public function scopeInProgress($query)
{
return $query->where('status', 'in_progress');
}
public function scopeCompleted($query)
{
return $query->where('status', 'completed');
}
public function scopeForDate($query, $date)
{
return $query->where('shift_date', $date);
}
public function scopeForDateRange($query, $from, $to)
{
return $query->whereBetween('shift_date', [$from, $to]);
}
public function scopeForStaff($query, $userId)
{
return $query->whereHas('staff', fn ($q) => $q->where('users.id', $userId));
}
}