webseite funktioniert, pay sdk, blog backend funktioniert

This commit is contained in:
Codex Agent
2025-09-29 22:16:12 +02:00
parent e52a4005aa
commit 21c9391e2c
51 changed files with 2093 additions and 1293 deletions

View File

@@ -0,0 +1,217 @@
<?php
namespace App\Filament\Blog\Resources;
use Illuminate\Support\Facades\Log;
use App\Filament\Blog\Resources\CategoryResource\Pages;
use App\Filament\Blog\Traits\HasContentEditor;
use App\Models\BlogCategory;
use Filament\Forms;
use Filament\Forms\Components\MarkdownEditor;
use Filament\Schemas\Components\Section;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Components\Textarea;
use Filament\Forms\Components\Toggle;
use Filament\Schemas\Components\Utilities\Get;
use Filament\Schemas\Components\Utilities\Set;
use Filament\Infolists\Components\TextEntry;
use Filament\Resources\Resource;
use Filament\Schemas\Schema;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\SoftDeletingScope;
use Illuminate\Support\Str;
use Filament\Schemas\Components\Tabs as SchemaTabs;
use Filament\Schemas\Components\Tabs\Tab as SchemaTab;
class CategoryResource extends Resource
{
use HasContentEditor;
protected static ?string $model = BlogCategory::class;
protected static string | \BackedEnum | null $navigationIcon = 'heroicon-o-list-bullet';
protected static ?string $navigationLabel = 'Kategorien';
protected static ?string $pluralLabel = 'Kategorien';
protected static ?string $modelLabel = 'Kategorie';
public static function form(Schema $schema): Schema
{
return $schema
->schema([
SchemaTabs::make('Übersetzungen')
->tabs([
SchemaTab::make('Deutsch')
->schema([
TextInput::make('name_de')
->label('Name')
->validationAttribute('name_de')
->live()
->afterStateUpdated(function (Get $get, Set $set, ?string $state) {
if (($get('slug') ?? '') !== Str::slug($state)) {
return;
}
$set('slug', Str::slug($state));
}),
MarkdownEditor::make('description_de')
->label('Beschreibung')
->validationAttribute('description_de'),
]),
SchemaTab::make('Englisch')
->schema([
TextInput::make('name_en')
->label('Name'),
MarkdownEditor::make('description_en')
->label('Beschreibung'),
]),
]),
TextInput::make('slug')
->label('Slug')
->required()
->unique(BlogCategory::class, 'slug', ignoreRecord: true),
Section::make('Sichtbarkeit')
->schema([
Toggle::make('is_visible')
->label('Sichtbar für Gäste')
->default(true),
]),
Section::make('Metadaten')
->schema([
TextEntry::make('created_at')
->label('Erstellt am')
->default('—')
->state(fn (?BlogCategory $record) => $record?->created_at?->diffForHumans()),
TextEntry::make('updated_at')
->label('Zuletzt geändert')
->default('—')
->state(fn (?BlogCategory $record) => $record?->updated_at?->diffForHumans()),
]),
])
->columns(3);
}
public static function mutateFormDataBeforeFill(array $data): array
{
$nameJson = $data['name'] ?? '[]';
$nameArray = json_decode($nameJson, true) ?: [];
$data['name_de'] = $nameArray['de'] ?? '';
$data['name_en'] = $nameArray['en'] ?? '';
$descJson = $data['description'] ?? '[]';
$descArray = json_decode($descJson, true) ?: [];
$data['description_de'] = $descArray['de'] ?? '';
$data['description_en'] = $descArray['en'] ?? '';
\Illuminate\Support\Facades\Log::info('BeforeFill Description Extraction:', [
'descJson' => $descJson,
'descArray' => $descArray,
'description_de' => $data['description_de'],
'description_en' => $data['description_en'],
]);
return $data;
}
public static function mutateFormDataBeforeCreate(array $data): array
{
\Illuminate\Support\Facades\Log::info('mutateFormDataBeforeCreate Input Data:', ['data' => $data]);
$nameData = [
'de' => $data['name_de'] ?? '',
'en' => $data['name_en'] ?? '',
];
$data['name'] = json_encode($nameData);
\Illuminate\Support\Facades\Log::info('mutateFormDataBeforeCreate Name JSON:', ['name' => $nameData]);
$descData = [
'de' => $data['description_de'] ?? '',
'en' => $data['description_en'] ?? '',
];
$data['description'] = json_encode($descData);
\Illuminate\Support\Facades\Log::info('mutateFormDataBeforeCreate Description JSON:', ['description' => $descData]);
unset($data['name_de'], $data['name_en'], $data['description_de'], $data['description_en']);
\Illuminate\Support\Facades\Log::info('mutateFormDataBeforeCreate Final Data:', $data);
return $data;
}
public static function mutateFormDataBeforeSave(array $data): array
{
$transformed = static::mutateFormDataBeforeCreate($data);
return $transformed;
}
public static function table(Table $table): Table
{
return $table
->columns([
TextColumn::make('name')
->label('Name (DE)')
->getStateUsing(fn ($record) => json_decode($record->name ?? '[]', true)['de'] ?? '—')
->searchable()
->sortable(),
TextColumn::make('slug')
->label('Slug')
->searchable()
->sortable(),
IconColumn::make('is_visible')
->label('Sichtbar')
->boolean()
->trueIcon('heroicon-o-check-circle')
->falseIcon('heroicon-o-x-circle'),
TextColumn::make('updated_at')
->label('Zuletzt geändert')
->date()
->sortable(),
])
->filters([
//
])
->actions([
EditAction::make(),
])
->bulkActions([
BulkActionGroup::make([
DeleteBulkAction::make(),
]),
]);
}
public static function getRelations(): array
{
return [
//
];
}
public static function getPages(): array
{
return [
'index' => Pages\ListCategories::route('/'),
'create' => Pages\CreateCategory::route('/create'),
'edit' => Pages\EditCategory::route('/{record}/edit'),
];
}
public static function getEloquentQuery(): Builder
{
return parent::getEloquentQuery()
->withoutGlobalScopes([
SoftDeletingScope::class,
]);
}
}

