Feature-Toggles, Administration, wiederkehrende Events und Event-Serien

- Administration & Rollenmanagement: Neuer Admin-Bereich mit Feature-Toggles
  und Sichtbarkeitseinstellungen pro Rolle (11 Toggles, 24 Visibility-Settings)
- AdministrationController mit eigenem Settings-Tab, aus SettingsController extrahiert
- Feature-Toggle-Guards in Controllers (Invitation, File, ListGenerator, Comment)
  und Views (events/show, events/edit, events/create)
- Setting::isFeatureEnabled() und isFeatureVisibleFor() Hilfsmethoden
- Wiederkehrende Trainings: Täglich/Wöchentlich/2-Wöchentlich mit Ende per
  Datum oder Anzahl (max. 52), Vorschau im Formular
- Event-Serien: Verknüpfung über event_series_id (UUID), Modal-Dialog beim
  Speichern und Löschen mit Optionen "nur dieses" / "alle folgenden"
- Löschen-Button direkt in der Event-Bearbeitung mit Serien-Dialog
- DemoDataSeeder: 4 Trainings als Serie mit gemeinsamer event_series_id
- Übersetzungen in allen 6 Sprachen (de, en, pl, ru, ar, tr)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Rhino
2026-03-03 08:38:45 +01:00
parent 0990e4249c
commit 8ccadbe89f
27 changed files with 1968 additions and 698 deletions

View File

@@ -6,6 +6,7 @@ use App\Http\Controllers\Controller;
use App\Models\ActivityLog;
use App\Models\Invitation;
use App\Models\Player;
use App\Models\Setting;
use App\Services\InvitationService;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
@@ -17,6 +18,10 @@ class InvitationController extends Controller
public function index(): View
{
if (!Setting::isFeatureVisibleFor('invitations', auth()->user())) {
abort(403);
}
$invitations = Invitation::with(['creator', 'players.team'])
->latest('created_at')
->paginate(20);
@@ -26,6 +31,10 @@ class InvitationController extends Controller
public function create(): View
{
if (!Setting::isFeatureVisibleFor('invitations', auth()->user())) {
abort(403);
}
$players = Player::with('team')->active()->orderBy('last_name')->get();
return view('admin.invitations.create', compact('players'));
@@ -33,6 +42,10 @@ class InvitationController extends Controller
public function store(Request $request): RedirectResponse
{
if (!Setting::isFeatureVisibleFor('invitations', auth()->user())) {
abort(403);
}
$validated = $request->validate([
'email' => ['nullable', 'email', 'max:255'],
'expires_in_days' => ['required', 'integer', 'min:1', 'max:90'],
@@ -52,6 +65,10 @@ class InvitationController extends Controller
public function destroy(Invitation $invitation): RedirectResponse
{
if (!Setting::isFeatureVisibleFor('invitations', auth()->user())) {
abort(403);
}
if ($invitation->isAccepted()) {
return back()->with('error', __('admin.invitation_already_used'));
}