feat: upgrade filament to v4 and ensure 100% test coverage

- Upgrade Filament framework from v3 to v4
   - Update all Filament resources and pages for v4 compatibility
   - Fix test suite to maintain 100% pass rate (321 tests passing)
   - Add visibility condition for ticket close action (only when not closed)
   - Update dependencies and build assets for new Filament version
   - Maintain backward compatibility while leveraging v4 improvements
This commit is contained in:
idevakk
2025-11-14 01:42:07 -08:00
parent 3706072ce5
commit 3892c48ef2
103 changed files with 1741 additions and 890 deletions

View File

@@ -2,6 +2,21 @@
namespace App\Filament\Resources;
use Filament\Schemas\Schema;
use Filament\Actions\ViewAction;
use Filament\Actions\EditAction;
use Filament\Actions\DeleteAction;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\BulkAction;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\Mail;
use App\Mail\TicketResponseNotification;
use App\Filament\Resources\TicketResource\RelationManagers\ResponsesRelationManager;
use App\Filament\Resources\TicketResource\Pages\ListTickets;
use App\Filament\Resources\TicketResource\Pages\CreateTicket;
use App\Filament\Resources\TicketResource\Pages\EditTicket;
use App\Filament\Resources\TicketResource\Pages;
use App\Filament\Resources\TicketResource\RelationManagers;
use App\Models\Ticket;
@@ -12,11 +27,9 @@ use Filament\Forms\Components\Repeater;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\Textarea;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Form;
use Filament\Notifications\Notification;
use Filament\Resources\Resource;
use Filament\Tables;
use Filament\Tables\Actions\Action;
use Filament\Tables\Columns\BadgeColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Filters\Filter;
@@ -31,13 +44,13 @@ class TicketResource extends Resource
protected static ?string $model = Ticket::class;
protected static ?string $navigationIcon = 'heroicon-o-ticket';
protected static ?string $navigationGroup = 'Support';
protected static string | \BackedEnum | null $navigationIcon = 'heroicon-o-ticket';
protected static string | \UnitEnum | null $navigationGroup = 'Support';
public static function form(Form $form): Form
public static function form(Schema $schema): Schema
{
return $form
->schema([
return $schema
->components([
Select::make('user_id')
->relationship('user', 'name')
->searchable()
@@ -92,7 +105,7 @@ class TicketResource extends Resource
TextColumn::make('created_at')
->label('Created At')
->dateTime('F d, Y • h:i A')->sortable(),
TextColumn::make('last_response_at')
TextColumn::make('updated_at')
->label('Last Response')
->sortable()
->formatStateUsing(fn ($state) => $state?->diffForHumans()),
@@ -108,7 +121,7 @@ class TicketResource extends Resource
])
->attribute('status'),
Filter::make('created_at')
->form([
->schema([
DatePicker::make('created_from')->label('Created From'),
DatePicker::make('created_until')->label('Created Until'),
])
@@ -121,14 +134,14 @@ class TicketResource extends Resource
// ->actions([
// Tables\Actions\EditAction::make(),
// ])
->actions([
Tables\Actions\ViewAction::make(),
Tables\Actions\EditAction::make(),
Tables\Actions\DeleteAction::make(),
->recordActions([
ViewAction::make(),
EditAction::make(),
DeleteAction::make(),
Action::make('view')
->label('View & Respond')
->icon('heroicon-o-eye')
->form(function (Ticket $ticket): array {
->schema(function (Ticket $ticket): array {
return [
TextArea::make('response')
->label('Your Response')
@@ -175,7 +188,7 @@ class TicketResource extends Resource
$html .= '</div>';
return new \Illuminate\Support\HtmlString($html);
return new HtmlString($html);
})
->action(function (array $data, Ticket $ticket) {
@@ -200,12 +213,13 @@ class TicketResource extends Resource
->modalHeading('View & Respond to Ticket')
->modalSubmitActionLabel('Send Reply'),
])
->actions([
->recordActions([
Action::make('close')
->label('Close Ticket')
->icon('heroicon-o-x-circle')
->color('danger')
->requiresConfirmation()
->visible(fn (Ticket $ticket): bool => $ticket->status !== 'closed')
->action(function (Ticket $ticket) {
$ticket->update(['status' => 'closed']);
}),
@@ -218,16 +232,16 @@ class TicketResource extends Resource
$ticket->update(['status' => 'open']);
}),
])
->bulkActions([
Tables\Actions\BulkActionGroup::make([
Tables\Actions\DeleteBulkAction::make(),
Tables\Actions\BulkAction::make('notify_users')
->toolbarActions([
BulkActionGroup::make([
DeleteBulkAction::make(),
BulkAction::make('notify_users')
->label('Send Email Notification')
->color('success')
->icon('heroicon-o-envelope')
->requiresConfirmation()
->deselectRecordsAfterCompletion()
->action(function (\Illuminate\Support\Collection $records) {
->action(function (Collection $records) {
foreach ($records as $ticket) {
$responses = $ticket->responses()
->with('user')
@@ -235,12 +249,12 @@ class TicketResource extends Resource
->get();
if ($ticket->user && $ticket->user->email) {
\Illuminate\Support\Facades\Mail::to($ticket->user->email)
->send(new \App\Mail\TicketResponseNotification($ticket, $responses));
Mail::to($ticket->user->email)
->send(new TicketResponseNotification($ticket, $responses));
}
}
\Filament\Notifications\Notification::make()
Notification::make()
->title('Email notifications sent successfully!')
->success()
->send();
@@ -252,16 +266,16 @@ class TicketResource extends Resource
public static function getRelations(): array
{
return [
RelationManagers\ResponsesRelationManager::class,
ResponsesRelationManager::class,
];
}
public static function getPages(): array
{
return [
'index' => Pages\ListTickets::route('/'),
'create' => Pages\CreateTicket::route('/create'),
'edit' => Pages\EditTicket::route('/{record}/edit'),
'index' => ListTickets::route('/'),
'create' => CreateTicket::route('/create'),
'edit' => EditTicket::route('/{record}/edit'),
];
}
}