View File

@@ -0,0 +1,33 @@
<?php
namespace App\Filament\Blog\Resources\CategoryResource\Pages;
use App\Filament\Blog\Resources\CategoryResource;
use Filament\Resources\Pages\CreateRecord;
class CreateCategory extends CreateRecord
{
protected static string $resource = CategoryResource::class;
protected function getFormValidationRules(): array
{
return [
'name_de' => 'required|string|max:255',
'description_de' => 'nullable|string',
'name_en' => 'nullable|string|max:255',
'description_en' => 'nullable|string',
'slug' => 'required|string|max:255|unique:blog_categories,slug',
'is_visible' => 'boolean',
];
}
protected function store()
{
$state = $this->form->getState();
$data = $state['data'] ?? $state;
$data = \App\Filament\Blog\Resources\CategoryResource::mutateFormDataBeforeCreate($data);
$this->record = static::getResource()::getModel()::create($data);
}
}

View File

@@ -0,0 +1,51 @@
<?php
namespace App\Filament\Blog\Resources\CategoryResource\Pages;
use App\Filament\Blog\Resources\CategoryResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditCategory extends EditRecord
{
protected static string $resource = CategoryResource::class;
protected function getHeaderActions(): array
{
return [
Actions\DeleteAction::make(),
];
}
protected function getFormValidationRules(): array
{
return [
'name_de' => 'required|string|max:255',
'description_de' => 'nullable|string',
'name_en' => 'nullable|string|max:255',
'description_en' => 'nullable|string',
'slug' => 'required|string|max:255|unique:blog_categories,slug,' . $this->record->id,
'is_visible' => 'boolean',
];
}
public function mount($record): void
{
parent::mount($record);
$data = $this->record->toArray();
$data = \App\Filament\Blog\Resources\CategoryResource::mutateFormDataBeforeFill($data);
$this->form->fill($data);
}
public function save(bool $shouldRedirect = true, bool $shouldSendSavedNotification = true): void
{
$state = $this->form->getState();
\Illuminate\Support\Facades\Log::info('EditCategory Save - Full State:', $state);
$data = $state['data'] ?? $state;
$data = \App\Filament\Blog\Resources\CategoryResource::mutateFormDataBeforeSave($data);
$this->record->update($data);
}
}

