updated stats, add logs and other data in user resource

This commit is contained in:
Gitea
2025-06-20 21:44:33 +05:30
parent 3afa74ecc2
commit 930144f8f8
6 changed files with 356 additions and 32 deletions

View File

@@ -3,8 +3,11 @@
namespace App\Filament\Resources;
use App\Filament\Resources\UserResource\Pages;
use App\Filament\Resources\UserResource\RelationManagers\LogsRelationManager;
use App\Filament\Resources\UserResource\RelationManagers\UsageLogsRelationManager;
use App\Models\User;
use DB;
use Filament\Actions\Action;
use Filament\Forms;
use Filament\Forms\Components\DatePicker;
use Filament\Forms\Components\Select;
@@ -14,17 +17,22 @@ use Filament\Notifications\Notification;
use Filament\Resources\Resource;
use Filament\Tables;
use Filament\Tables\Columns\BadgeColumn;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Filters\SelectFilter;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Database\Eloquent\SoftDeletingScope;
use Illuminate\Support\Facades\Response;
class UserResource extends Resource
{
protected static ?string $model = User::class;
protected static ?string $navigationIcon = 'heroicon-o-rectangle-stack';
protected static ?string $navigationIcon = 'heroicon-o-users';
protected static ?string $navigationGroup = 'Admin';
public static function form(Form $form): Form
{
@@ -38,7 +46,13 @@ class UserResource extends Resource
->email()
->required()
->maxLength(255),
TextInput::make('email_verified_at')
->label('Email Verification Status')
->disabled()
->formatStateUsing(fn ($record) => $record->email_verified_at ?? ''
? 'Verified at ' . $record->email_verified_at->toDateTimeString()
: 'Not Verified')
->helperText('Shows whether the user has verified their email address.'),
TextInput::make('stripe_id')
->label('Stripe ID')
->disabled()
@@ -74,6 +88,15 @@ class UserResource extends Resource
->columns([
TextColumn::make('name')->sortable()->searchable(),
TextColumn::make('email')->sortable()->searchable(),
IconColumn::make('email_verified_at')
->label('Verified')
->boolean()
->trueIcon('heroicon-o-check-circle')
->falseIcon('heroicon-o-x-circle')
->trueColor('success')
->falseColor('danger')
->getStateUsing(fn ($record) => !is_null($record->email_verified_at))
->sortable(),
BadgeColumn::make('level')
->label('User Level')
->getStateUsing(function ($record) {
@@ -96,7 +119,38 @@ class UserResource extends Resource
])
->defaultSort('created_at', 'desc')
->filters([
//
SelectFilter::make('subscription_status')
->label('Subscription Status')
->options([
'subscribed' => 'Has Active Subscription',
'not_subscribed' => 'No Active Subscription',
])
->query(function ($query, array $data) {
if ($data['value'] === 'subscribed') {
$query->whereHas('subscriptions', function ($query) {
$query->where('stripe_status', 'active')
->orWhere('stripe_status', 'trialing');
});
} elseif ($data['value'] === 'not_subscribed') {
$query->whereDoesntHave('subscriptions', function ($query) {
$query->where('stripe_status', 'active')
->orWhere('stripe_status', 'trialing');
});
}
}),
SelectFilter::make('email_verified')
->label('Email Verification')
->options([
'verified' => 'Verified',
'not_verified' => 'Not Verified',
])
->query(function ($query, array $data) {
if ($data['value'] === 'verified') {
$query->whereNotNull('email_verified_at');
} elseif ($data['value'] === 'not_verified') {
$query->whereNull('email_verified_at');
}
}),
])
->actions([
Tables\Actions\EditAction::make(),
@@ -144,7 +198,8 @@ class UserResource extends Resource
public static function getRelations(): array
{
return [
//
LogsRelationManager::class,
UsageLogsRelationManager::class,
];
}
@@ -156,4 +211,6 @@ class UserResource extends Resource
'edit' => Pages\EditUser::route('/{record}/edit'),
];
}
}

View File

@@ -3,8 +3,11 @@
namespace App\Filament\Resources\UserResource\Pages;
use App\Filament\Resources\UserResource;
use App\Models\User;
use Filament\Actions;
use Filament\Actions\Action;
use Filament\Resources\Pages\EditRecord;
use Illuminate\Support\Facades\Response;
class EditUser extends EditRecord
{
@@ -14,6 +17,72 @@ class EditUser extends EditRecord
{
return [
Actions\DeleteAction::make(),
Action::make('download_report')
->label('Download User Report')
->icon('heroicon-o-user')
->action(function (User $record) {
$userData = [
'Name' => $record->name,
'Email' => $record->email,
'Stripe ID' => $record->stripe_id ?? 'N/A',
'Payment Method Type' => $record->pm_type ?? 'N/A',
'Card Last 4' => $record->pm_last_four ?? 'N/A',
'Trial Ends At' => $record->trial_ends_at ? $record->trial_ends_at->toDateTimeString() : 'N/A',
'User Level' => match ($record->level) {
0 => 'Normal User',
1 => 'Banned',
9 => 'Super Admin',
default => 'Unknown',
},
'Email Verified At' => $record->email_verified_at ? $record->email_verified_at->toDateTimeString() : 'Not Verified',
];
$csv = fopen('php://temp', 'r+');
// User Details Header
fputcsv($csv, ['User Details']);
fputcsv($csv, array_keys($userData));
fputcsv($csv, array_values($userData));
fputcsv($csv, []);
// Usage Logs Header
fputcsv($csv, ['Usage Logs']);
fputcsv($csv, ['IP Address', 'Emails Created Count', 'Emails Received Count', 'Emails Created History', 'Emails Received History', 'Created At']);
foreach ($record->usageLogs as $log) {
fputcsv($csv, [
$log->ip_address,
$log->emails_created_count,
$log->emails_received_count,
is_array($log->emails_created_history) ? implode('; ', $log->emails_created_history) : 'None',
is_array($log->emails_received_history) ? implode('; ', $log->emails_received_history) : 'None',
$log->created_at->toDateTimeString(),
]);
}
fputcsv($csv, []);
// General Logs Header
fputcsv($csv, ['General Logs']);
fputcsv($csv, ['IP Address', 'Email', 'Created At']);
foreach ($record->logs as $log) {
fputcsv($csv, [
$log->ip,
$log->email,
$log->created_at->toDateTimeString(),
]);
}
rewind($csv);
$csvContent = stream_get_contents($csv);
fclose($csv);
return Response::streamDownload(
function () use ($csvContent) {
echo $csvContent;
},
"user_{$record->id}_report_" . now()->format('Ymd_His') . '.csv',
['Content-Type' => 'text/csv']
);
})
->color('primary'),
];
}
}

View File

@@ -0,0 +1,50 @@
<?php
namespace App\Filament\Resources\UserResource\RelationManagers;
use Filament\Forms;
use Filament\Forms\Form;
use Filament\Resources\RelationManagers\RelationManager;
use Filament\Tables;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\SoftDeletingScope;
class LogsRelationManager extends RelationManager
{
protected static string $relationship = 'logs';
protected static ?string $title = 'General Logs';
public function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('ip')
->label('IP Address')
->sortable()
->searchable(),
Tables\Columns\TextColumn::make('email')
->sortable()
->searchable(),
Tables\Columns\TextColumn::make('created_at')
->label('Logged At')
->dateTime()
->sortable(),
])
->filters([
//
])
->headerActions([
//Tables\Actions\CreateAction::make(),
])
->actions([
//Tables\Actions\EditAction::make(),
//Tables\Actions\DeleteAction::make(),
])
->bulkActions([
Tables\Actions\BulkActionGroup::make([
//Tables\Actions\DeleteBulkAction::make(),
]),
]);
}
}

View File

@@ -0,0 +1,55 @@
<?php
namespace App\Filament\Resources\UserResource\RelationManagers;
use Filament\Forms;
use Filament\Forms\Form;
use Filament\Resources\RelationManagers\RelationManager;
use Filament\Tables;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\SoftDeletingScope;
class UsageLogsRelationManager extends RelationManager
{
protected static string $relationship = 'usageLogs';
protected static ?string $title = 'Email Usage Logs';
public function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('ip_address')
->label('IP Address')
->sortable()
->searchable(),
Tables\Columns\TextColumn::make('emails_created_count')
->label('Emails Created')
->sortable(),
Tables\Columns\TextColumn::make('emails_received_count')
->label('Emails Received')
->sortable(),
Tables\Columns\TextColumn::make('updated_at')
->label('Last Activity At')
->dateTime()
->sortable(),
])
->filters([
//
])
->headerActions([
//Tables\Actions\CreateAction::make(),
])
->actions([
//Tables\Actions\EditAction::make(),
//Tables\Actions\DeleteAction::make(),
])
->bulkActions([
Tables\Actions\BulkActionGroup::make([
//Tables\Actions\DeleteBulkAction::make(),
]),
]);
}
}