- Galerien sind nun eine Entität - es kann mehrere geben
- Neues Sparkbooth-Upload-Feature: Endpoint /api/sparkbooth/upload (Token-basiert pro Galerie), Controller Api/SparkboothUploadController, Migration 2026_01_21_000001_add_upload_fields_to_galleries_table.php mit Upload-Flags/Token/Expiry;
Galerie-Modell und Factory/Seeder entsprechend erweitert.
- Filament: Neue Setup-Seite SparkboothSetup (mit View) zur schnellen Galerie- und Token-Erstellung inkl. QR/Endpoint/Snippet;
Galerie-Link-Views nutzen jetzt simple-qrcode (Composer-Dependency hinzugefügt) und bieten PNG-Download.
- Galerie-Tabelle: Slug/Pfad-Spalten entfernt, Action „Link-Details“ mit Modal; Created-at-Spalte hinzugefügt.
- Zugriffshärtung: Galerie-IDs in API (ImageController, Download/Print) geprüft; GalleryAccess/Middleware + Gallery-Modell/Slug-UUID
eingeführt; GalleryAccess-Inertia-Seite.
- UI/UX: LoadingSpinner/StyledImageDisplay verbessert, Delete-Confirm, Übersetzungen ergänzt.
This commit is contained in:
@@ -5,6 +5,8 @@ namespace App\Filament\Pages;
|
||||
use App\Services\PrinterService;
|
||||
use App\Settings\GeneralSettings;
|
||||
use BackedEnum;
|
||||
use Carbon\Carbon;
|
||||
use Filament\Forms\Components\DateTimePicker;
|
||||
use Filament\Forms\Components\Select;
|
||||
use Filament\Forms\Components\TextInput;
|
||||
use Filament\Forms\Components\Toggle;
|
||||
@@ -14,6 +16,7 @@ use Filament\Notifications\Notification;
|
||||
use Filament\Pages\Page;
|
||||
use Filament\Schemas\Schema;
|
||||
use Illuminate\Support\Arr;
|
||||
use Illuminate\Support\Facades\Hash;
|
||||
use UnitEnum;
|
||||
|
||||
class GlobalSettings extends Page implements HasForms
|
||||
@@ -40,7 +43,10 @@ class GlobalSettings extends Page implements HasForms
|
||||
|
||||
public function mount(GeneralSettings $settings): void
|
||||
{
|
||||
$this->form->fill($settings->toArray());
|
||||
$data = $settings->toArray();
|
||||
$data['gallery_password'] = null;
|
||||
|
||||
$this->form->fill($data);
|
||||
}
|
||||
|
||||
public function form(Schema $schema): Schema
|
||||
@@ -59,6 +65,24 @@ class GlobalSettings extends Page implements HasForms
|
||||
TextInput::make('gallery_heading')
|
||||
->label(__('filament.resource.setting.form.gallery_heading'))
|
||||
->required(),
|
||||
Toggle::make('require_gallery_password')
|
||||
->label(__('filament.resource.setting.form.require_gallery_password'))
|
||||
->helperText(__('filament.resource.setting.form.require_gallery_password_help')),
|
||||
TextInput::make('gallery_password')
|
||||
->label(__('filament.resource.setting.form.gallery_password'))
|
||||
->password()
|
||||
->revealable()
|
||||
->helperText(__('filament.resource.setting.form.gallery_password_help'))
|
||||
->dehydrated(true),
|
||||
DateTimePicker::make('gallery_expires_at')
|
||||
->label(__('filament.resource.setting.form.gallery_expires_at'))
|
||||
->native(false)
|
||||
->seconds(false),
|
||||
TextInput::make('gallery_access_duration_minutes')
|
||||
->label(__('filament.resource.setting.form.gallery_access_duration_minutes'))
|
||||
->numeric()
|
||||
->minValue(1)
|
||||
->helperText(__('filament.resource.setting.form.gallery_access_duration_help')),
|
||||
TextInput::make('new_image_timespan_minutes')
|
||||
->label(__('filament.resource.setting.form.new_image_timespan_minutes'))
|
||||
->numeric()
|
||||
@@ -92,11 +116,39 @@ class GlobalSettings extends Page implements HasForms
|
||||
$data['custom_printer_address'] = null;
|
||||
}
|
||||
|
||||
$data['require_gallery_password'] = (bool) Arr::get($data, 'require_gallery_password', false);
|
||||
$data['new_image_timespan_minutes'] = (int) Arr::get($data, 'new_image_timespan_minutes', 0);
|
||||
$data['image_refresh_interval'] = (int) Arr::get($data, 'image_refresh_interval', 0);
|
||||
$data['max_number_of_copies'] = (int) Arr::get($data, 'max_number_of_copies', 0);
|
||||
$data['show_print_button'] = (bool) Arr::get($data, 'show_print_button', false);
|
||||
$data['custom_printer_address'] = $data['custom_printer_address'] ?: null;
|
||||
$duration = Arr::get($data, 'gallery_access_duration_minutes');
|
||||
$data['gallery_access_duration_minutes'] = $duration === null || $duration === '' ? null : (int) $duration;
|
||||
|
||||
$expiresAt = Arr::get($data, 'gallery_expires_at');
|
||||
$data['gallery_expires_at'] = $expiresAt ? Carbon::parse($expiresAt) : null;
|
||||
|
||||
$newPassword = Arr::get($data, 'gallery_password');
|
||||
$currentHash = $settings->gallery_password_hash;
|
||||
|
||||
if (! $data['require_gallery_password']) {
|
||||
$data['gallery_password_hash'] = null;
|
||||
} else {
|
||||
$data['gallery_password_hash'] = $newPassword
|
||||
? Hash::make($newPassword)
|
||||
: $currentHash;
|
||||
|
||||
if (! $data['gallery_password_hash']) {
|
||||
Notification::make()
|
||||
->title(__('filament.resource.setting.form.gallery_password_missing'))
|
||||
->danger()
|
||||
->send();
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
unset($data['gallery_password']);
|
||||
|
||||
$settings->fill($data)->save();
|
||||
|
||||
|
||||
Reference in New Issue
Block a user