View File

@@ -0,0 +1,19 @@
<?php
namespace App\Filament\Blog\Resources\CategoryResource\Pages;
use App\Filament\Blog\Resources\CategoryResource;
use Filament\Actions;
use Filament\Resources\Pages\ListRecords;
class ListCategories extends ListRecords
{
protected static string $resource = CategoryResource::class;
protected function getHeaderActions(): array
{
return [
Actions\CreateAction::make(),
];
}
}

View File

@@ -0,0 +1,304 @@
<?php
namespace App\Filament\Blog\Resources;
use App\Filament\Blog\Resources\PostResource\Pages;
use App\Filament\Blog\Traits\HasContentEditor;
use App\Models\BlogPost;
use App\Models\BlogCategory;
use Filament\Forms;
use Filament\Forms\Components\DateTimePicker;
use Filament\Forms\Components\FileUpload;
use Filament\Forms\Components\MarkdownEditor;
use Filament\Schemas\Components\Section;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\Textarea;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Components\Toggle;
use Filament\Schemas\Components\Utilities\Get;
use Filament\Schemas\Components\Utilities\Set;
use Filament\Infolists\Components\TextEntry;
use Filament\Resources\Resource;
use Filament\Schemas\Schema;
use Filament\Tables;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\CreateAction;
use Filament\Actions\DeleteAction;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction;
use Filament\Actions\ViewAction;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Filters\TernaryFilter;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\SoftDeletingScope;
use Illuminate\Support\Str;
use Filament\Schemas\Components\Tabs as SchemaTabs;
use Filament\Schemas\Components\Tabs\Tab as SchemaTab;
class PostResource extends Resource
{
use HasContentEditor;
protected static ?string $model = BlogPost::class;
protected static string | \BackedEnum | null $navigationIcon = 'heroicon-o-document-text';
protected static ?string $navigationLabel = 'Beiträge';
protected static ?string $pluralLabel = 'Beiträge';
protected static ?string $modelLabel = 'Beitrag';
public static function form(Schema $schema): Schema
{
return $schema
->schema([
SchemaTabs::make('Übersetzungen')
->tabs([
SchemaTab::make('Deutsch')
->schema([
TextInput::make('title_de')
->label('Titel')
->required()
->maxLength(255)
->live()
->afterStateUpdated(function (Get $get, Set $set, ?string $old, ?string $state) {
if (($get('slug') ?? '') !== Str::slug($state)) {
return;
}
$set('slug', Str::slug($state));
}),
MarkdownEditor::make('content_de')
->label('Inhalt')
->required()
->columnSpanFull(),
TextInput::make('excerpt_de')
->label('Auszug')
->maxLength(255),
TextInput::make('meta_title_de')
->label('Meta-Titel')
->maxLength(255),
Textarea::make('meta_description_de')
->label('Meta-Beschreibung')
->maxLength(65535)
->columnSpanFull(),
])
->columns(2),
SchemaTab::make('Englisch')
->schema([
TextInput::make('title_en')
->label('Titel')
->maxLength(255),
MarkdownEditor::make('content_en')
->label('Inhalt')
->columnSpanFull(),
TextInput::make('excerpt_en')
->label('Auszug')
->maxLength(255),
TextInput::make('meta_title_en')
->label('Meta-Titel')
->maxLength(255),
Textarea::make('meta_description_en')
->label('Meta-Beschreibung')
->maxLength(65535)
->columnSpanFull(),
])
->columns(2),
]),
TextInput::make('slug')
->label('Slug')
->required()
->unique(BlogPost::class, 'slug', ignoreRecord: true)
->maxLength(255),
Section::make('Bild und Kategorie')
->schema([
FileUpload::make('featured_image')
->label('Featured Image')
->image()
->directory('blog')
->visibility('public'),
Select::make('category_id')
->label('Kategorie')
->relationship('category', 'name_de')
->required()
->preload()
->createOptionForm([
TextInput::make('name_de')
->label('Name (DE)')
->required()
->maxLength(255)
->afterStateUpdated(fn (Set $set, $state) => $set('name_en', $state)),
TextInput::make('slug')
->label('Slug')
->required()
->unique(\App\Models\BlogCategory::class, 'slug', ignoreRecord: true)
->maxLength(255),
]),
])
->columns(2),
Section::make('Veröffentlichung')
->schema([
Toggle::make('is_published')
->label('Veröffentlicht'),
DateTimePicker::make('published_at')
->label('Veröffentlicht am')
->displayFormat('Y-m-d H:i:s')
->default(now()),
]),
]);
}
public static function mutateFormDataBeforeCreate(array $data): array
{
$data['translations'] = [
'title' => [
'de' => $data['title_de'] ?? '',
'en' => $data['title_en'] ?? '',
],
'content' => [
'de' => $data['content_de'] ?? '',
'en' => $data['content_en'] ?? '',
],
'excerpt' => [
'de' => $data['excerpt_de'] ?? '',
'en' => $data['excerpt_en'] ?? '',
],
'meta_title' => [
'de' => $data['meta_title_de'] ?? '',
'en' => $data['meta_title_en'] ?? '',
],
'meta_description' => [
'de' => $data['meta_description_de'] ?? '',
'en' => $data['meta_description_en'] ?? '',
],
];
unset($data['title_de'], $data['title_en'], $data['content_de'], $data['content_en'], $data['excerpt_de'], $data['excerpt_en'], $data['meta_title_de'], $data['meta_title_en'], $data['meta_description_de'], $data['meta_description_en']);
return $data;
}
public static function mutateFormDataBeforeFill(array $data): array
{
$record = static::getModel()::find(request()?->route()?->parameter('record') ?? request()?->input('record_id') ?? null);
if (!$record) {
return $data;
}
$data['title_de'] = $record->getTranslation('title', 'de');
$data['title_en'] = $record->getTranslation('title', 'en');
$data['content_de'] = $record->getTranslation('content', 'de');
$data['content_en'] = $record->getTranslation('content', 'en');
$data['excerpt_de'] = $record->getTranslation('excerpt', 'de');
$data['excerpt_en'] = $record->getTranslation('excerpt', 'en');
$data['meta_title_de'] = $record->getTranslation('meta_title', 'de');
$data['meta_title_en'] = $record->getTranslation('meta_title', 'en');
$data['meta_description_de'] = $record->getTranslation('meta_description', 'de');
$data['meta_description_en'] = $record->getTranslation('meta_description', 'en');
return $data;
}
public static function mutateFormDataBeforeSave(array $data): array
{
$data['translations'] = [
'title' => [
'de' => $data['title_de'] ?? '',
'en' => $data['title_en'] ?? '',
],
'content' => [
'de' => $data['content_de'] ?? '',
'en' => $data['content_en'] ?? '',
],
'excerpt' => [
'de' => $data['excerpt_de'] ?? '',
'en' => $data['excerpt_en'] ?? '',
],
'meta_title' => [
'de' => $data['meta_title_de'] ?? '',
'en' => $data['meta_title_en'] ?? '',
],
'meta_description' => [
'de' => $data['meta_description_de'] ?? '',
'en' => $data['meta_description_en'] ?? '',
],
];
unset($data['title_de'], $data['title_en'], $data['content_de'], $data['content_en'], $data['excerpt_de'], $data['excerpt_en'], $data['meta_title_de'], $data['meta_title_en'], $data['meta_description_de'], $data['meta_description_en']);
return $data;
}
public static function table(Table $table): Table
{
return $table
->columns([
TextColumn::make('title')
->label('Titel (DE)')
->getStateUsing(fn ($record) => $record->getTranslation('title', 'de'))
->searchable()
->sortable(),
TextColumn::make('category.name_de')
->label('Kategorie')
->badge()
->color('primary'),
IconColumn::make('is_published')
->label('Veröffentlicht')
->boolean()
->trueIcon('heroicon-o-check-circle')
->falseIcon('heroicon-o-x-circle'),
TextColumn::make('published_at')
->label('Veröffentlicht am')
->dateTime()
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('created_at')
->label('Erstellt am')
->dateTime()
->sortable()
->toggleable(isToggledHiddenByDefault: true),
])
->filters([
TernaryFilter::make('is_published')
->label('Veröffentlicht'),
])
->actions([
ViewAction::make(),
EditAction::make(),
DeleteAction::make(),
])
->bulkActions([
BulkActionGroup::make([
DeleteBulkAction::make(),
]),
]);
}
public static function getRelations(): array
{
return [
//
];
}
public static function getPages(): array
{
return [
'index' => Pages\ListPosts::route('/'),
'create' => Pages\CreatePost::route('/create'),
'view' => Pages\ViewPost::route('/{record}'),
'edit' => Pages\EditPost::route('/{record}/edit'),
];
}
public static function getEloquentQuery(): Builder
{
return parent::getEloquentQuery()
->withoutGlobalScopes([
SoftDeletingScope::class,
]);
}
}

