input('search')) { $query->where(function ($q) use ($search) { $q->where('title', 'like', "%{$search}%") ->orWhere('ccli_id', 'like', "%{$search}%"); }); } $songs = $query->orderBy('title') ->paginate($request->input('per_page', 20)); return response()->json([ 'data' => $songs->map(fn (Song $song) => [ 'id' => $song->id, 'title' => $song->title, 'ccli_id' => $song->ccli_id, 'author' => $song->author, 'has_translation' => $song->has_translation, 'last_used_at' => $song->last_used_at?->toDateString(), 'last_used_in_service' => $song->last_used_in_service, 'created_at' => $song->created_at->toDateTimeString(), 'updated_at' => $song->updated_at->toDateTimeString(), ]), 'meta' => [ 'current_page' => $songs->currentPage(), 'last_page' => $songs->lastPage(), 'per_page' => $songs->perPage(), 'total' => $songs->total(), ], ]); } /** * Neuen Song erstellen mit Default-Gruppen und -Arrangement. */ public function store(SongRequest $request): JsonResponse { $song = DB::transaction(function () use ($request) { $song = Song::create($request->validated()); $this->songService->createDefaultGroups($song); $this->songService->createDefaultArrangement($song); return $song; }); return response()->json([ 'message' => 'Song erfolgreich erstellt', 'data' => $this->formatSongDetail($song->fresh(['groups.slides', 'arrangements.arrangementGroups'])), ], 201); } /** * Song mit Gruppen, Slides und Arrangements anzeigen. */ public function show(int $id): JsonResponse { $song = Song::with(['groups.slides', 'arrangements.arrangementGroups'])->find($id); if (! $song) { return response()->json(['message' => 'Song nicht gefunden'], 404); } return response()->json([ 'data' => $this->formatSongDetail($song), ]); } /** * Song-Metadaten aktualisieren. */ public function update(SongRequest $request, int $id): JsonResponse { $song = Song::find($id); if (! $song) { return response()->json(['message' => 'Song nicht gefunden'], 404); } $song->update($request->validated()); return response()->json([ 'message' => 'Song erfolgreich aktualisiert', 'data' => $this->formatSongDetail($song->fresh(['groups.slides', 'arrangements.arrangementGroups'])), ]); } /** * Song soft-löschen. */ public function destroy(int $id): JsonResponse { $song = Song::find($id); if (! $song) { return response()->json(['message' => 'Song nicht gefunden'], 404); } $song->delete(); return response()->json([ 'message' => 'Song erfolgreich gelöscht', ]); } /** * Song-Detail formatieren. */ private function formatSongDetail(Song $song): array { return [ 'id' => $song->id, 'title' => $song->title, 'ccli_id' => $song->ccli_id, 'author' => $song->author, 'copyright_text' => $song->copyright_text, 'copyright_year' => $song->copyright_year, 'publisher' => $song->publisher, 'has_translation' => $song->has_translation, 'last_used_at' => $song->last_used_at?->toDateString(), 'last_used_in_service' => $song->last_used_in_service, 'created_at' => $song->created_at->toDateTimeString(), 'updated_at' => $song->updated_at->toDateTimeString(), 'groups' => $song->groups->sortBy('order')->values()->map(fn ($group) => [ 'id' => $group->id, 'name' => $group->name, 'color' => $group->color, 'order' => $group->order, 'slides' => $group->slides->sortBy('order')->values()->map(fn ($slide) => [ 'id' => $slide->id, 'order' => $slide->order, 'text_content' => $slide->text_content, 'text_content_translated' => $slide->text_content_translated, 'notes' => $slide->notes, ])->toArray(), ])->toArray(), 'arrangements' => $song->arrangements->map(fn ($arr) => [ 'id' => $arr->id, 'name' => $arr->name, 'is_default' => $arr->is_default, 'arrangement_groups' => $arr->arrangementGroups->sortBy('order')->values()->map(fn ($ag) => [ 'id' => $ag->id, 'song_group_id' => $ag->song_group_id, 'order' => $ag->order, ])->toArray(), ])->toArray(), ]; } }