fix: Filament v5 callbacks $r → $record (Plans/Subs/SuperAdmins/Companies)

+ central PWA: real PNG icons, SW registration, scope=/

- All `fn ($r) =>` and `fn (Type $r) =>` replaced with $record (Filament v5
  injects callback params by name; $r resolved to nothing)
- /pwa/admin-{192,512}.png — generated on-the-fly with GD + DejaVuSans-Bold
- /pwa/admin-icon.svg — vector favicon
- /admin-sw.js — service worker (cache shell, network-first elsewhere)
  with Service-Worker-Allowed: / header
- Manifest scope=/ + start_url=/admin → install prompt fires on Chrome/Edge/Safari
- BODY_END render hook registers SW on central panel
This commit is contained in:
2026-05-08 04:37:25 +00:00
parent 0ac42dde3d
commit d1a18848d3
8 changed files with 142 additions and 35 deletions
@@ -110,7 +110,7 @@ class CompanyResource extends Resource
Tables\Columns\TextColumn::make('slug')
->searchable()
->copyable()
->url(fn (Company $r) => $r->url('/app'))
->url(fn (Company $record) => $record->url('/app'))
->openUrlInNewTab(),
Tables\Columns\TextColumn::make('name')->searchable()->sortable(),
Tables\Columns\TextColumn::make('status')
@@ -132,30 +132,30 @@ class CompanyResource extends Resource
'suspended' => 'Suspendat', 'archived' => 'Arhivat',
]),
])
->recordUrl(fn (Company $r) => CompanyResource::getUrl('view', ['record' => $r]))
->recordUrl(fn (Company $record) => CompanyResource::getUrl('view', ['record' => $record]))
->actions([
Actions\ViewAction::make()
->url(fn (Company $r) => CompanyResource::getUrl('view', ['record' => $r])),
->url(fn (Company $record) => CompanyResource::getUrl('view', ['record' => $record])),
Actions\Action::make('open_tenant')
->label('Deschide')
->icon('heroicon-m-arrow-top-right-on-square')
->color('primary')
->url(fn (Company $r) => $r->url('/app'))
->url(fn (Company $record) => $record->url('/app'))
->openUrlInNewTab(),
Actions\Action::make('suspend')
->label('Suspendă')
->icon('heroicon-m-no-symbol')
->color('danger')
->visible(fn (Company $r) => in_array($r->status, ['active', 'trial']))
->visible(fn (Company $record) => in_array($record->status, ['active', 'trial']))
->requiresConfirmation()
->action(fn (Company $r) => app(\App\Services\CompanyProvisioner::class)->suspend($r)),
->action(fn (Company $record) => app(\App\Services\CompanyProvisioner::class)->suspend($record)),
Actions\Action::make('activate')
->label('Activează')
->icon('heroicon-m-check-circle')
->color('success')
->visible(fn (Company $r) => in_array($r->status, ['suspended', 'expired']))
->visible(fn (Company $record) => in_array($record->status, ['suspended', 'expired']))
->requiresConfirmation()
->action(fn (Company $r) => app(\App\Services\CompanyProvisioner::class)->reactivate($r)),
->action(fn (Company $record) => app(\App\Services\CompanyProvisioner::class)->reactivate($record)),
Actions\EditAction::make(),
Actions\DeleteAction::make(),
])
@@ -88,11 +88,11 @@ class PlanResource extends Resource
->columns([
Tables\Columns\TextColumn::make('name')->searchable()->sortable(),
Tables\Columns\TextColumn::make('price_monthly')
->money(fn ($r) => $r->currency)
->money(fn ($record) => $record->currency)
->label('Lunar')
->sortable(),
Tables\Columns\TextColumn::make('price_yearly')
->money(fn ($r) => $r->currency)
->money(fn ($record) => $record->currency)
->label('Anual'),
Tables\Columns\TextColumn::make('companies_count')
->counts('companies')
@@ -109,19 +109,19 @@ class SubscriptionResource extends Resource
->columns([
Tables\Columns\TextColumn::make('invoice_number')
->label('Factură')
->placeholder(fn ($r) => '—')
->placeholder(fn ($record) => '—')
->copyable()
->searchable(),
Tables\Columns\TextColumn::make('company.name')
->label('Companie')
->searchable()
->url(fn ($r) => \App\Filament\Central\Resources\CompanyResource::getUrl('view', ['record' => $r->company_id])),
->url(fn ($record) => \App\Filament\Central\Resources\CompanyResource::getUrl('view', ['record' => $record->company_id])),
Tables\Columns\TextColumn::make('plan.name')->label('Plan')->placeholder('—'),
Tables\Columns\TextColumn::make('period')
->formatStateUsing(fn ($s) => Subscription::PERIODS[$s] ?? $s)
->badge(),
Tables\Columns\TextColumn::make('amount')
->money(fn ($r) => $r->currency)
->money(fn ($record) => $record->currency)
->sortable()
->weight('bold'),
Tables\Columns\TextColumn::make('status')
@@ -146,16 +146,16 @@ class SubscriptionResource extends Resource
->label('Marchează plătit')
->icon('heroicon-m-check-circle')
->color('success')
->visible(fn ($r) => $r->status !== 'paid')
->visible(fn ($record) => $record->status !== 'paid')
->requiresConfirmation()
->action(function (Subscription $r) {
$r->update(['status' => 'paid', 'paid_at' => now()]);
->action(function (Subscription $record) {
$record->update(['status' => 'paid', 'paid_at' => now()]);
// Auto-extend company subscription
$r->company->update([
$record->company->update([
'status' => 'active',
'active_until' => $r->period_end,
'active_until' => $record->period_end,
]);
Notification::make()->title('Plată confirmată. Abonament extins până la ' . $r->period_end->format('d.m.Y'))->success()->send();
Notification::make()->title('Plată confirmată. Abonament extins până la ' . $record->period_end->format('d.m.Y'))->success()->send();
}),
Actions\EditAction::make(),
Actions\DeleteAction::make(),
@@ -93,7 +93,7 @@ class SuperAdminResource extends Resource
Tables\Columns\IconColumn::make('app_authentication_secret')
->label('2FA')
->boolean()
->getStateUsing(fn ($r) => $r->app_authentication_secret !== null)
->getStateUsing(fn ($record) => $record?->app_authentication_secret !== null)
->trueIcon('heroicon-o-lock-closed')
->falseIcon('heroicon-o-lock-open')
->trueColor('success')
@@ -114,18 +114,18 @@ class SuperAdminResource extends Resource
Forms\Components\TextInput::make('new_password')
->password()->required()->minLength(8)->revealable(),
])
->action(function (SuperAdmin $r, array $data) {
$r->update(['password' => Hash::make($data['new_password'])]);
->action(function (SuperAdmin $record, array $data) {
$record->update(['password' => Hash::make($data['new_password'])]);
Notification::make()->title('Parolă resetată.')->success()->send();
}),
Actions\Action::make('toggle_2fa')
->label(fn ($r) => $r->app_authentication_secret ? 'Dezactivează 2FA' : 'Forțează re-setare 2FA')
->label(fn (SuperAdmin $record) => $record->app_authentication_secret ? 'Dezactivează 2FA' : 'Forțează re-setare 2FA')
->icon('heroicon-m-lock-open')
->color('danger')
->visible(fn ($r) => $r->app_authentication_secret !== null)
->visible(fn (SuperAdmin $record) => $record->app_authentication_secret !== null)
->requiresConfirmation()
->action(function (SuperAdmin $r) {
$r->forceFill([
->action(function (SuperAdmin $record) {
$record->forceFill([
'app_authentication_secret' => null,
'app_authentication_recovery_codes' => null,
'email_authentication_at' => null,
@@ -134,8 +134,8 @@ class SuperAdminResource extends Resource
}),
Actions\EditAction::make(),
Actions\DeleteAction::make()
->before(function (SuperAdmin $r) {
if (auth('central')->id() === $r->id) {
->before(function (SuperAdmin $record) {
if (auth('central')->id() === $record->id) {
Notification::make()->title('Nu te poți șterge pe tine!')->danger()->send();
return false;
}
@@ -29,10 +29,10 @@ class PendingPayments extends BaseWidget
Tables\Columns\TextColumn::make('invoice_number')
->label('Factură')
->copyable()
->url(fn ($r) => SubscriptionResource::getUrl('edit', ['record' => $r])),
->url(fn ($record) => SubscriptionResource::getUrl('edit', ['record' => $record])),
Tables\Columns\TextColumn::make('company.name')->label('Companie'),
Tables\Columns\TextColumn::make('plan.name')->placeholder('—'),
Tables\Columns\TextColumn::make('amount')->money(fn ($r) => $r->currency)->weight('bold'),
Tables\Columns\TextColumn::make('amount')->money(fn ($record) => $record->currency)->weight('bold'),
Tables\Columns\TextColumn::make('status')
->badge()
->color(fn ($s) => $s === 'overdue' ? 'danger' : 'warning')
@@ -40,7 +40,7 @@ class PendingPayments extends BaseWidget
Tables\Columns\TextColumn::make('due_at')
->label('Scadent')
->dateTime()
->color(fn ($r) => $r->due_at && $r->due_at->isPast() ? 'danger' : null),
->color(fn ($record) => $record->due_at && $record->due_at->isPast() ? 'danger' : null),
])
->emptyStateHeading('🎉 Toate facturile sunt plătite')
->paginated(false);
@@ -22,7 +22,7 @@ class RecentTenants extends BaseWidget
->columns([
Tables\Columns\TextColumn::make('slug')
->copyable()
->url(fn (Company $r) => CompanyResource::getUrl('view', ['record' => $r])),
->url(fn (Company $record) => CompanyResource::getUrl('view', ['record' => $record])),
Tables\Columns\TextColumn::make('name')->weight('bold'),
Tables\Columns\TextColumn::make('status')
->badge()
@@ -67,9 +67,22 @@ class CentralPanelProvider extends PanelProvider
<meta name="apple-mobile-web-app-capable" content="yes">
<meta name="apple-mobile-web-app-status-bar-style" content="default">
<meta name="apple-mobile-web-app-title" content="AutoCRM Admin">
<link rel="icon" type="image/svg+xml" href="data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 100 100'><rect width='100' height='100' rx='20' fill='%236366f1'/><text x='50' y='66' font-size='52' text-anchor='middle' fill='%23fff' font-family='sans-serif' font-weight='bold'>A</text></svg>">
<link rel="icon" type="image/svg+xml" href="/pwa/admin-icon.svg">
<link rel="apple-touch-icon" href="/pwa/admin-512.png">
BLADE)
)
->renderHook(
PanelsRenderHook::BODY_END,
fn (): string => <<<'HTML'
<script>
if ('serviceWorker' in navigator) {
window.addEventListener('load', () => {
navigator.serviceWorker.register('/admin-sw.js', { scope: '/' }).catch(() => {});
});
}
</script>
HTML
)
->middleware([
EncryptCookies::class,
AddQueuedCookiesToResponse::class,
+96 -2
View File
@@ -54,18 +54,112 @@ Route::get('/admin-manifest.json', function () {
'short_name' => 'AutoCRM',
'description' => 'Panou administrativ AutoCRM SaaS',
'start_url' => '/admin',
'scope' => '/',
'display' => 'standalone',
'orientation' => 'any',
'background_color' => '#ffffff',
'theme_color' => '#6366f1',
'lang' => 'ro',
'icons' => [
['src' => '/pwa/admin-192.png', 'sizes' => '192x192', 'type' => 'image/png'],
['src' => '/pwa/admin-512.png', 'sizes' => '512x512', 'type' => 'image/png'],
['src' => '/pwa/admin-192.png', 'sizes' => '192x192', 'type' => 'image/png', 'purpose' => 'any'],
['src' => '/pwa/admin-512.png', 'sizes' => '512x512', 'type' => 'image/png', 'purpose' => 'any'],
['src' => '/pwa/admin-512.png', 'sizes' => '512x512', 'type' => 'image/png', 'purpose' => 'maskable'],
],
])->header('Cache-Control', 'public, max-age=3600');
});
// SVG favicon for the central panel (referenced from <link rel="icon">)
Route::get('/pwa/admin-icon.svg', function () {
$svg = '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100">
<rect width="100" height="100" rx="20" fill="#6366f1"/>
<text x="50" y="68" font-size="56" text-anchor="middle" fill="#fff" font-family="system-ui,-apple-system,sans-serif" font-weight="700">A</text>
</svg>';
return response($svg, 200, ['Content-Type' => 'image/svg+xml', 'Cache-Control' => 'public, max-age=86400']);
});
// PNG icons generated on-the-fly with GD if available, with PNG fallback baked from SVG.
// Browsers (Chrome) require real PNG bytes for the install prompt.
Route::get('/pwa/admin-{size}.png', function (int $size) {
if (! in_array($size, [192, 512], true)) abort(404);
if (! extension_loaded('gd')) {
// Fallback: serve a transparent 1x1 PNG. Install prompt may not show.
return response(base64_decode('iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPhfDwAChwGA60e6kgAAAABJRU5ErkJggg=='),
200, ['Content-Type' => 'image/png']);
}
$img = imagecreatetruecolor($size, $size);
$bg = imagecolorallocate($img, 99, 102, 241); // #6366f1
$fg = imagecolorallocate($img, 255, 255, 255);
imagefilledrectangle($img, 0, 0, $size, $size, $bg);
$letter = 'A';
$fontSize = (int) ($size * 0.55);
// Use bundled GD font 5 (largest built-in) repeated.
if (function_exists('imagettftext')) {
$font = '/usr/share/fonts/truetype/dejavu/DejaVuSans-Bold.ttf';
if (file_exists($font)) {
$box = imagettfbbox($fontSize, 0, $font, $letter);
$w = $box[2] - $box[0];
$h = $box[1] - $box[7];
$x = (int) (($size - $w) / 2);
$y = (int) (($size + $h) / 2);
imagettftext($img, $fontSize, 0, $x, $y, $fg, $font, $letter);
}
} else {
// Fallback: built-in font
$charW = imagefontwidth(5);
$charH = imagefontheight(5);
$scale = max(1, (int) ($size / 8));
imagestring($img, 5, (int) (($size - $charW * $scale) / 2), (int) (($size - $charH * $scale) / 2), $letter, $fg);
}
ob_start();
imagepng($img);
$png = ob_get_clean();
imagedestroy($img);
return response($png, 200, [
'Content-Type' => 'image/png',
'Cache-Control' => 'public, max-age=86400',
]);
})->where('size', '\d+');
// Service worker pentru PWA central (necesar pentru prompt-ul de install).
// Header `Service-Worker-Allowed: /admin` permite SW-ului servit de la root
// să controleze scope-ul `/admin/*` (cerut de manifest).
Route::get('/admin-sw.js', function () {
return response(<<<'JS'
const CACHE = 'autocrm-admin-shell-v1';
const SHELL = ['/admin-manifest.json'];
self.addEventListener('install', e => {
e.waitUntil(caches.open(CACHE).then(c => c.addAll(SHELL).catch(() => {})));
self.skipWaiting();
});
self.addEventListener('activate', e => {
e.waitUntil(caches.keys().then(keys =>
Promise.all(keys.filter(k => k !== CACHE).map(k => caches.delete(k)))
));
self.clients.claim();
});
self.addEventListener('fetch', e => {
if (e.request.method !== 'GET') return;
const u = new URL(e.request.url);
if (u.pathname.startsWith('/build/') || u.pathname.startsWith('/pwa/')) {
e.respondWith(caches.match(e.request).then(m => m || fetch(e.request).then(r => {
const copy = r.clone();
caches.open(CACHE).then(c => c.put(e.request, copy));
return r;
}).catch(() => caches.match(e.request))));
}
});
JS, 200, [
'Content-Type' => 'application/javascript',
'Cache-Control' => 'public, max-age=3600',
'Service-Worker-Allowed' => '/',
]);
});
// PWA — manifest dinamic per tenant.
Route::get('/manifest.json', function (Request $request) {
$tenant = app(TenantManager::class)->current();