Initial scaffold : Laravel 12 + PostgreSQL + auth + domaine métier (étapes 1-5)

- Laravel 12 sur PHP 8.5, Breeze (Blade/Tailwind/Alpine.js)
- Docker Compose dev (PostgreSQL 18 + Redis) et prod (stack complète + nginx)
- Migrations et models : lieux, sections, dépôts, source_types/fields, sources, relevés
  - Colonne JSONB data sur releves avec colonnes générées indexées (nom, prenom, date_evenement)
  - Index GIN pour la recherche fulltext
- Enums : UserRole, SourceStatus (avec transitions), CalendarType, FieldType
- RoleMiddleware (alias `role`) + helpers isAdmin/isSectionManager sur User
- CRUD Lieux (arbre hiérarchique, calcul nom_long en cascade)
- CRUD admin : Sections (+ gestion membres), Dépôts, Types de sources (+ champs dynamiques, drag & drop)
- CRUD Sources : visibilité filtrée par rôle, assignation membres, workflow de statut

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-06-04 16:16:37 +02:00
commit 7609d35287
172 changed files with 19179 additions and 0 deletions
@@ -0,0 +1,58 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Http\Requests\Admin\StoreDepotRequest;
use App\Models\Depot;
use Illuminate\Http\RedirectResponse;
use Illuminate\View\View;
class DepotController extends Controller
{
public function index(): View
{
$depots = Depot::withCount('sources')->orderBy('nom')->paginate(25);
return view('admin.depots.index', compact('depots'));
}
public function create(): View
{
return view('admin.depots.create');
}
public function store(StoreDepotRequest $request): RedirectResponse
{
$depot = Depot::create($request->validated());
return redirect()->route('admin.depots.show', $depot)
->with('success', 'Dépôt créé.');
}
public function show(Depot $depot): View
{
$depot->load('sources');
return view('admin.depots.show', compact('depot'));
}
public function edit(Depot $depot): View
{
return view('admin.depots.edit', compact('depot'));
}
public function update(StoreDepotRequest $request, Depot $depot): RedirectResponse
{
$depot->update($request->validated());
return redirect()->route('admin.depots.show', $depot)
->with('success', 'Dépôt mis à jour.');
}
public function destroy(Depot $depot): RedirectResponse
{
if ($depot->sources()->exists()) {
return back()->with('error', 'Impossible de supprimer un dépôt qui contient des sources.');
}
$depot->delete();
return redirect()->route('admin.depots.index')
->with('success', 'Dépôt supprimé.');
}
}
@@ -0,0 +1,82 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Http\Requests\Admin\StoreSectionRequest;
use App\Http\Requests\Admin\UpdateSectionRequest;
use App\Models\Lieu;
use App\Models\Section;
use App\Models\User;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\View\View;
class SectionController extends Controller
{
public function index(): View
{
$sections = Section::with('lieu')->orderBy('nom')->paginate(25);
return view('admin.sections.index', compact('sections'));
}
public function create(): View
{
$lieux = Lieu::orderBy('nom_long')->get(['id', 'nom_long', 'nom']);
return view('admin.sections.create', compact('lieux'));
}
public function store(StoreSectionRequest $request): RedirectResponse
{
$section = Section::create($request->validated());
return redirect()->route('admin.sections.show', $section)
->with('success', 'Section créée.');
}
public function show(Section $section): View
{
$section->load('lieu', 'membres');
$users = User::orderBy('name')->get(['id', 'name', 'email']);
return view('admin.sections.show', compact('section', 'users'));
}
public function edit(Section $section): View
{
$lieux = Lieu::orderBy('nom_long')->get(['id', 'nom_long', 'nom']);
return view('admin.sections.edit', compact('section', 'lieux'));
}
public function update(UpdateSectionRequest $request, Section $section): RedirectResponse
{
$section->update($request->validated());
return redirect()->route('admin.sections.show', $section)
->with('success', 'Section mise à jour.');
}
public function destroy(Section $section): RedirectResponse
{
$section->delete();
return redirect()->route('admin.sections.index')
->with('success', 'Section supprimée.');
}
public function addMembre(Request $request, Section $section): RedirectResponse
{
$data = $request->validate([
'user_id' => ['required', 'exists:users,id'],
'role_in_section' => ['required', 'in:member,section_manager'],
]);
$section->membres()->syncWithoutDetaching([
$data['user_id'] => ['role_in_section' => $data['role_in_section']],
]);
return back()->with('success', 'Membre ajouté.');
}
public function removeMembre(Section $section, User $user): RedirectResponse
{
$section->membres()->detach($user->id);
return back()->with('success', 'Membre retiré.');
}
}
@@ -0,0 +1,105 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Http\Requests\Admin\StoreSourceTypeFieldRequest;
use App\Http\Requests\Admin\StoreSourceTypeRequest;
use App\Models\SourceType;
use App\Models\SourceTypeField;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\View\View;
class SourceTypeController extends Controller
{
public function index(): View
{
$sourceTypes = SourceType::withCount('sources')->orderBy('nom')->paginate(25);
return view('admin.source-types.index', compact('sourceTypes'));
}
public function create(): View
{
return view('admin.source-types.create');
}
public function store(StoreSourceTypeRequest $request): RedirectResponse
{
$sourceType = SourceType::create($request->validated());
return redirect()->route('admin.source-types.show', $sourceType)
->with('success', 'Type de source créé. Ajoutez maintenant ses champs.');
}
public function show(SourceType $sourceType): View
{
$sourceType->load('fields');
return view('admin.source-types.show', compact('sourceType'));
}
public function edit(SourceType $sourceType): View
{
return view('admin.source-types.edit', compact('sourceType'));
}
public function update(StoreSourceTypeRequest $request, SourceType $sourceType): RedirectResponse
{
$sourceType->update($request->validated());
return redirect()->route('admin.source-types.show', $sourceType)
->with('success', 'Type de source mis à jour.');
}
public function destroy(SourceType $sourceType): RedirectResponse
{
if ($sourceType->sources()->exists()) {
return back()->with('error', 'Impossible de supprimer un type utilisé par des sources.');
}
$sourceType->delete();
return redirect()->route('admin.source-types.index')
->with('success', 'Type de source supprimé.');
}
public function storeField(StoreSourceTypeFieldRequest $request, SourceType $sourceType): RedirectResponse
{
$maxOrder = $sourceType->fields()->max('order') ?? -1;
$data = $request->validated();
if ($request->filled('options_raw')) {
$data['options'] = array_filter(array_map('trim', explode("\n", $request->input('options_raw'))));
}
$sourceType->fields()->create([...$data, 'order' => $maxOrder + 1]);
return back()->with('success', 'Champ ajouté.');
}
public function updateField(StoreSourceTypeFieldRequest $request, SourceType $sourceType, SourceTypeField $field): RedirectResponse
{
abort_if($field->source_type_id !== $sourceType->id, 404);
$field->update($request->validated());
return back()->with('success', 'Champ mis à jour.');
}
public function destroyField(SourceType $sourceType, SourceTypeField $field): RedirectResponse
{
abort_if($field->source_type_id !== $sourceType->id, 404);
if ($sourceType->sources()->exists()) {
return back()->with('error', 'Impossible de supprimer un champ d\'un type déjà utilisé par des sources (les relevés existants conservent leurs données).');
}
$field->delete();
return back()->with('success', 'Champ supprimé.');
}
public function reorderFields(Request $request, SourceType $sourceType): RedirectResponse
{
$ordered = $request->validate(['ids' => ['required', 'array'], 'ids.*' => ['integer']])['ids'];
foreach ($ordered as $position => $id) {
$sourceType->fields()->where('id', $id)->update(['order' => $position]);
}
return back()->with('success', 'Ordre mis à jour.');
}
}