Files
innovatieplatform/app/Services/CommitmentService.php
znetsixe f0aca26642 Sprint 2: Live data, CRUD modals, commitments, document upload
Frontend:
- Connect MetroMap to live Inertia props (replace hardcoded demo data)
- Drill-down navigation via router.visit for project-level maps
- Reactive breadcrumb based on map level
- Empty state when no projects exist
- Reusable Modal component with retro styling
- ProjectForm and CommitmentForm with Inertia useForm
- FormInput reusable component (text, date, textarea, select)
- FloatingActions FAB button for creating projects/themes

Backend:
- CommitmentService + CommitmentController (CRUD, mark complete, overdue)
- DocumentService + DocumentController (upload, download, delete)
- MapController now passes users and speerpunten to frontend
- 7 new routes (4 commitment, 3 document)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-01 16:02:38 +02:00

104 lines
3.1 KiB
PHP

<?php
namespace App\Services;
use App\Enums\CommitmentStatus;
use App\Models\AuditLog;
use App\Models\Commitment;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\DB;
class CommitmentService
{
/**
* Get all commitments for a project with owner and acties.
*/
public function getForProject(int $projectId): Collection
{
return Commitment::with(['eigenaar', 'acties'])
->where('project_id', $projectId)
->orderBy('deadline')
->get();
}
/**
* Create a new commitment with audit log.
*/
public function create(array $data): Commitment
{
return DB::transaction(function () use ($data) {
$commitment = Commitment::create([
'project_id' => $data['project_id'],
'beschrijving' => $data['beschrijving'],
'eigenaar_id' => $data['eigenaar_id'],
'deadline' => $data['deadline'],
'status' => CommitmentStatus::Open,
'bron' => $data['bron'] ?? null,
'besluit_id' => $data['besluit_id'] ?? null,
]);
$this->audit('created', $commitment);
return $commitment;
});
}
/**
* Update a commitment with audit log.
*/
public function update(Commitment $commitment, array $data): Commitment
{
return DB::transaction(function () use ($commitment, $data) {
$commitment->update(array_filter([
'beschrijving' => $data['beschrijving'] ?? null,
'eigenaar_id' => $data['eigenaar_id'] ?? null,
'deadline' => $data['deadline'] ?? null,
'status' => $data['status'] ?? null,
'bron' => $data['bron'] ?? null,
], fn ($v) => $v !== null));
$this->audit('updated', $commitment);
return $commitment->fresh();
});
}
/**
* Mark a commitment as afgerond with audit log.
*/
public function markComplete(Commitment $commitment): Commitment
{
return DB::transaction(function () use ($commitment) {
$commitment->update(['status' => CommitmentStatus::Afgerond]);
$this->audit('completed', $commitment);
return $commitment->fresh();
});
}
/**
* Get all commitments past their deadline that are not afgerond.
*/
public function getOverdue(): Collection
{
return Commitment::with(['eigenaar', 'project'])
->where('deadline', '<', now()->toDateString())
->where('status', '!=', CommitmentStatus::Afgerond)
->orderBy('deadline')
->get();
}
private function audit(string $action, Commitment $commitment, ?array $extra = null): void
{
AuditLog::create([
'user_id' => Auth::id(),
'action' => "commitment.{$action}",
'entity_type' => 'commitment',
'entity_id' => $commitment->id,
'payload' => $extra,
]);
}
}