View File

@@ -0,0 +1,11 @@
<?php
namespace App\Filament\Blog\Resources\PostResource\Pages;
use App\Filament\Blog\Resources\PostResource;
use Filament\Resources\Pages\CreateRecord;
class CreatePost extends CreateRecord
{
protected static string $resource = PostResource::class;
}

View File

@@ -0,0 +1,20 @@
<?php
namespace App\Filament\Blog\Resources\PostResource\Pages;
use App\Filament\Blog\Resources\PostResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditPost extends EditRecord
{
protected static string $resource = PostResource::class;
protected function getHeaderActions(): array
{
return [
Actions\ViewAction::make(),
Actions\DeleteAction::make(),
];
}
}

View File

@@ -0,0 +1,19 @@
<?php
namespace App\Filament\Blog\Resources\PostResource\Pages;
use App\Filament\Blog\Resources\PostResource;
use Filament\Actions;
use Filament\Resources\Pages\ListRecords;
class ListPosts extends ListRecords
{
protected static string $resource = PostResource::class;
protected function getHeaderActions(): array
{
return [
Actions\CreateAction::make(),
];
}
}

View File

@@ -0,0 +1,11 @@
<?php
namespace App\Filament\Blog\Resources\PostResource\Pages;
use App\Filament\Blog\Resources\PostResource;
use Filament\Resources\Pages\ViewRecord;
class ViewPost extends ViewRecord
{
protected static string $resource = PostResource::class;
}

View File

@@ -0,0 +1,29 @@
<?php
namespace App\Filament\Blog\Traits;
use Filament\Forms\Components\MarkdownEditor;
trait HasContentEditor
{
public static function getContentEditor(string $field)
{
return MarkdownEditor::make($field)
->label('Inhalt')
->columnSpanFull()
->toolbarButtons(config('filament-blog.toolbar_buttons', [
'bold',
'italic',
'underline',
'strike',
'bulletList',
'orderedList',
'link',
'table',
'codeBlock',
'h1',
'h2',
'h3',
]));
}
}