Étapes 6-9 + types de lieux + picker + filtres
- Étape 6 : formulaire de saisie dynamique des relevés (piloté par source_type_fields, calendriers grégorien/julien/républicain) - Étape 7 : workflow de statut des sources + notifications mail+DB (SourceAValider, SourceRejetee) - Étape 8 : recherche fulltext PostgreSQL avec filtres type/lieu/années et CTE récursive pour les subdivisions de lieux - Étape 9 : export GEDCOM 5.5.1 (GedcomExportService + DateConversionService) - Types de lieux : CRUD admin (LieuTypeController) avec champ ordre - Composant lieu-picker : modale Alpine.js avec recherche AJAX + debounce - Filtres sources : statut, type, lieu (CTE récursive), période annee_debut/annee_fin - Filtres lieux : type, texte, lieu parent avec descendants (CTE récursive) - Migration : lieu_id + annee_debut + annee_fin sur sources Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -5,30 +5,77 @@ namespace App\Http\Controllers;
|
||||
use App\Http\Requests\StoreLieuRequest;
|
||||
use App\Http\Requests\UpdateLieuRequest;
|
||||
use App\Models\Lieu;
|
||||
use App\Models\LieuType;
|
||||
use Illuminate\Http\JsonResponse;
|
||||
use Illuminate\Http\RedirectResponse;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\View\View;
|
||||
|
||||
class LieuController extends Controller
|
||||
{
|
||||
public function index(): View
|
||||
public function search(Request $request): JsonResponse
|
||||
{
|
||||
$q = trim($request->get('q', ''));
|
||||
|
||||
$lieux = Lieu::with('lieuType')
|
||||
->where(function ($query) use ($q) {
|
||||
$query->where('nom_long', 'ilike', "%{$q}%")
|
||||
->orWhere('nom', 'ilike', "%{$q}%")
|
||||
->orWhere('code', 'ilike', "%{$q}%");
|
||||
})
|
||||
->orderBy('nom_long')
|
||||
->limit(25)
|
||||
->get();
|
||||
|
||||
return response()->json($lieux->map(fn ($l) => [
|
||||
'id' => $l->id,
|
||||
'nom_long' => $l->nom_long ?? $l->nom,
|
||||
'code' => $l->code,
|
||||
'type' => $l->lieuType?->nom,
|
||||
]));
|
||||
}
|
||||
|
||||
public function index(Request $request): View
|
||||
{
|
||||
$this->authorize('viewAny', Lieu::class);
|
||||
|
||||
// Arbre complet trié par nom_long pour l'affichage
|
||||
$lieux = Lieu::with('parent')
|
||||
->orderBy('nom_long')
|
||||
->paginate(50);
|
||||
$query = Lieu::with(['parent', 'lieuType'])->orderBy('nom_long');
|
||||
|
||||
return view('lieux.index', compact('lieux'));
|
||||
if ($request->filled('lieu_type_id')) {
|
||||
$query->where('lieu_type_id', $request->integer('lieu_type_id'));
|
||||
}
|
||||
|
||||
if ($request->filled('q')) {
|
||||
$q = trim($request->get('q'));
|
||||
$query->where(function ($wq) use ($q) {
|
||||
$wq->where('nom_long', 'ilike', "%{$q}%")
|
||||
->orWhere('nom', 'ilike', "%{$q}%")
|
||||
->orWhere('code', 'ilike', "%{$q}%");
|
||||
});
|
||||
}
|
||||
|
||||
if ($request->filled('lieu_id')) {
|
||||
$ids = $this->getDescendantAndSelfIds($request->integer('lieu_id'));
|
||||
$query->whereIn('id', $ids);
|
||||
}
|
||||
|
||||
$lieuTypes = LieuType::orderBy('ordre')->get(['id', 'nom']);
|
||||
$lieuSelectionne = $request->filled('lieu_id')
|
||||
? Lieu::find($request->integer('lieu_id'), ['id', 'nom', 'nom_long'])
|
||||
: null;
|
||||
$lieux = $query->paginate(50)->withQueryString();
|
||||
|
||||
return view('lieux.index', compact('lieux', 'lieuTypes', 'lieuSelectionne'));
|
||||
}
|
||||
|
||||
public function create(): View
|
||||
{
|
||||
$this->authorize('create', Lieu::class);
|
||||
|
||||
$parents = Lieu::orderBy('nom_long')->get(['id', 'nom_long']);
|
||||
$lieuTypes = LieuType::orderBy('ordre')->get(['id', 'nom']);
|
||||
|
||||
return view('lieux.create', compact('parents'));
|
||||
return view('lieux.create', compact('lieuTypes'));
|
||||
}
|
||||
|
||||
public function store(StoreLieuRequest $request): RedirectResponse
|
||||
@@ -43,7 +90,7 @@ class LieuController extends Controller
|
||||
{
|
||||
$this->authorize('view', $lieu);
|
||||
|
||||
$lieu->load('parent', 'enfants');
|
||||
$lieu->load('parent', 'enfants', 'lieuType');
|
||||
|
||||
return view('lieux.show', compact('lieu'));
|
||||
}
|
||||
@@ -52,20 +99,15 @@ class LieuController extends Controller
|
||||
{
|
||||
$this->authorize('update', $lieu);
|
||||
|
||||
// Exclure le lieu lui-même et ses descendants pour éviter les cycles
|
||||
$descendants = $this->getDescendantIds($lieu);
|
||||
$parents = Lieu::whereNotIn('id', [...$descendants, $lieu->id])
|
||||
->orderBy('nom_long')
|
||||
->get(['id', 'nom_long']);
|
||||
$lieu->load('parent', 'lieuType');
|
||||
$lieuTypes = LieuType::orderBy('ordre')->get(['id', 'nom']);
|
||||
|
||||
return view('lieux.edit', compact('lieu', 'parents'));
|
||||
return view('lieux.edit', compact('lieu', 'lieuTypes'));
|
||||
}
|
||||
|
||||
public function update(UpdateLieuRequest $request, Lieu $lieu): RedirectResponse
|
||||
{
|
||||
$lieu->update($request->validated());
|
||||
|
||||
// Recalculer nom_long des enfants en cascade
|
||||
$this->recalculerEnfants($lieu);
|
||||
|
||||
return redirect()->route('lieux.show', $lieu)
|
||||
@@ -86,12 +128,27 @@ class LieuController extends Controller
|
||||
->with('success', 'Lieu supprimé.');
|
||||
}
|
||||
|
||||
private function getDescendantAndSelfIds(int $lieuId): array
|
||||
{
|
||||
$rows = DB::select("
|
||||
WITH RECURSIVE descendants AS (
|
||||
SELECT id FROM lieux WHERE id = ?
|
||||
UNION ALL
|
||||
SELECT l.id FROM lieux l
|
||||
INNER JOIN descendants d ON l.lieu_parent_id = d.id
|
||||
)
|
||||
SELECT id FROM descendants
|
||||
", [$lieuId]);
|
||||
|
||||
return collect($rows)->pluck('id')->toArray();
|
||||
}
|
||||
|
||||
private function getDescendantIds(Lieu $lieu): array
|
||||
{
|
||||
$ids = [];
|
||||
foreach ($lieu->enfants as $enfant) {
|
||||
$ids[] = $enfant->id;
|
||||
$ids = array_merge($ids, $this->getDescendantIds($enfant));
|
||||
$ids = array_merge($ids, $this->getDescendantIds($enfant));
|
||||
}
|
||||
return $ids;
|
||||
}
|
||||
@@ -100,7 +157,7 @@ class LieuController extends Controller
|
||||
{
|
||||
$lieu->load('enfants');
|
||||
foreach ($lieu->enfants as $enfant) {
|
||||
$enfant->update([]); // déclenche le booted() hook qui recalcule nom_long
|
||||
$enfant->update([]);
|
||||
$this->recalculerEnfants($enfant);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user