Compare commits

...

30 Commits
3.1.3 ... 3.31

Author SHA1 Message Date
Dennis
6556cf017a wip 2023-11-22 12:50:30 +01:00
Dennis
514c010804 fix 2023-11-20 10:12:21 +01:00
Dennis
b3a5624ad4 fx 2023-11-20 10:08:48 +01:00
Dennis
3857aa33d2 wip 2023-11-20 10:07:09 +01:00
Dennis
c625a5c967 rename this method 2023-11-20 10:06:19 +01:00
Dennis
b4d35adfb4 wip 2023-10-18 08:19:48 +02:00
Dennis
0c1d970a9c cleaning 2023-10-17 16:22:45 +02:00
Dennis
3df600bcda wip 2023-10-07 09:50:11 +02:00
Dennis Smink
c9e0bb3bda Merge pull request #24 from ploi/rjs/provider-plans-per-package
RJS/Assign provider plans to specific packages
2023-10-02 11:14:26 +02:00
Ralph J. Smit
3aae5068ce Simplify 2023-09-28 20:58:54 +02:00
Ralph J. Smit
255353763f Add clarifying comment 2023-09-28 20:57:58 +02:00
Ralph J. Smit
99a49848ca Translations 2023-09-28 20:55:20 +02:00
Ralph J. Smit
def9e3c722 Style 2023-09-28 20:50:07 +02:00
Ralph J. Smit
6cc46cf652 Allow limiting provider plans per package 2023-09-28 20:48:12 +02:00
Ralph J. Smit
9ac72ffda8 Fix headerActions package resource 2023-09-28 13:36:32 +02:00
Dennis
3f1bdb1d8e wip 2023-09-28 13:29:45 +02:00
Dennis
f3d2b0c71f wip 2023-09-28 13:28:51 +02:00
Dennis
4071ba6d49 wip 2023-09-28 13:25:32 +02:00
Dennis Smink
4ec50d7ca1 Merge pull request #23 from ploi/rjs/filament-v3
RJS/Filament V3 upgrade
2023-09-28 13:23:37 +02:00
Ralph J. Smit
51b7f28634 Style 2023-09-28 12:15:55 +02:00
Ralph J. Smit
7f6b59cd4f WIP 2023-09-28 12:15:33 +02:00
Ralph J. Smit
80b4428b72 Filament V3 2023-09-28 12:12:04 +02:00
Dennis
010ecd63ac Merge branch 'develop' of https://github.com/ploi/ploi-core into develop 2023-09-28 08:21:51 +02:00
Dennis
741104de05 wip 2023-09-28 08:21:44 +02:00
Dennis Smink
5254ca3ebe Merge pull request #22 from ploi/rjs/fix-laravel-data
RJS/Upgrade to Laravel 10 – Fix testsuite / Laravel Data implementation
2023-09-27 08:44:45 +02:00
Ralph J. Smit
aefbb5be33 Put attributes on single line 2023-09-26 23:48:48 +02:00
Ralph J. Smit
d22bb52f35 Apply style 2023-09-26 23:47:24 +02:00
Ralph J. Smit
088d951bea Fix Laravel Data/testsuite 2023-09-26 23:44:58 +02:00
Dennis
01fe642a9d wip 2023-08-31 08:18:14 +02:00
Dennis
258e7127f7 wip upgrade 2023-04-03 10:45:59 +02:00
239 changed files with 15638 additions and 13199 deletions

View File

@@ -11,7 +11,7 @@ jobs:
fail-fast: false fail-fast: false
matrix: matrix:
os: [ubuntu-latest] os: [ubuntu-latest]
php: [8.1] php: [8.2]
runs-on: ${{ matrix.os }} runs-on: ${{ matrix.os }}
steps: steps:

View File

@@ -1,6 +1,6 @@
# Ploi Core # Ploi Core
With Ploi Core, you'll power-launch your webhosting company. With Ploi Core, you'll power-launch your webhosting company.
Using the ploi.io system as backbone you will be able to serve your customers your custom panel & feeling. Using the ploi.io system as backbone you will be able to serve your customers your custom panel & feeling.
<p align="center"><img src="https://ploi-core.io/images/og.jpg" width="100%"></p> <p align="center"><img src="https://ploi-core.io/images/og.jpg" width="100%"></p>
@@ -17,4 +17,4 @@ https://ploi.io
The contribution guide can be found inside our documentation: The contribution guide can be found inside our documentation:
https://docs.ploi-core.io/getting-started/contribution-guide https://docs.ploi-core.io/261-getting-started/639-contribution-guide

View File

@@ -2,6 +2,7 @@
namespace App\Actions\Server; namespace App\Actions\Server;
use Throwable;
use App\Models\Server; use App\Models\Server;
use App\Services\Ploi\Ploi; use App\Services\Ploi\Ploi;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
@@ -12,9 +13,19 @@ class SynchronizeServerAction
{ {
try { try {
$serverData = Ploi::make()->server()->get($ploiServerId)->getData(); $serverData = Ploi::make()->server()->get($ploiServerId)->getData();
} catch (\Throwable $exception) { } catch (Throwable $exception) {
Notification::make() Notification::make()
->body('An error has occurred: ' . $exception->getMessage()) ->title('An error has occurred: ' . $exception->getMessage())
->danger()
->send();
return null;
}
if(!$serverData){
Notification::make()
->title('Server synchronization')
->body('It was not possible to synchronize servers, it seems the API key has the wrong scopes. Please make sure the Ploi API key you\'ve entered has all the scopes enabled.')
->danger() ->danger()
->send(); ->send();
@@ -33,9 +44,9 @@ class SynchronizeServerAction
'internal_ip' => $serverData->internal_ip, 'internal_ip' => $serverData->internal_ip,
'available_php_versions' => $serverData->installed_php_versions, 'available_php_versions' => $serverData->installed_php_versions,
]); ]);
} catch (\Throwable $exception) { } catch (Throwable $exception) {
Notification::make() Notification::make()
->body('An error has occurred: ' . $exception->getMessage()) ->title('An error has occurred: ' . $exception->getMessage())
->danger() ->danger()
->send(); ->send();
@@ -43,7 +54,7 @@ class SynchronizeServerAction
} }
Notification::make() Notification::make()
->body(__('Server :server synchronized successfully.', ['server' => $server->name])) ->title(__('Server :server synchronized successfully.', ['server' => $server->name]))
->success() ->success()
->send(); ->send();

View File

@@ -46,7 +46,7 @@ class SynchronizeSiteAction
} }
Notification::make() Notification::make()
->body(__('Site :site synchronized successfully.', ['site' => $site->domain])) ->title(__('Site :site synchronized successfully.', ['site' => $site->domain]))
->success() ->success()
->send(); ->send();

View File

@@ -43,8 +43,8 @@ class Install extends Command
$this->writeSeparationLine(); $this->writeSeparationLine();
$this->info('Make sure to also setup emailing, the cronjob and the queue worker.'); $this->info('Make sure to also setup emailing, the cronjob and the queue worker.');
$this->line(' '); $this->line(' ');
$this->info('Setting up emailing: https://docs.ploi-core.io/getting-started/setting-up-email'); $this->info('Setting up emailing: https://docs.ploi-core.io/261-getting-started/918-setting-up-email');
$this->info('Setting up cronjob & queue worker: https://docs.ploi-core.io/getting-started/installation'); $this->info('Setting up cronjob & queue worker: https://docs.ploi-core.io/261-getting-started/638-installation');
$this->writeSeparationLine(); $this->writeSeparationLine();
$this->line(' '); $this->line(' ');
$this->info('Visit your platform at ' . env('APP_URL')); $this->info('Visit your platform at ' . env('APP_URL'));

View File

@@ -26,24 +26,17 @@ class ServerData extends Data
public ?int $id = null, public ?int $id = null,
#[StringType] #[StringType]
public ?string $status = null, public ?string $status = null,
#[StringType, #[StringType, AlphaDash, Max(40)]
AlphaDash,
Max(40)]
public string $name, public string $name,
#[NotIn(0), #[NotIn(0), Exists(Provider::class, 'id')]
Exists(Provider::class, 'id')]
public int $provider_id, public int $provider_id,
#[NotIn(0), #[NotIn(0), Exists(ProviderRegion::class, 'id')]
Exists(ProviderRegion::class, 'id')]
public int $provider_region_id, public int $provider_region_id,
#[NotIn(0), #[NotIn(0), Exists(ProviderPlan::class, 'id')]
Exists(ProviderPlan::class, 'id')]
public int $provider_plan_id, public int $provider_plan_id,
#[StringType, #[StringType, In(['mysql', 'mariadb', 'postgresql', 'postgresql13'])]
In(['mysql', 'mariadb', 'postgresql', 'postgresql13'])]
public string $database_type, public string $database_type,
#[Exists(User::class, 'id'), #[Exists(User::class, 'id'), IntegerType]
IntegerType]
public ?int $user_id = null, public ?int $user_id = null,
public ?Carbon $created_at = null, public ?Carbon $created_at = null,
) { ) {

View File

@@ -23,14 +23,11 @@ class SiteData extends Data
public function __construct( public function __construct(
public ?int $id = null, public ?int $id = null,
public ?string $status = null, public ?string $status = null,
#[Exists(Server::class, 'id'), #[Exists(Server::class, 'id'), IntegerType]
IntegerType]
public ?int $server_id = null, public ?int $server_id = null,
#[StringType, #[StringType, CustomRule(Hostname::class, ValidateMaximumSites::class)]
CustomRule(Hostname::class, ValidateMaximumSites::class)]
public ?string $domain = null, public ?string $domain = null,
#[Exists(User::class, 'id'), #[Exists(User::class, 'id'), IntegerType]
IntegerType]
public ?int $user_id = null, public ?int $user_id = null,
public ?Carbon $created_at = null, public ?Carbon $created_at = null,
) { ) {

View File

@@ -8,7 +8,7 @@ use Spatie\LaravelData\Support\DataProperty;
class CarbonCast implements Cast class CarbonCast implements Cast
{ {
public function cast(DataProperty $property, mixed $value): mixed public function cast(DataProperty $property, mixed $value, array $context): Carbon
{ {
return Carbon::parse($value); return Carbon::parse($value);
} }

View File

@@ -2,16 +2,13 @@
namespace App\DataTransferObjects\Support; namespace App\DataTransferObjects\Support;
use Illuminate\Support\Enumerable;
use Spatie\LaravelData\DataCollection;
use Illuminate\Pagination\AbstractPaginator;
use Illuminate\Contracts\Pagination\Paginator;
use Illuminate\Pagination\AbstractCursorPaginator;
class Data extends \Spatie\LaravelData\Data class Data extends \Spatie\LaravelData\Data
{ {
public static function collection(Paginator|Enumerable|array|AbstractCursorPaginator|DataCollection|AbstractPaginator $items): \App\DataTransferObjects\Support\DataCollection /**
{ * When working with paginated data, we want to include pagination details in JSON
return new \App\DataTransferObjects\Support\DataCollection(static::class, $items); * responses from the API. However, due to legacy requirements Ploi Core is using
} * a different structure than this package assumes. Therefore, we will override
* the data collection, register a custom transformer and output the structure.
*/
protected static string $_paginatedCollectionClass = PaginatedDataCollection::class;
} }

View File

@@ -2,14 +2,14 @@
namespace App\DataTransferObjects\Support; namespace App\DataTransferObjects\Support;
use Illuminate\Support\Arr; class DataCollectableTransformer extends \Spatie\LaravelData\Transformers\DataCollectableTransformer
class DataCollectionTransformer extends \Spatie\LaravelData\Transformers\DataCollectionTransformer
{ {
protected function wrapPaginatedArray(array $paginated): array protected function wrapPaginatedArray(array $paginated): array
{ {
$wrapKey = $this->wrap->getKey() ?? 'data';
return [ return [
'data' => $paginated['data'], $wrapKey => $paginated['data'],
'links' => [ 'links' => [
'first' => $paginated['first_page_url'], 'first' => $paginated['first_page_url'],
'last' => $paginated['last_page_url'], 'last' => $paginated['last_page_url'],
@@ -27,14 +27,5 @@ class DataCollectionTransformer extends \Spatie\LaravelData\Transformers\DataCol
'total' => $paginated['total'], 'total' => $paginated['total'],
], ],
]; ];
return [
'data' => $paginated['data'],
'links' => $paginated['links'] ?? [],
'meta' => Arr::except($paginated, [
'data',
'links',
]),
];
} }
} }

View File

@@ -1,23 +0,0 @@
<?php
namespace App\DataTransferObjects\Support;
use Spatie\LaravelData\Support\TransformationType;
class DataCollection extends \Spatie\LaravelData\DataCollection
{
public function transform(TransformationType $type): array
{
$transformer = new DataCollectionTransformer(
$this->dataClass,
$type,
$this->getInclusionTree(),
$this->getExclusionTree(),
$this->items,
$this->through,
$this->filter
);
return $transformer->transform();
}
}

View File

@@ -0,0 +1,23 @@
<?php
namespace App\DataTransferObjects\Support;
use Spatie\LaravelData\Support\Wrapping\WrapExecutionType;
class PaginatedDataCollection extends \Spatie\LaravelData\PaginatedDataCollection
{
public function transform(bool $transformValues = true, WrapExecutionType $wrapExecutionType = WrapExecutionType::Disabled, bool $mapPropertyNames = true): array
{
$transformer = new DataCollectableTransformer(
$this->dataClass,
$transformValues,
$wrapExecutionType,
$mapPropertyNames,
$this->getPartialTrees(),
$this->items,
$this->getWrap(),
);
return $transformer->transform();
}
}

View File

@@ -3,10 +3,11 @@
namespace App\DataTransferObjects\Support\Rules; namespace App\DataTransferObjects\Support\Rules;
use Attribute; use Attribute;
use Spatie\LaravelData\Attributes\Validation\ValidationAttribute; use Spatie\LaravelData\Support\Validation\ValidationPath;
use Spatie\LaravelData\Attributes\Validation\CustomValidationAttribute;
#[Attribute(Attribute::TARGET_PROPERTY)] #[Attribute(Attribute::TARGET_PROPERTY | Attribute::TARGET_PARAMETER)]
class CustomRule extends ValidationAttribute class CustomRule extends CustomValidationAttribute
{ {
protected array $rules = []; protected array $rules = [];
@@ -15,10 +16,14 @@ class CustomRule extends ValidationAttribute
$this->rules = $rules; $this->rules = $rules;
} }
public function getRules(): array /**
* @return array<object|string>|object|string
*/
public function getRules(ValidationPath $path): array|object|string
{ {
return collect($this->rules) return array_map(
->map(fn (string $rule) => new $rule()) fn (string $ruleClass) => new $ruleClass(),
->all(); $this->rules
);
} }
} }

View File

@@ -2,13 +2,15 @@
namespace App\DataTransferObjects\Support\Transformers; namespace App\DataTransferObjects\Support\Transformers;
use Illuminate\Support\Carbon;
use Spatie\LaravelData\Support\DataProperty; use Spatie\LaravelData\Support\DataProperty;
use Spatie\LaravelData\Transformers\Transformer; use Spatie\LaravelData\Transformers\Transformer;
class CarbonTransformer implements Transformer class CarbonTransformer implements Transformer
{ {
public function transform(DataProperty $property, mixed $value): mixed public function transform(DataProperty $property, mixed $value): string
{ {
/** @var Carbon $value */
return $value->toISOString(); return $value->toISOString();
} }
} }

View File

@@ -19,16 +19,11 @@ class UserData extends Data
public function __construct( public function __construct(
public ?int $id = null, public ?int $id = null,
public ?string $avatar = null, public ?string $avatar = null,
#[StringType, #[StringType, Max(255)]
Max(255)]
public ?string $name = null, public ?string $name = null,
#[StringType, #[StringType, Email, Max(255), Unique(User::class)]
Email,
Max(255),
Unique(User::class)]
public ?string $email = null, public ?string $email = null,
#[Exists(Package::class, 'id'), #[Exists(Package::class, 'id'), IntegerType]
IntegerType]
public ?int $package_id = null, public ?int $package_id = null,
#[StringType] #[StringType]
public ?string $blocked = null, public ?string $blocked = null,

View File

@@ -2,18 +2,15 @@
namespace App\Filament\Pages; namespace App\Filament\Pages;
use Filament\Forms;
use App\Models\Server; use App\Models\Server;
use App\Models\Package; use App\Models\Package;
use Filament\Pages\Page; use Filament\Pages\Page;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Illuminate\Support\HtmlString; use Illuminate\Support\HtmlString;
use Filament\Forms\Components\Grid;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\Toggle;
use Illuminate\Support\Facades\Storage; use Illuminate\Support\Facades\Storage;
use Filament\Forms\Components\TextInput; use Filament\Forms\Components\TextInput;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use Filament\Forms\Components\FileUpload;
use Filament\Forms\Components\Actions\Action; use Filament\Forms\Components\Actions\Action;
class Settings extends Page class Settings extends Page
@@ -40,15 +37,15 @@ class Settings extends Page
'default_package' => setting('default_package'), 'default_package' => setting('default_package'),
'default_language' => setting('default_language'), 'default_language' => setting('default_language'),
'rotate_logs_after' => setting('rotate_logs_after'), 'rotate_logs_after' => setting('rotate_logs_after'),
'trial' => (bool) setting('trial'), 'trial' => (bool)setting('trial'),
'support' => (bool) setting('support'), 'support' => (bool)setting('support'),
'documentation' => (bool) setting('documentation'), 'documentation' => (bool)setting('documentation'),
'allow_registration' => (bool) setting('allow_registration'), 'allow_registration' => (bool)setting('allow_registration'),
'receive_email_on_server_creation' => (bool) setting('receive_email_on_server_creation'), 'receive_email_on_server_creation' => (bool)setting('receive_email_on_server_creation'),
'receive_email_on_site_creation' => (bool) setting('receive_email_on_site_creation'), 'receive_email_on_site_creation' => (bool)setting('receive_email_on_site_creation'),
'enable_api' => (bool) setting('enable_api'), 'enable_api' => (bool)setting('enable_api'),
'api_token' => setting('api_token'), 'api_token' => setting('api_token'),
'isolate_per_site_per_user' => (bool) setting('isolate_per_site_per_user'), 'isolate_per_site_per_user' => (bool)setting('isolate_per_site_per_user'),
'default_os' => setting('default_os', Server::OS_UBUNTU_22), 'default_os' => setting('default_os', Server::OS_UBUNTU_22),
]); ]);
} }
@@ -56,35 +53,34 @@ class Settings extends Page
public function getFormSchema(): array public function getFormSchema(): array
{ {
return [ return [
Grid::make(2) Forms\Components\Grid::make(2)
->schema([ ->schema([
Forms\Components\Grid::make(2)
Grid::make(1)
->schema([ ->schema([
TextInput::make('name') Forms\Components\TextInput::make('name')
->label(__('Company name')) ->label(__('Company name'))
->required(), ->required(),
TextInput::make('email') Forms\Components\TextInput::make('email')
->label(__('E-mail address')) ->label(__('E-mail address'))
->email(), ->email(),
TextInput::make('support_emails') Forms\Components\TextInput::make('support_emails')
->label(__('Support email address')) ->label(__('Support email address'))
->helperText('Separate by comma to allow more email addresses'), ->helperText('Separate by comma to allow more email addresses'),
]) ])
->columnSpan(2), ->columnSpan(2),
Select::make('default_package') Forms\Components\Select::make('default_package')
->options(fn () => Package::orderBy('name')->get()->mapWithKeys(fn (Package $package) => [$package->id => $package->name])) ->options(fn () => Package::orderBy('name')->pluck('name', 'id'))
->label(__('Select default package')) ->label(__('Select default package'))
->helperText(__('Select the default package a user should get when you create or they register')), ->helperText(__('Select the default package a user should get when you create or they register')),
Select::make('default_language') Forms\Components\Select::make('default_language')
->options(collect(languages())->mapWithKeys(fn (string $language) => [$language => $language])) ->options(collect(languages())->mapWithKeys(fn (string $language) => [$language => $language]))
->label('Select default language') ->label('Select default language')
->helperText('Select the default language a user should get when you create or they register'), ->helperText('Select the default language a user should get when you create or they register'),
FileUpload::make('logo') Forms\Components\FileUpload::make('logo')
->label(__('Logo')) ->label(__('Logo'))
->disk('logos') ->disk('logos')
->columnSpan(2), ->columnSpan(2),
Select::make('rotate_logs_after') Forms\Components\Select::make('rotate_logs_after')
->label(__('This will rotate any logs older than selected, this helps cleanup your database')) ->label(__('This will rotate any logs older than selected, this helps cleanup your database'))
->options([ ->options([
null => __("Don't rotate logs"), null => __("Don't rotate logs"),
@@ -98,7 +94,7 @@ class Settings extends Page
'years-4' => __('Older than 4 years'), 'years-4' => __('Older than 4 years'),
]) ])
->columnSpan(1), ->columnSpan(1),
Select::make('default_os') Forms\Components\Select::make('default_os')
->label(__('Select the default OS that should be used when users create a server')) ->label(__('Select the default OS that should be used when users create a server'))
->default(Server::OS_UBUNTU_22) ->default(Server::OS_UBUNTU_22)
->options([ ->options([
@@ -107,30 +103,30 @@ class Settings extends Page
Server::OS_UBUNTU_22 => __('Ubuntu 22'), Server::OS_UBUNTU_22 => __('Ubuntu 22'),
]) ])
->columnSpan(1), ->columnSpan(1),
Toggle::make('trial') Forms\Components\Toggle::make('trial')
->label(__('Enable trial')) ->label(__('Enable trial'))
->helperText(__('This will allow you to have users with trials.')), ->helperText(__('This will allow you to have users with trials.')),
Toggle::make('allow_registration') Forms\Components\Toggle::make('allow_registration')
->label(__('Allow registration')) ->label(__('Allow registration'))
->helperText(__('Allow customer registration')), ->helperText(__('Allow customer registration')),
Toggle::make('support') Forms\Components\Toggle::make('support')
->label(__('Enable support platform')) ->label(__('Enable support platform'))
->helperText(__('This will allow your customers to make support requests to you.')), ->helperText(__('This will allow your customers to make support requests to you.')),
Toggle::make('documentation') Forms\Components\Toggle::make('documentation')
->label(__('Enable documentation platform')) ->label(__('Enable documentation platform'))
->helperText(__('This will allow you to create articles for your users to look at.')), ->helperText(__('This will allow you to create articles for your users to look at.')),
Toggle::make('receive_email_on_server_creation') Forms\Components\Toggle::make('receive_email_on_server_creation')
->label(__('Receive email when customers create server')) ->label(__('Receive email when customers create server'))
->helperText(__('This will send an email to all admins notifying them about a new server installation.')), ->helperText(__('This will send an email to all admins notifying them about a new server installation.')),
Toggle::make('receive_email_on_site_creation') Forms\Components\Toggle::make('receive_email_on_site_creation')
->label(__('Receive email when customers create site')) ->label(__('Receive email when customers create site'))
->helperText(__('This will send an email to all admins notifying them about a new site installation.')), ->helperText(__('This will send an email to all admins notifying them about a new site installation.')),
Toggle::make('enable_api') Forms\Components\Toggle::make('enable_api')
->label(__('Enable API')) ->label(__('Enable API'))
->helperText(new HtmlString(__('This will allow you to interact with your system via the API. ') . '<a href="https://docs.ploi-core.io/core-api/introduction" target="_blank" class="text-primary-600">' . __('More information') . '</a>')), ->helperText(new HtmlString(__('This will allow you to interact with your system via the API. ') . '<a href="https://docs.ploi-core.io/304-core-api/737-introduction" target="_blank" class="text-primary-600">' . __('More information') . '</a>')),
TextInput::make('api_token') Forms\Components\TextInput::make('api_token')
->label(__('API token')) ->label(__('API token'))
->afterStateHydrated(function (?string $state, TextInput $component) { ->afterStateHydrated(function (?string $state, Forms\Components\TextInput $component) {
$state = filled($state) ? decrypt($state) : null; $state = filled($state) ? decrypt($state) : null;
$component->state($state); $component->state($state);
@@ -148,7 +144,7 @@ class Settings extends Page
->tooltip('Generate'), ->tooltip('Generate'),
]) ])
->suffixAction($generateAction), ->suffixAction($generateAction),
Toggle::make('isolate_per_site_per_user') Forms\Components\Toggle::make('isolate_per_site_per_user')
->label(__('Enable site isolation per site & user')) ->label(__('Enable site isolation per site & user'))
->helperText(__('This will make sure each site created by one user is always isolated from another.')), ->helperText(__('This will make sure each site created by one user is always isolated from another.')),
]), ]),
@@ -178,11 +174,11 @@ class Settings extends Page
Notification::make() Notification::make()
->success() ->success()
->body(__('Settings saved.')) ->title(__('Settings saved.'))
->send(); ->send();
if ($state['logo'] !== $oldLogo || $state['documentation'] !== $oldDocumentation || $state['support'] !== $oldSupport) { if ($state['logo'] !== $oldLogo || $state['documentation'] !== $oldDocumentation || $state['support'] !== $oldSupport) {
$this->redirectRoute('filament.pages.settings'); $this->redirect(Settings::getUrl());
} }
} }
} }

View File

@@ -9,7 +9,7 @@ use Laravel\Horizon\Contracts\MasterSupervisorRepository;
class System extends Page class System extends Page
{ {
protected static ?string $navigationIcon = 'heroicon-o-adjustments'; protected static ?string $navigationIcon = 'heroicon-o-adjustments-vertical';
protected static string $view = 'filament.pages.system'; protected static string $view = 'filament.pages.system';
@@ -33,13 +33,13 @@ class System extends Page
Notification::make() Notification::make()
->success() ->success()
->body(__('Refreshed versions')) ->title(__('Refreshed versions'))
->send(); ->send();
} }
public function getHorizonWorkerStatus(): bool public function getHorizonWorkerStatus(): bool
{ {
return rescue(fn () => (bool) app(MasterSupervisorRepository::class)->all(), false, false); return rescue(fn () => (bool)app(MasterSupervisorRepository::class)->all(), false, false);
} }
public function hasAvailableUpdate(): bool public function hasAvailableUpdate(): bool
@@ -47,7 +47,7 @@ class System extends Page
return app(VersionChecker::class)->getVersions()->isOutOfDate(); return app(VersionChecker::class)->getVersions()->isOutOfDate();
} }
protected static function getNavigationBadge(): ?string public static function getNavigationBadge(): ?string
{ {
$systemChecker = app(VersionChecker::class); $systemChecker = app(VersionChecker::class);

View File

@@ -2,12 +2,11 @@
namespace App\Filament\Pages; namespace App\Filament\Pages;
use Filament\Forms;
use Filament\Actions;
use Filament\Pages\Page; use Filament\Pages\Page;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Filament\Pages\Actions\Action;
use Filament\Forms\Components\Toggle;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use Filament\Forms\Components\MarkdownEditor;
class Terms extends Page class Terms extends Page
{ {
@@ -26,7 +25,7 @@ class Terms extends Page
cache()->forget('core.settings'); cache()->forget('core.settings');
$this->form->fill([ $this->form->fill([
'accept_terms_required' => (bool) setting('accept_terms_required'), 'accept_terms_required' => (bool)setting('accept_terms_required'),
'terms' => setting('terms'), 'terms' => setting('terms'),
'privacy' => setting('privacy'), 'privacy' => setting('privacy'),
]); ]);
@@ -35,20 +34,20 @@ class Terms extends Page
protected function getFormSchema(): array protected function getFormSchema(): array
{ {
return [ return [
Toggle::make('accept_terms_required') Forms\Components\Toggle::make('accept_terms_required')
->label(__(' Require users to accept terms of service on registration')) ->label(__(' Require users to accept terms of service on registration'))
->helperText(__('This will require newly registered users to accept the terms of service.')), ->helperText(__('This will require newly registered users to accept the terms of service.')),
MarkdownEditor::make('terms') Forms\Components\MarkdownEditor::make('terms')
->label(__('Content Terms Of Service')), ->label(__('Content Terms Of Service')),
MarkdownEditor::make('privacy') Forms\Components\MarkdownEditor::make('privacy')
->label(__('Content Privacy Policy')), ->label(__('Content Privacy Policy')),
]; ];
} }
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('load_terms_template') Actions\Action::make('load_terms_template')
->label(__('Load Terms Of Service Template')) ->label(__('Load Terms Of Service Template'))
->action(function (self $livewire) { ->action(function (self $livewire) {
$template = Str::of(file_get_contents(storage_path('templates/terms-of-service.md'))) $template = Str::of(file_get_contents(storage_path('templates/terms-of-service.md')))
@@ -67,7 +66,7 @@ class Terms extends Page
Notification::make() Notification::make()
->success() ->success()
->body(__('Loaded Terms Of Service Template')) ->title(__('Loaded Terms Of Service Template'))
->send(); ->send();
}), }),
]; ];
@@ -88,7 +87,7 @@ class Terms extends Page
Notification::make() Notification::make()
->success() ->success()
->body(__('Terms saved.')) ->title(__('Terms saved.'))
->send(); ->send();
} }
} }

View File

@@ -2,17 +2,14 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Forms;
use Filament\Tables;
use App\Models\Alert; use App\Models\Alert;
use Filament\Forms\Form;
use Filament\Tables\Table;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Filament\Resources\Form;
use Filament\Resources\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Illuminate\Support\HtmlString; use Illuminate\Support\HtmlString;
use Filament\Forms\Components\Select;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Columns\BadgeColumn;
use Filament\Forms\Components\DateTimePicker;
use Filament\Forms\Components\MarkdownEditor;
use App\Filament\Resources\AlertResource\Pages; use App\Filament\Resources\AlertResource\Pages;
class AlertResource extends Resource class AlertResource extends Resource
@@ -31,11 +28,11 @@ class AlertResource extends Resource
{ {
return $form return $form
->schema([ ->schema([
MarkdownEditor::make('message') Forms\Components\MarkdownEditor::make('message')
->label(__('Content')) ->label(__('Content'))
->columnSpan(2) ->columnSpan(2)
->required(), ->required(),
Select::make('type') Forms\Components\Select::make('type')
->label(__('Type')) ->label(__('Type'))
->options([ ->options([
Alert::TYPE_INFO => __('Informational'), Alert::TYPE_INFO => __('Informational'),
@@ -43,9 +40,9 @@ class AlertResource extends Resource
Alert::TYPE_DANGER => __('Danger'), Alert::TYPE_DANGER => __('Danger'),
]) ])
->required(), ->required(),
DateTimePicker::make('expires_at') Forms\Components\DateTimePicker::make('expires_at')
->label(__('Expires at')) ->label(__('Expires at'))
->withoutSeconds(), ->seconds(false),
]); ]);
} }
@@ -53,24 +50,25 @@ class AlertResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
TextColumn::make('message') Tables\Columns\TextColumn::make('message')
->label(__('Content')) ->label(__('Content'))
->formatStateUsing(fn (?string $state) => new HtmlString(Str::markdown($state))), ->formatStateUsing(fn (?string $state) => new HtmlString(Str::markdown($state))),
BadgeColumn::make('type') Tables\Columns\TextColumn::make('type')
->label(__('Type')) ->label(__('Type'))
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Alert::TYPE_INFO => __('Informational'), Alert::TYPE_INFO => __('Informational'),
Alert::TYPE_WARNING => __('Warning'), Alert::TYPE_WARNING => __('Warning'),
Alert::TYPE_DANGER => __('Danger'), Alert::TYPE_DANGER => __('Danger'),
]) })
->colors([ ->colors([
'primary' => Alert::TYPE_INFO, 'primary' => Alert::TYPE_INFO,
'warning' => Alert::TYPE_WARNING, 'warning' => Alert::TYPE_WARNING,
'danger' => Alert::TYPE_DANGER, 'danger' => Alert::TYPE_DANGER,
]), ]),
TextColumn::make('expires_at') Tables\Columns\TextColumn::make('expires_at')
->label('Expires Date') ->label('Expires')
->formatStateUsing(fn (?string $state) => filled($state) ? $state : '-'), ->default('-'),
]); ]);
} }

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\AlertResource\Pages; namespace App\Filament\Resources\AlertResource\Pages;
use Filament\Pages\Actions\DeleteAction; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\AlertResource; use App\Filament\Resources\AlertResource;
@@ -10,10 +10,10 @@ class EditAlert extends EditRecord
{ {
protected static string $resource = AlertResource::class; protected static string $resource = AlertResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
DeleteAction::make(), Actions\DeleteAction::make(),
]; ];
} }
} }

View File

@@ -2,18 +2,21 @@
namespace App\Filament\Resources\AlertResource\Pages; namespace App\Filament\Resources\AlertResource\Pages;
use Filament\Pages\Actions\CreateAction; use Filament\Actions;
use App\Filament\Resources\AlertResource; use App\Filament\Resources\AlertResource;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use Illuminate\Contracts\Support\Htmlable;
class ListAlerts extends ListRecords class ListAlerts extends ListRecords
{ {
protected static string $resource = AlertResource::class; protected static string $resource = AlertResource::class;
protected function getActions(): array protected ?string $subheading = 'Alerts are meant to inform your users about things that are going on. For example server migrations, pricing changes. They will display as top-banner inside the panel.';
protected function getHeaderActions(): array
{ {
return [ return [
CreateAction::make(), Actions\CreateAction::make(),
]; ];
} }
} }

View File

@@ -4,19 +4,19 @@ namespace App\Filament\Resources;
use Filament\Forms; use Filament\Forms;
use Filament\Tables; use Filament\Tables;
use Filament\Forms\Form;
use Filament\Tables\Table;
use App\Models\Certificate; use App\Models\Certificate;
use Filament\Resources\Form;
use Filament\Resources\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Illuminate\Support\HtmlString;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
use App\Filament\Resources\CertificateResource\Pages; use App\Filament\Resources\CertificateResource\Pages;
use Illuminate\Support\HtmlString;
class CertificateResource extends Resource class CertificateResource extends Resource
{ {
protected static ?string $model = Certificate::class; protected static ?string $model = Certificate::class;
protected static ?string $navigationIcon = 'heroicon-o-annotation'; protected static ?string $navigationIcon = 'heroicon-o-chat-bubble-bottom-center-text';
protected static ?string $navigationGroup = 'Site management'; protected static ?string $navigationGroup = 'Site management';
@@ -28,15 +28,12 @@ class CertificateResource extends Resource
->schema([ ->schema([
Forms\Components\TextInput::make('site.name'), Forms\Components\TextInput::make('site.name'),
Forms\Components\TextInput::make('server_id'), Forms\Components\TextInput::make('server_id'),
Forms\Components\TextInput::make('status') Forms\Components\TextInput::make('status'),
->maxLength(255),
Forms\Components\TextInput::make('ploi_id'), Forms\Components\TextInput::make('ploi_id'),
Forms\Components\TextInput::make('domain') Forms\Components\TextInput::make('domain'),
->maxLength(255),
Forms\Components\Textarea::make('certificate'), Forms\Components\Textarea::make('certificate'),
Forms\Components\Textarea::make('private'), Forms\Components\Textarea::make('private'),
Forms\Components\TextInput::make('type') Forms\Components\TextInput::make('type'),
->maxLength(255),
]); ]);
} }
@@ -52,11 +49,12 @@ class CertificateResource extends Resource
->label(__('Main domain')), ->label(__('Main domain')),
Tables\Columns\TextColumn::make('type') Tables\Columns\TextColumn::make('type')
->label('Type'), ->label('Type'),
Tables\Columns\BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Certificate::STATUS_BUSY => __('Busy'), Certificate::STATUS_BUSY => __('Busy'),
Certificate::STATUS_ACTIVE => __('Active'), Certificate::STATUS_ACTIVE => __('Active'),
]) })
->colors([ ->colors([
'warning' => Certificate::STATUS_BUSY, 'warning' => Certificate::STATUS_BUSY,
'success' => Certificate::STATUS_ACTIVE, 'success' => Certificate::STATUS_ACTIVE,
@@ -81,6 +79,7 @@ class CertificateResource extends Resource
]) ])
->actions([ ->actions([
Tables\Actions\EditAction::make(), Tables\Actions\EditAction::make(),
Tables\Actions\DeleteAction::make(),
]) ])
->bulkActions([ ->bulkActions([
Tables\Actions\DeleteBulkAction::make(), Tables\Actions\DeleteBulkAction::make(),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\CertificateResource\Pages; namespace App\Filament\Resources\CertificateResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\CertificateResource; use App\Filament\Resources\CertificateResource;
@@ -10,7 +10,7 @@ class ListCertificates extends ListRecords
{ {
protected static string $resource = CertificateResource::class; protected static string $resource = CertificateResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -4,8 +4,8 @@ namespace App\Filament\Resources;
use Filament\Tables; use Filament\Tables;
use App\Models\Cronjob; use App\Models\Cronjob;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use App\Filament\Resources\CronjobResource\Pages; use App\Filament\Resources\CronjobResource\Pages;
@@ -34,11 +34,12 @@ class CronjobResource extends Resource
Tables\Columns\TextColumn::make('site.domain') Tables\Columns\TextColumn::make('site.domain')
->searchable() ->searchable()
->label(__('Site')), ->label(__('Site')),
Tables\Columns\BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Cronjob::STATUS_BUSY => __('Busy'), Cronjob::STATUS_BUSY => __('Busy'),
Cronjob::STATUS_ACTIVE => __('Active'), Cronjob::STATUS_ACTIVE => __('Active'),
]) })
->colors([ ->colors([
'warning' => Cronjob::STATUS_BUSY, 'warning' => Cronjob::STATUS_BUSY,
'success' => Cronjob::STATUS_ACTIVE, 'success' => Cronjob::STATUS_ACTIVE,

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\CronjobResource\Pages; namespace App\Filament\Resources\CronjobResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\CronjobResource; use App\Filament\Resources\CronjobResource;
@@ -10,7 +10,7 @@ class EditCronjob extends EditRecord
{ {
protected static string $resource = CronjobResource::class; protected static string $resource = CronjobResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\DeleteAction::make(), Actions\DeleteAction::make(),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\CronjobResource\Pages; namespace App\Filament\Resources\CronjobResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\CronjobResource; use App\Filament\Resources\CronjobResource;
@@ -10,7 +10,7 @@ class ListCronjobs extends ListRecords
{ {
protected static string $resource = CronjobResource::class; protected static string $resource = CronjobResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -4,8 +4,8 @@ namespace App\Filament\Resources;
use Filament\Tables; use Filament\Tables;
use App\Models\Database; use App\Models\Database;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use App\Filament\Resources\DatabaseResource\Pages; use App\Filament\Resources\DatabaseResource\Pages;
@@ -13,7 +13,7 @@ class DatabaseResource extends Resource
{ {
protected static ?string $model = Database::class; protected static ?string $model = Database::class;
protected static ?string $navigationIcon = 'heroicon-o-database'; protected static ?string $navigationIcon = 'heroicon-o-circle-stack';
protected static ?string $navigationGroup = 'Site management'; protected static ?string $navigationGroup = 'Site management';
@@ -40,17 +40,17 @@ class DatabaseResource extends Resource
Tables\Columns\TextColumn::make('site.domain') Tables\Columns\TextColumn::make('site.domain')
->label(__('Site')) ->label(__('Site'))
->searchable(), ->searchable(),
Tables\Columns\BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Database::STATUS_BUSY => __('Busy'), Database::STATUS_BUSY => __('Busy'),
Database::STATUS_ACTIVE => __('Active'), Database::STATUS_ACTIVE => __('Active'),
]) })
->colors([ ->colors([
'warning' => Database::STATUS_BUSY, 'warning' => Database::STATUS_BUSY,
'success' => Database::STATUS_ACTIVE, 'success' => Database::STATUS_ACTIVE,
]) ])
->label(__('Status')), ->label(__('Status')),
Tables\Columns\TextColumn::make('created_at') Tables\Columns\TextColumn::make('created_at')
->label(__('Date')) ->label(__('Date'))
->sortable() ->sortable()

View File

@@ -56,7 +56,7 @@ class EditDatabase extends Page
$this->recentlyUpdatedPassword = $data->new_password; $this->recentlyUpdatedPassword = $data->new_password;
Notification::make() Notification::make()
->body(__('Successfully reset database password.')) ->title(__('Successfully reset database password.'))
->success() ->success()
->send(); ->send();

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\DatabaseResource\Pages; namespace App\Filament\Resources\DatabaseResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\DatabaseResource; use App\Filament\Resources\DatabaseResource;
@@ -10,7 +10,7 @@ class ListDatabases extends ListRecords
{ {
protected static string $resource = DatabaseResource::class; protected static string $resource = DatabaseResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -2,17 +2,16 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Forms;
use Filament\Tables;
use Filament\Forms\Form;
use Filament\Tables\Table;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Filament\Resources\Form;
use Filament\Resources\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Illuminate\Support\HtmlString; use Illuminate\Support\HtmlString;
use App\Models\DocumentationCategory; use App\Models\DocumentationCategory;
use Filament\Tables\Columns\TextColumn;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Components\MarkdownEditor;
use App\Filament\Resources\DocumentationCategoryResource\Pages; use App\Filament\Resources\DocumentationCategoryResource\Pages;
use App\Filament\Resources\DocumentationCategoryResource\RelationManagers\DocumentationItemsRelationManager; use App\Filament\Resources\DocumentationCategoryResource\RelationManagers;
class DocumentationCategoryResource extends Resource class DocumentationCategoryResource extends Resource
{ {
@@ -28,21 +27,21 @@ class DocumentationCategoryResource extends Resource
protected static ?string $label = 'Category'; protected static ?string $label = 'Category';
protected static function shouldRegisterNavigation(): bool public static function shouldRegisterNavigation(): bool
{ {
return (bool) setting('documentation'); return (bool)setting('documentation');
} }
public static function form(Form $form): Form public static function form(Form $form): Form
{ {
return $form return $form
->schema([ ->schema([
TextInput::make('title') Forms\Components\TextInput::make('title')
->label(__('Title')) ->label(__('Title'))
->unique(table: DocumentationCategory::class, column: 'title', ignoreRecord: true) ->unique(table: DocumentationCategory::class, column: 'title', ignoreRecord: true)
->required() ->required()
->columnSpan(2), ->columnSpan(2),
MarkdownEditor::make('description') Forms\Components\MarkdownEditor::make('description')
->label(__('Description')) ->label(__('Description'))
->required() ->required()
->columnSpan(2), ->columnSpan(2),
@@ -53,11 +52,11 @@ class DocumentationCategoryResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
TextColumn::make('title') Tables\Columns\TextColumn::make('title')
->searchable() ->searchable()
->sortable() ->sortable()
->label(__('Title')), ->label(__('Title')),
TextColumn::make('description') Tables\Columns\TextColumn::make('description')
->label(__('Description')) ->label(__('Description'))
->formatStateUsing(fn (string $state) => new HtmlString(Str::markdown($state))), ->formatStateUsing(fn (string $state) => new HtmlString(Str::markdown($state))),
]); ]);
@@ -66,7 +65,7 @@ class DocumentationCategoryResource extends Resource
public static function getRelations(): array public static function getRelations(): array
{ {
return [ return [
DocumentationItemsRelationManager::class, RelationManagers\DocumentationItemsRelationManager::class,
]; ];
} }

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\DocumentationCategoryResource\Pages; namespace App\Filament\Resources\DocumentationCategoryResource\Pages;
use Filament\Pages\Actions\DeleteAction; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\DocumentationCategoryResource; use App\Filament\Resources\DocumentationCategoryResource;
@@ -10,10 +10,10 @@ class EditDocumentationCategory extends EditRecord
{ {
protected static string $resource = DocumentationCategoryResource::class; protected static string $resource = DocumentationCategoryResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
DeleteAction::make(), Actions\DeleteAction::make(),
]; ];
} }
} }

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\DocumentationCategoryResource\Pages; namespace App\Filament\Resources\DocumentationCategoryResource\Pages;
use Filament\Pages\Actions\CreateAction; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\DocumentationCategoryResource; use App\Filament\Resources\DocumentationCategoryResource;
@@ -10,10 +10,10 @@ class ListDocumentationCategories extends ListRecords
{ {
protected static string $resource = DocumentationCategoryResource::class; protected static string $resource = DocumentationCategoryResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
CreateAction::make(), Actions\CreateAction::make(),
]; ];
} }
} }

View File

@@ -4,8 +4,8 @@ namespace App\Filament\Resources\DocumentationCategoryResource\RelationManagers;
use Filament\Forms; use Filament\Forms;
use Filament\Tables; use Filament\Tables;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
class DocumentationItemsRelationManager extends RelationManager class DocumentationItemsRelationManager extends RelationManager
@@ -18,7 +18,7 @@ class DocumentationItemsRelationManager extends RelationManager
protected static ?string $pluralLabel = 'Articles'; protected static ?string $pluralLabel = 'Articles';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return $form return $form
->schema([ ->schema([
@@ -28,7 +28,7 @@ class DocumentationItemsRelationManager extends RelationManager
]); ]);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return $table return $table
->columns([ ->columns([

View File

@@ -2,14 +2,12 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Resources\Form; use Filament\Forms;
use Filament\Resources\Table; use Filament\Tables;
use Filament\Forms\Form;
use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use App\Models\DocumentationItem; use App\Models\DocumentationItem;
use Filament\Forms\Components\Select;
use Filament\Tables\Columns\TextColumn;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Components\MarkdownEditor;
use App\Filament\Resources\DocumentationItemResource\Pages; use App\Filament\Resources\DocumentationItemResource\Pages;
class DocumentationItemResource extends Resource class DocumentationItemResource extends Resource
@@ -26,23 +24,23 @@ class DocumentationItemResource extends Resource
protected static ?string $label = 'Article'; protected static ?string $label = 'Article';
protected static function shouldRegisterNavigation(): bool public static function shouldRegisterNavigation(): bool
{ {
return (bool) setting('documentation'); return (bool)setting('documentation');
} }
public static function form(Form $form): Form public static function form(Form $form): Form
{ {
return $form return $form
->schema([ ->schema([
TextInput::make('title') Forms\Components\TextInput::make('title')
->label(__('Title')) ->label(__('Title'))
->required(), ->required(),
Select::make('documentation_category_id') Forms\Components\Select::make('documentation_category_id')
->relationship('category', 'title') ->relationship('category', 'title')
->searchable() ->searchable()
->preload(), ->preload(),
MarkdownEditor::make('content') Forms\Components\MarkdownEditor::make('content')
->label(__('Content')) ->label(__('Content'))
->required() ->required()
->columnSpan(2), ->columnSpan(2),
@@ -53,11 +51,10 @@ class DocumentationItemResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
TextColumn::make('title') Tables\Columns\TextColumn::make('title')
->searchable() ->searchable()
->sortable(), ->sortable(),
Tables\Columns\TextColumn::make('category.title')
TextColumn::make('category.title')
->searchable() ->searchable()
->sortable(), ->sortable(),
]); ]);

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\DocumentationItemResource\Pages; namespace App\Filament\Resources\DocumentationItemResource\Pages;
use Filament\Pages\Actions\DeleteAction; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\DocumentationItemResource; use App\Filament\Resources\DocumentationItemResource;
@@ -10,10 +10,10 @@ class EditDocumentationItem extends EditRecord
{ {
protected static string $resource = DocumentationItemResource::class; protected static string $resource = DocumentationItemResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
DeleteAction::make(), Actions\DeleteAction::make(),
]; ];
} }
} }

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\DocumentationItemResource\Pages; namespace App\Filament\Resources\DocumentationItemResource\Pages;
use Filament\Pages\Actions\CreateAction; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\DocumentationItemResource; use App\Filament\Resources\DocumentationItemResource;
@@ -10,10 +10,10 @@ class ListDocumentationItems extends ListRecords
{ {
protected static string $resource = DocumentationItemResource::class; protected static string $resource = DocumentationItemResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
CreateAction::make(), Actions\CreateAction::make(),
]; ];
} }
} }

View File

@@ -2,25 +2,24 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Tables;
use App\Models\Package;
use Filament\Resources\Form;
use Filament\Resources\Table;
use Filament\Resources\Resource;
use Filament\Forms\Components\Grid;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\Section;
use Filament\Forms\Components\Checkbox;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Components\CheckboxList;
use App\Filament\Resources\PackageResource\Pages; use App\Filament\Resources\PackageResource\Pages;
use App\Filament\Resources\PackageResource\RelationManagers; use App\Filament\Resources\PackageResource\RelationManagers;
use App\Models\Package;
use App\Models\Provider;
use App\Models\ProviderPlan;
use Filament\Forms;
use Filament\Forms\Form;
use Filament\Notifications\Notification;
use Filament\Resources\Resource;
use Filament\Tables;
use Filament\Tables\Table;
use Illuminate\Support\HtmlString;
class PackageResource extends Resource class PackageResource extends Resource
{ {
protected static ?string $model = Package::class; protected static ?string $model = Package::class;
protected static ?string $navigationIcon = 'heroicon-o-color-swatch'; protected static ?string $navigationIcon = 'heroicon-o-swatch';
protected static ?int $navigationSort = 3; protected static ?int $navigationSort = 3;
@@ -28,31 +27,31 @@ class PackageResource extends Resource
{ {
return $form return $form
->schema([ ->schema([
TextInput::make('name') Forms\Components\TextInput::make('name')
->label(__('Name')) ->label(__('Name'))
->required() ->required()
->columnSpan(2), ->columnSpan(2),
TextInput::make('maximum_sites') Forms\Components\TextInput::make('maximum_sites')
->helperText(__('Set to 0 for unlimited')) ->helperText(__('Set to 0 for unlimited'))
->integer() ->integer()
->required(), ->required(),
TextInput::make('maximum_servers') Forms\Components\TextInput::make('maximum_servers')
->helperText(__('Set to 0 for unlimited')) ->helperText(__('Set to 0 for unlimited'))
->integer() ->integer()
->required(), ->required(),
TextInput::make('stripe_plan_id') Forms\Components\TextInput::make('stripe_plan_id')
->helperText(__('Enter the pricing ID from Stripe here') . ' - <a href="https://docs.ploi-core.io/digging-deeper/using-stripe" target="ploi-docs-stripe" class="text-primary-500">How does this work?</a>') ->helperText(new HtmlString(__('Enter the pricing ID from Stripe here') . ' - <a href="https://docs.ploi-core.io/263-digging-deeper/743-using-stripe" target="ploi-docs-stripe" class="text-primary-500">How does this work?</a>'))
->label(__('Stripe ID')) ->label(__('Stripe ID'))
->columnSpan(2), ->columnSpan(2),
TextInput::make('price_monthly') Forms\Components\TextInput::make('price_monthly')
->label(__('Monthly price')) ->label(__('Monthly price'))
->helperText(__('Fill this in if you want it to be monthly payments')) ->helperText(__('Fill this in if you want it to be monthly payments'))
->required(), ->required(),
TextInput::make('price_yearly') Forms\Components\TextInput::make('price_yearly')
->label(__('Yearly price')) ->label(__('Yearly price'))
->helperText(__('Fill this in if you want it to be yearly payments')) ->helperText(__('Fill this in if you want it to be yearly payments'))
->required(), ->required(),
Select::make('currency') Forms\Components\Select::make('currency')
->label(__('Currency')) ->label(__('Currency'))
->options([ ->options([
'usd' => 'USD $', 'usd' => 'USD $',
@@ -67,44 +66,132 @@ class PackageResource extends Resource
'nz' => 'NZD $ (New Zealand Dollar)', 'nz' => 'NZD $ (New Zealand Dollar)',
]) ])
->required(), ->required(),
Grid::make() Forms\Components\Grid::make()
->schema([ ->schema([
Section::make(__('Server permissions')) Forms\Components\Section::make(__('Server permissions'))
->icon(ServerResource::getNavigationIcon())
->schema([ ->schema([
Checkbox::make('server_permissions.create') Forms\Components\Checkbox::make('server_permissions.create')
->reactive() ->reactive()
->label('Allow server creation') ->label('Allow server creation')
->helperText('This will allow users to create servers'), ->helperText('This will allow users to create servers'),
Checkbox::make('server_permissions.update') Forms\Components\Checkbox::make('server_permissions.update')
->label('Allow server updates') ->label('Allow server updates')
->helperText('This will allow users to update servers'), ->helperText('This will allow users to update servers'),
Checkbox::make('server_permissions.delete') Forms\Components\Checkbox::make('server_permissions.delete')
->label('Allow server deletion') ->label('Allow server deletion')
->helperText('This will allow users to delete servers'), ->helperText('This will allow users to delete servers'),
]) ])
->columnSpan(1), ->columnSpan(1),
Section::make(__('Site permissions')) Forms\Components\Section::make(__('Site permissions'))
->icon(SiteResource::getNavigationIcon())
->schema([ ->schema([
Checkbox::make('site_permissions.create') Forms\Components\Checkbox::make('site_permissions.create')
->label('Allow site creation') ->label('Allow site creation')
->helperText('This will allow users to create sites'), ->helperText('This will allow users to create sites'),
Checkbox::make('site_permissions.update') Forms\Components\Checkbox::make('site_permissions.update')
->label('Allow site updates') ->label('Allow site updates')
->helperText('This will allow users to update sites'), ->helperText('This will allow users to update sites'),
Checkbox::make('site_permissions.delete') Forms\Components\Checkbox::make('site_permissions.delete')
->label('Allow site deletion') ->label('Allow site deletion')
->helperText('This will allow users to delete sites'), ->helperText('This will allow users to delete sites'),
]) ])
->columnSpan(1), ->columnSpan(1),
]), ]),
Grid::make() Forms\Components\Grid::make()
->schema([ ->schema([
Section::make(__('Available server providers')) Forms\Components\Section::make(__('Available server providers'))
->description(__('These server providers will be available for users that are attached to this package.')) ->description(__('These server providers will be available for users that are attached to this package.'))
->icon(ProviderResource::getNavigationIcon())
->schema([ ->schema([
CheckboxList::make('providers') Forms\Components\CheckboxList::make('providers')
->relationship('providers', 'name') ->relationship('providers', 'name')
->reactive(),
Forms\Components\Grid::make(1)
->schema([
Forms\Components\Actions::make([
Forms\Components\Actions\Action::make('manage_provider_plans')
->label(__('Manage provider plans'))
->icon('heroicon-o-adjustments-horizontal')
->form(function (Package $record) {
return $record->providers->sortBy('name')->map(function (Provider $provider) {
return Forms\Components\Section::make($provider->label)
->description(__('Select the plans that should be available for this provider on this package.'))
->icon(ProviderResource::getNavigationIcon())
->statePath($provider->id)
->schema([
Forms\Components\Toggle::make('select_specific_provider_plans')
->label(__('Select subset'))
->helperText(__('Check this box if you want to limit the provider plans available on this package.'))
->default(false)
->reactive()
->afterStateUpdated(function (Forms\Components\Toggle $component, Forms\Set $set) use ($provider) {
$set(
path: "provider_plans",
state: $component->getState() ? $provider->plans->pluck('id') : [],
);
}),
Forms\Components\CheckboxList::make("provider_plans")
->label(__('Select plans'))
->options(fn() => $provider->plans->mapWithKeys(fn(ProviderPlan $providerPlan) => [$providerPlan->id => $providerPlan->label ?? $providerPlan->plan_id])->all())
->visible(fn(Forms\Get $get) => $get('select_specific_provider_plans'))
->reactive()
->bulkToggleable()
->columns(2)
])
->collapsible();
})->all();
})
->fillForm(function (Package $record) {
return $record->providers->mapWithKeys(function (Provider $provider) use ($record) {
$providerPlanIds = $record->providerPlans()->whereBelongsTo($provider)->pluck('provider_plans.id');
return [$provider->id => [
'select_specific_provider_plans' => $providerPlanIds->isNotEmpty(),
'provider_plans' => $providerPlanIds->all(),
]];
})->all();
})
->action(function (Package $record, array $data) {
$providerPlanIds = collect($data)
// If `select_specific_provider_plans`, all provider plans are available. It could be that this
// option was deselected, and that we have some left over provider plans in the field that
// is now hidden. We will not include theSE IDs so that they ARE detached automatically.
->where('select_specific_provider_plans', true)
->pluck('provider_plans')
->flatten();
// Detaches provider plans not specifically selected.
$record->providerPlans()->sync($providerPlanIds);
Notification::make()
->title(__('Provider plans saved'))
->success()
->send();
})
->modalSubmitActionLabel(__('Save'))
->color('gray')
->disabled(function (Package $record, Forms\Get $get) {
$providers = collect($get('providers'))
->map(fn(string $id): int => (int)$id)
->sort();
return $record->providers->pluck('id')->map(fn(string $id): int => (int)$id)->sort()->toArray() !== $providers->all();
})
]),
Forms\Components\Placeholder::make('save_warning')
->content(__('You\'ve changed the available server providers. Please save your changes before you can manage the provider plans.'))
->visible(function (Package $record, Forms\Get $get) {
$providers = collect($get('providers'))
->map(fn(string $id): int => (int)$id)
->sort();
return $record->providers->pluck('id')->map(fn(string $id): int => (int)$id)->sort()->toArray() !== $providers->all();
})
->hiddenLabel(),
])
->hiddenOn('create'),
]) ])
->columnSpan(1) ->columnSpan(1)
]) ])
@@ -118,7 +205,9 @@ class PackageResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
Tables\Columns\TextColumn::make('id')->label('ID')->searchable(), Tables\Columns\TextColumn::make('id')
->label('ID')
->searchable(),
Tables\Columns\TextColumn::make('name') Tables\Columns\TextColumn::make('name')
->label(__('Name')) ->label(__('Name'))
->description(function (Package $record) { ->description(function (Package $record) {
@@ -129,10 +218,10 @@ class PackageResource extends Resource
return "Attached to stripe - {$record->price_monthly} {$record->currency}"; return "Attached to stripe - {$record->price_monthly} {$record->currency}";
}), }),
Tables\Columns\TextColumn::make('maximum_sites') Tables\Columns\TextColumn::make('maximum_sites')
->formatStateUsing(fn (int $state) => $state === 0 ? __('Unlimited') : $state) ->formatStateUsing(fn(int $state) => $state === 0 ? __('Unlimited') : $state)
->label(__('Maximum sites')), ->label(__('Maximum sites')),
Tables\Columns\TextColumn::make('maximum_servers') Tables\Columns\TextColumn::make('maximum_servers')
->formatStateUsing(fn (int $state) => $state === 0 ? __('Unlimited') : $state) ->formatStateUsing(fn(int $state) => $state === 0 ? __('Unlimited') : $state)
->label(__('Maximum servers')), ->label(__('Maximum servers')),
Tables\Columns\TextColumn::make('users_count') Tables\Columns\TextColumn::make('users_count')
->counts('users'), ->counts('users'),
@@ -142,6 +231,7 @@ class PackageResource extends Resource
]) ])
->actions([ ->actions([
Tables\Actions\EditAction::make(), Tables\Actions\EditAction::make(),
Tables\Actions\DeleteAction::make(),
]) ])
->bulkActions([ ->bulkActions([
Tables\Actions\DeleteBulkAction::make(), Tables\Actions\DeleteBulkAction::make(),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\PackageResource\Pages; namespace App\Filament\Resources\PackageResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\PackageResource; use App\Filament\Resources\PackageResource;
@@ -10,10 +10,17 @@ class EditPackage extends EditRecord
{ {
protected static string $resource = PackageResource::class; protected static string $resource = PackageResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\DeleteAction::make(), Actions\DeleteAction::make(),
]; ];
} }
public function afterSave(): void
{
// Necessary to refresh, in order to load the updated saved relationships and
// correctly show or hide the "manage provider plans" warning placeholder.
$this->getRecord()->refresh();
}
} }

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\PackageResource\Pages; namespace App\Filament\Resources\PackageResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\PackageResource; use App\Filament\Resources\PackageResource;
@@ -10,7 +10,7 @@ class ListPackages extends ListRecords
{ {
protected static string $resource = PackageResource::class; protected static string $resource = PackageResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -3,8 +3,8 @@
namespace App\Filament\Resources\PackageResource\RelationManagers; namespace App\Filament\Resources\PackageResource\RelationManagers;
use App\Models\User; use App\Models\User;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Tables\Actions\Action; use Filament\Tables\Actions\Action;
use Filament\Forms\Components\Select; use Filament\Forms\Components\Select;
use App\Filament\Resources\UserResource; use App\Filament\Resources\UserResource;
@@ -16,15 +16,16 @@ class UsersRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'name'; protected static ?string $recordTitleAttribute = 'name';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return UserResource::form($form); return UserResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return UserResource::table($table) return UserResource::table($table)
->appendHeaderActions([ ->headerActions([
...$table->getHeaderActions(),
Action::make('add_user') Action::make('add_user')
->label(__('Add user')) ->label(__('Add user'))
->form(fn (self $livewire) => [ ->form(fn (self $livewire) => [

View File

@@ -2,13 +2,13 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Forms;
use Filament\Tables; use Filament\Tables;
use App\Models\Provider; use App\Models\Provider;
use Filament\Forms\Form;
use Filament\Tables\Table;
use App\Models\ProviderPlan; use App\Models\ProviderPlan;
use Filament\Resources\Form;
use Filament\Resources\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Filament\Forms\Components\TextInput;
use App\Filament\Resources\ProviderPlanResource\Pages; use App\Filament\Resources\ProviderPlanResource\Pages;
class ProviderPlanResource extends Resource class ProviderPlanResource extends Resource
@@ -25,7 +25,7 @@ class ProviderPlanResource extends Resource
{ {
return $form return $form
->schema([ ->schema([
TextInput::make('label'), Forms\Components\TextInput::make('label'),
]); ]);
} }
@@ -33,7 +33,9 @@ class ProviderPlanResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
Tables\Columns\TextColumn::make('id')->label('ID')->searchable(), Tables\Columns\TextColumn::make('id')
->label('ID')
->searchable(),
Tables\Columns\TextColumn::make('provider.name') Tables\Columns\TextColumn::make('provider.name')
->label(__('Provider')) ->label(__('Provider'))
->searchable(), ->searchable(),

View File

@@ -9,7 +9,7 @@ class ListProviderPlans extends ListRecords
{ {
protected static string $resource = ProviderPlanResource::class; protected static string $resource = ProviderPlanResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
// //

View File

@@ -4,8 +4,8 @@ namespace App\Filament\Resources;
use Filament\Tables; use Filament\Tables;
use App\Models\Provider; use App\Models\Provider;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Models\ProviderRegion; use App\Models\ProviderRegion;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use App\Filament\Resources\ProviderRegionResource\Pages; use App\Filament\Resources\ProviderRegionResource\Pages;
@@ -14,7 +14,7 @@ class ProviderRegionResource extends Resource
{ {
protected static ?string $model = ProviderRegion::class; protected static ?string $model = ProviderRegion::class;
protected static ?string $navigationIcon = 'heroicon-o-globe'; protected static ?string $navigationIcon = 'heroicon-o-globe-americas';
protected static ?string $navigationGroup = 'Providers'; protected static ?string $navigationGroup = 'Providers';
@@ -32,7 +32,9 @@ class ProviderRegionResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
Tables\Columns\TextColumn::make('id')->label('ID')->searchable(), Tables\Columns\TextColumn::make('id')
->label('ID')
->searchable(),
Tables\Columns\TextColumn::make('provider.name') Tables\Columns\TextColumn::make('provider.name')
->label(__('Provider')) ->label(__('Provider'))
->searchable(), ->searchable(),
@@ -46,7 +48,7 @@ class ProviderRegionResource extends Resource
->filters([ ->filters([
Tables\Filters\SelectFilter::make('provider_id') Tables\Filters\SelectFilter::make('provider_id')
->label(__('Provider')) ->label(__('Provider'))
->options(fn () => Provider::orderBy('name')->get()->mapWithKeys(fn (Provider $provider) => [$provider->id => $provider->name])), ->options(fn () => Provider::orderBy('name')->pluck('name', 'id'))
]) ])
->actions([ ->actions([
// //

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\ProviderRegionResource\Pages; namespace App\Filament\Resources\ProviderRegionResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\ProviderRegionResource; use App\Filament\Resources\ProviderRegionResource;
@@ -10,7 +10,7 @@ class ListProviderRegions extends ListRecords
{ {
protected static string $resource = ProviderRegionResource::class; protected static string $resource = ProviderRegionResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -5,9 +5,9 @@ namespace App\Filament\Resources;
use Filament\Forms; use Filament\Forms;
use Filament\Tables; use Filament\Tables;
use App\Models\Provider; use App\Models\Provider;
use Filament\Forms\Form;
use Filament\Tables\Table;
use App\Models\ProviderPlan; use App\Models\ProviderPlan;
use Filament\Resources\Form;
use Filament\Resources\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
@@ -19,7 +19,7 @@ class ProviderResource extends Resource
{ {
protected static ?string $model = Provider::class; protected static ?string $model = Provider::class;
protected static ?string $navigationIcon = 'heroicon-o-cloud-upload'; protected static ?string $navigationIcon = 'heroicon-o-cloud-arrow-up';
protected static ?string $navigationGroup = 'Providers'; protected static ?string $navigationGroup = 'Providers';
@@ -48,7 +48,9 @@ class ProviderResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
Tables\Columns\TextColumn::make('id')->label('ID')->searchable(), Tables\Columns\TextColumn::make('id')
->label('ID')
->searchable(),
Tables\Columns\TextColumn::make('name') Tables\Columns\TextColumn::make('name')
->description(function (Provider $record) { ->description(function (Provider $record) {
return "{$record->plans_count} plan(s) · {$record->regions_count} region(s)"; return "{$record->plans_count} plan(s) · {$record->regions_count} region(s)";
@@ -71,12 +73,12 @@ class ProviderResource extends Resource
Tables\Actions\Action::make('synchronize_provider') Tables\Actions\Action::make('synchronize_provider')
->label(__('Synchronize')) ->label(__('Synchronize'))
->tooltip(__('This will synchronize the latest data from this provider to your Ploi Core installation')) ->tooltip(__('This will synchronize the latest data from this provider to your Ploi Core installation'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->action(function (Provider $record) { ->action(function (Provider $record) {
$provider = app(SynchronizeProviderAction::class)->execute($record->ploi_id); $provider = app(SynchronizeProviderAction::class)->execute($record->ploi_id);
Notification::make() Notification::make()
->body(__('Provider :provider synchronized successfully.', ['provider' => $provider->name])) ->title(__('Provider :provider synchronized successfully.', ['provider' => $provider->name]))
->success() ->success()
->send(); ->send();
}), }),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\ProviderResource\Pages; namespace App\Filament\Resources\ProviderResource\Pages;
use Filament\Pages\Actions\Action; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\ProviderResource; use App\Filament\Resources\ProviderResource;
@@ -14,14 +14,14 @@ class ListProviders extends ListRecords
protected static string $resource = ProviderResource::class; protected static string $resource = ProviderResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('synchronize_providers') Actions\Action::make('synchronize_providers')
->label(__('Synchronize providers')) ->label(__('Synchronize providers'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->color('secondary') ->color('gray')
->url(route('filament.resources.providers.synchronize')), ->url(ProviderResource::getUrl('synchronize')),
]; ];
} }
} }

View File

@@ -20,7 +20,7 @@ class SynchronizeProviders extends Page
]; ];
} }
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
// //

View File

@@ -46,14 +46,14 @@ class AvailableProvidersOverview extends TableWidget
return [ return [
Action::make('synchronize_provider') Action::make('synchronize_provider')
->label(__('Synchronize')) ->label(__('Synchronize'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->action(function (AvailableProvider $record, self $livewire) { ->action(function (AvailableProvider $record, self $livewire) {
$provider = app(SynchronizeProviderAction::class)->execute($record->id); $provider = app(SynchronizeProviderAction::class)->execute($record->id);
$livewire->emit('$refresh'); $livewire->dispatch('$refresh');
Notification::make() Notification::make()
->body(__('Provider :provider synchronized successfully.', ['provider' => $provider->name])) ->title(__('Provider :provider synchronized successfully.', ['provider' => $provider->name]))
->success() ->success()
->send(); ->send();
}), }),

View File

@@ -5,8 +5,8 @@ namespace App\Filament\Resources;
use Filament\Forms; use Filament\Forms;
use Filament\Tables; use Filament\Tables;
use App\Models\Redirect; use App\Models\Redirect;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use App\Filament\Resources\RedirectResource\Pages; use App\Filament\Resources\RedirectResource\Pages;
@@ -14,7 +14,7 @@ class RedirectResource extends Resource
{ {
protected static ?string $model = Redirect::class; protected static ?string $model = Redirect::class;
protected static ?string $navigationIcon = 'heroicon-o-external-link'; protected static ?string $navigationIcon = 'heroicon-o-arrow-top-right-on-square';
protected static ?string $navigationGroup = 'Site management'; protected static ?string $navigationGroup = 'Site management';
@@ -26,15 +26,11 @@ class RedirectResource extends Resource
->schema([ ->schema([
Forms\Components\TextInput::make('site_id'), Forms\Components\TextInput::make('site_id'),
Forms\Components\TextInput::make('server_id'), Forms\Components\TextInput::make('server_id'),
Forms\Components\TextInput::make('status') Forms\Components\TextInput::make('status'),
->maxLength(255),
Forms\Components\TextInput::make('ploi_id'), Forms\Components\TextInput::make('ploi_id'),
Forms\Components\TextInput::make('redirect_from') Forms\Components\TextInput::make('redirect_from'),
->maxLength(255), Forms\Components\TextInput::make('redirect_to'),
Forms\Components\TextInput::make('redirect_to') Forms\Components\TextInput::make('type'),
->maxLength(255),
Forms\Components\TextInput::make('type')
->maxLength(255),
]); ]);
} }
@@ -58,11 +54,12 @@ class RedirectResource extends Resource
->searchable(), ->searchable(),
Tables\Columns\TextColumn::make('type') Tables\Columns\TextColumn::make('type')
->label(__('Type')), ->label(__('Type')),
Tables\Columns\BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Redirect::STATUS_BUSY => __('Busy'), Redirect::STATUS_BUSY => __('Busy'),
Redirect::STATUS_ACTIVE => __('Active'), Redirect::STATUS_ACTIVE => __('Active'),
]) })
->colors([ ->colors([
'warning' => Redirect::STATUS_BUSY, 'warning' => Redirect::STATUS_BUSY,
'success' => Redirect::STATUS_ACTIVE, 'success' => Redirect::STATUS_ACTIVE,

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\RedirectResource\Pages; namespace App\Filament\Resources\RedirectResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\RedirectResource; use App\Filament\Resources\RedirectResource;
@@ -10,7 +10,7 @@ class ListRedirects extends ListRecords
{ {
protected static string $resource = RedirectResource::class; protected static string $resource = RedirectResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -6,8 +6,8 @@ use Filament\Forms;
use App\Models\User; use App\Models\User;
use Filament\Tables; use Filament\Tables;
use App\Models\Server; use App\Models\Server;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Illuminate\Support\HtmlString; use Illuminate\Support\HtmlString;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
@@ -61,12 +61,13 @@ class ServerResource extends Resource
Tables\Columns\TextColumn::make('name') Tables\Columns\TextColumn::make('name')
->label(__('Name')) ->label(__('Name'))
->searchable(), ->searchable(),
Tables\Columns\BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->label(__('Status')) ->label(__('Status'))
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Server::STATUS_BUSY => __('Busy'), Server::STATUS_BUSY => __('Busy'),
Server::STATUS_ACTIVE => __('Active'), Server::STATUS_ACTIVE => __('Active'),
]) })
->colors([ ->colors([
'warning' => Server::STATUS_BUSY, 'warning' => Server::STATUS_BUSY,
'success' => Server::STATUS_ACTIVE, 'success' => Server::STATUS_ACTIVE,
@@ -74,19 +75,27 @@ class ServerResource extends Resource
Tables\Columns\TextColumn::make('users') Tables\Columns\TextColumn::make('users')
->label(__('Users')) ->label(__('Users'))
->wrap() ->wrap()
->getStateUsing(function (Server $record) { ->formatStateUsing(function (Server $record) {
$state = $record $state = $record
->users ->users
->map(function (User $user) { ->map(function (User $user) {
return '<a href="' . route('filament.resources.users.edit', ['record' => $user]) . '" class="text-primary-600">' . $user->name . '</a>'; return '<a href="' . UserResource::getUrl('edit', ['record' => $user]) . '" class="text-primary-600" style="white-space: nowrap">' . $user->name . '</a>';
}) })
->implode(', ') ?: '-'; ->implode(', ') ?: '-';
return new HtmlString($state); return new HtmlString($state);
})
->searchable(query: function (Builder $query, string $search) {
return $query->whereHas('users', function (Builder $query) use ($search) {
return $query
->where('name', 'LIKE', "%{$search}%")
->orWhere('email', 'LIKE', "%{$search}%");
});
}), }),
Tables\Columns\TextColumn::make('maximum_sites') Tables\Columns\TextColumn::make('maximum_sites')
->label(__('Max sites')) ->label(__('Max sites'))
->formatStateUsing(fn (Server $record) => $record->maximum_sites . " (Current: {$record->sites_count})"), ->formatStateUsing(fn (Server $record) => $record->maximum_sites . " (Current: {$record->sites_count})")
->counts('sites'),
Tables\Columns\TextColumn::make('ip') Tables\Columns\TextColumn::make('ip')
->label(__('IP')), ->label(__('IP')),
Tables\Columns\TextColumn::make('created_at') Tables\Columns\TextColumn::make('created_at')
@@ -101,9 +110,10 @@ class ServerResource extends Resource
Tables\Actions\Action::make('synchronize_server') Tables\Actions\Action::make('synchronize_server')
->label(__('Synchronize')) ->label(__('Synchronize'))
->tooltip(__('This will synchronize the latest data from this provider to your Ploi Core installation')) ->tooltip(__('This will synchronize the latest data from this provider to your Ploi Core installation'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->action(fn (Server $record) => app(SynchronizeServerAction::class)->execute($record->ploi_id)) ->action(fn (Server $record) => app(SynchronizeServerAction::class)->execute($record->ploi_id))
->visible(fn (Server $record) => $record->status === Server::STATUS_ACTIVE), ->visible(fn (Server $record) => $record->status === Server::STATUS_ACTIVE),
Tables\Actions\DeleteAction::make(),
]) ])
->bulkActions([ ->bulkActions([
Tables\Actions\DeleteBulkAction::make(), Tables\Actions\DeleteBulkAction::make(),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\ServerResource\Pages; namespace App\Filament\Resources\ServerResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\ServerResource; use App\Filament\Resources\ServerResource;
@@ -10,7 +10,7 @@ class EditServer extends EditRecord
{ {
protected static string $resource = ServerResource::class; protected static string $resource = ServerResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\DeleteAction::make(), Actions\DeleteAction::make(),

View File

@@ -2,36 +2,23 @@
namespace App\Filament\Resources\ServerResource\Pages; namespace App\Filament\Resources\ServerResource\Pages;
use Filament\Pages\Actions\Action; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use Illuminate\Database\Eloquent\Builder;
use App\Filament\Resources\ServerResource; use App\Filament\Resources\ServerResource;
class ListServers extends ListRecords class ListServers extends ListRecords
{ {
protected static string $resource = ServerResource::class; protected static string $resource = ServerResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('synchronize_servers') Actions\Action::make('synchronize_servers')
->label(__('Synchronize servers')) ->label(__('Synchronize servers'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->color('secondary') ->color('gray')
->url(route('filament.resources.servers.synchronize')), ->url(ServerResource::getUrl('synchronize')),
...parent::getActions(), ...parent::getHeaderActions(),
]; ];
} }
protected function applySearchToTableQuery(Builder $query): Builder
{
if (filled($searchTerm = $this->getTableSearchQuery())) {
$query
->where('domain', 'LIKE', "%{$searchTerm}%")
->orWhereHas('users', fn (Builder $query) => $query->where('name', 'LIKE', "%{$searchTerm}%"))
->orWhereHas('users', fn (Builder $query) => $query->where('email', 'LIKE', "%{$searchTerm}%"));
}
return $query;
}
} }

View File

@@ -2,9 +2,9 @@
namespace App\Filament\Resources\ServerResource\Pages; namespace App\Filament\Resources\ServerResource\Pages;
use Filament\Actions;
use App\Models\Server; use App\Models\Server;
use App\Services\Ploi\Ploi; use App\Services\Ploi\Ploi;
use Filament\Pages\Actions\Action;
use Filament\Resources\Pages\Page; use Filament\Resources\Pages\Page;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use App\Filament\Resources\ServerResource; use App\Filament\Resources\ServerResource;
@@ -24,15 +24,15 @@ class SynchronizeServers extends Page
]; ];
} }
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('synchronize_servers') Actions\Action::make('synchronize_servers')
->label(__('Synchronize all servers')) ->label(__('Synchronize all servers'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->requiresConfirmation() ->requiresConfirmation()
->modalHeading('Synchronize servers') ->modalHeading('Synchronize servers')
->modalSubheading('This will synchronize all the servers that are listed in the table, to your Ploi Core installation.') ->modalDescription('This will synchronize all the servers that are listed in the table, to your Ploi Core installation.')
->action(function () { ->action(function () {
$availableServers = Ploi::make()->synchronize()->servers()->getData(); $availableServers = Ploi::make()->synchronize()->servers()->getData();
@@ -51,7 +51,7 @@ class SynchronizeServers extends Page
} }
Notification::make() Notification::make()
->body(__('Servers synchronized successfully.')) ->title(__('Servers synchronized successfully.'))
->success() ->success()
->send(); ->send();
}), }),

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\ServerResource\RelationManagers; namespace App\Filament\Resources\ServerResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\SiteResource; use App\Filament\Resources\SiteResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -23,12 +23,12 @@ class SitesRelationManager extends RelationManager
return __('Sites'); return __('Sites');
} }
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return SiteResource::form($form); return SiteResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return SiteResource::table($table); return SiteResource::table($table);
} }

View File

@@ -3,8 +3,8 @@
namespace App\Filament\Resources\ServerResource\RelationManagers; namespace App\Filament\Resources\ServerResource\RelationManagers;
use Filament\Tables; use Filament\Tables;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\UserResource; use App\Filament\Resources\UserResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -24,18 +24,21 @@ class UsersRelationManager extends RelationManager
return __('Users'); return __('Users');
} }
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return UserResource::form($form); return UserResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return UserResource::table($table) return UserResource::table($table)
->appendHeaderActions([ ->headerActions([
Tables\Actions\AttachAction::make()->preloadRecordSelect(), ...$table->getHeaderActions(),
Tables\Actions\AttachAction::make()
->preloadRecordSelect(),
]) ])
->appendActions([ ->actions([
...$table->getActions(),
Tables\Actions\DetachAction::make(), Tables\Actions\DetachAction::make(),
]); ]);
} }

View File

@@ -44,7 +44,7 @@ class AvailableServersOverview extends TableWidget
return [ return [
Action::make('synchronize_server') Action::make('synchronize_server')
->label(__('Synchronize')) ->label(__('Synchronize'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->action(function (AvailableServer $record) { ->action(function (AvailableServer $record) {
app(SynchronizeServerAction::class)->execute($record->id); app(SynchronizeServerAction::class)->execute($record->id);
}), }),

View File

@@ -6,10 +6,11 @@ use Filament\Forms;
use App\Models\Site; use App\Models\Site;
use App\Models\User; use App\Models\User;
use Filament\Tables; use Filament\Tables;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Illuminate\Support\HtmlString; use Illuminate\Support\HtmlString;
use Illuminate\Database\Eloquent\Builder;
use App\Actions\Site\SynchronizeSiteAction; use App\Actions\Site\SynchronizeSiteAction;
use App\Filament\Resources\SiteResource\Pages; use App\Filament\Resources\SiteResource\Pages;
use App\Filament\Resources\SiteResource\RelationManagers; use App\Filament\Resources\SiteResource\RelationManagers;
@@ -18,7 +19,7 @@ class SiteResource extends Resource
{ {
protected static ?string $model = Site::class; protected static ?string $model = Site::class;
protected static ?string $navigationIcon = 'heroicon-o-code'; protected static ?string $navigationIcon = 'heroicon-o-code-bracket';
protected static ?string $navigationGroup = 'Site management'; protected static ?string $navigationGroup = 'Site management';
@@ -26,9 +27,6 @@ class SiteResource extends Resource
protected static ?string $recordTitleAttribute = 'domain'; protected static ?string $recordTitleAttribute = 'domain';
/**
* @return string|null
*/
public static function getLabel(): ?string public static function getLabel(): ?string
{ {
return __('Site'); return __('Site');
@@ -67,11 +65,12 @@ class SiteResource extends Resource
->label(__('Server')) ->label(__('Server'))
->sortable() ->sortable()
->searchable(), ->searchable(),
Tables\Columns\BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
Site::STATUS_BUSY => __('Busy'), Site::STATUS_BUSY => __('Busy'),
Site::STATUS_ACTIVE => __('Active'), Site::STATUS_ACTIVE => __('Active'),
]) })
->colors([ ->colors([
'warning' => Site::STATUS_BUSY, 'warning' => Site::STATUS_BUSY,
'success' => Site::STATUS_ACTIVE, 'success' => Site::STATUS_ACTIVE,
@@ -79,15 +78,23 @@ class SiteResource extends Resource
->label(__('Status')), ->label(__('Status')),
Tables\Columns\TextColumn::make('users') Tables\Columns\TextColumn::make('users')
->label(__('Users')) ->label(__('Users'))
->getStateUsing(function (Site $record) { ->wrap()
->formatStateUsing(function (Site $record) {
$state = $record $state = $record
->users ->users
->map(function (User $user) { ->map(function (User $user) {
return '<a href="' . route('filament.resources.users.edit', ['record' => $user]) . '" class="text-primary-600">' . $user->name . '</a>'; return '<a href="' . UserResource::getUrl('edit', ['record' => $user]) . '" class="text-primary-600" style="white-space: nowrap">' . $user->name . '</a>';
}) })
->implode(', ') ?: '-'; ->implode(', ') ?: '-';
return new HtmlString($state); return new HtmlString($state);
})
->searchable(query: function (Builder $query, string $search) {
return $query->whereHas('users', function (Builder $query) use ($search) {
return $query
->where('name', 'LIKE', "%{$search}%")
->orWhere('email', 'LIKE', "%{$search}%");
});
}), }),
Tables\Columns\TextColumn::make('created_at') Tables\Columns\TextColumn::make('created_at')
->label(__('Date')) ->label(__('Date'))
@@ -102,11 +109,12 @@ class SiteResource extends Resource
Tables\Actions\Action::make('synchronize_site') Tables\Actions\Action::make('synchronize_site')
->label(__('Synchronize')) ->label(__('Synchronize'))
->tooltip(__('This will synchronize the latest data from this provider to your Ploi Core installation')) ->tooltip(__('This will synchronize the latest data from this provider to your Ploi Core installation'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->action(function (Site $record) { ->action(function (Site $record) {
app(SynchronizeSiteAction::class)->execute($record->server->ploi_id, $record->ploi_id); app(SynchronizeSiteAction::class)->execute($record->server->ploi_id, $record->ploi_id);
}) })
->visible(fn (Site $record) => $record->status === Site::STATUS_ACTIVE), ->visible(fn (Site $record) => $record->status === Site::STATUS_ACTIVE),
Tables\Actions\DeleteAction::make(),
]) ])
->bulkActions([ ->bulkActions([
Tables\Actions\DeleteBulkAction::make(), Tables\Actions\DeleteBulkAction::make(),
@@ -114,7 +122,7 @@ class SiteResource extends Resource
->defaultSort('sites.created_at', 'desc'); ->defaultSort('sites.created_at', 'desc');
} }
public static function getEloquentQuery(): \Illuminate\Database\Eloquent\Builder public static function getEloquentQuery(): Builder
{ {
return parent::getEloquentQuery() return parent::getEloquentQuery()
->with(['users', 'server']); ->with(['users', 'server']);

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\SiteResource\Pages; namespace App\Filament\Resources\SiteResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use App\Filament\Resources\SiteResource; use App\Filament\Resources\SiteResource;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
@@ -10,7 +10,7 @@ class EditSite extends EditRecord
{ {
protected static string $resource = SiteResource::class; protected static string $resource = SiteResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\DeleteAction::make(), Actions\DeleteAction::make(),

View File

@@ -2,11 +2,10 @@
namespace App\Filament\Resources\SiteResource\Pages; namespace App\Filament\Resources\SiteResource\Pages;
use Filament\Actions;
use App\Traits\HasPloi; use App\Traits\HasPloi;
use Filament\Pages\Actions\Action;
use App\Filament\Resources\SiteResource; use App\Filament\Resources\SiteResource;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use Illuminate\Database\Eloquent\Builder;
class ListSites extends ListRecords class ListSites extends ListRecords
{ {
@@ -14,28 +13,15 @@ class ListSites extends ListRecords
protected static string $resource = SiteResource::class; protected static string $resource = SiteResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('synchronize_sites') Actions\Action::make('synchronize_sites')
->label(__('Synchronize sites')) ->label(__('Synchronize sites'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->color('secondary') ->color('gray')
->url(route('filament.resources.sites.synchronize')), ->url(SiteResource::getUrl('synchronize')),
...parent::getHeaderActions()
...parent::getActions()
]; ];
} }
protected function applySearchToTableQuery(Builder $query): Builder
{
if (filled($searchTerm = $this->getTableSearchQuery())) {
$query
->where('domain', 'LIKE', "%{$searchTerm}%")
->orWhereHas('users', fn (Builder $query) => $query->where('name', 'LIKE', "%{$searchTerm}%"))
->orWhereHas('users', fn (Builder $query) => $query->where('email', 'LIKE', "%{$searchTerm}%"));
}
return $query;
}
} }

View File

@@ -3,9 +3,9 @@
namespace App\Filament\Resources\SiteResource\Pages; namespace App\Filament\Resources\SiteResource\Pages;
use App\Models\Site; use App\Models\Site;
use Filament\Actions;
use App\Models\Server; use App\Models\Server;
use App\Services\Ploi\Ploi; use App\Services\Ploi\Ploi;
use Filament\Pages\Actions\Action;
use Filament\Resources\Pages\Page; use Filament\Resources\Pages\Page;
use App\Filament\Resources\SiteResource; use App\Filament\Resources\SiteResource;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
@@ -23,15 +23,15 @@ class SynchronizeSites extends Page
]; ];
} }
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('synchronize_sites') Actions\Action::make('synchronize_sites')
->label(__('Synchronize all sites')) ->label(__('Synchronize all sites'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->requiresConfirmation() ->requiresConfirmation()
->modalHeading('Synchronize sites') ->modalHeading('Synchronize sites')
->modalSubheading('This will synchronize all the sites that are listed in the table, to your Ploi Core installation.') ->modalDescription('This will synchronize all the sites that are listed in the table, to your Ploi Core installation.')
->action(function () { ->action(function () {
$availableSites = Ploi::make()->synchronize()->sites()->getData(); $availableSites = Ploi::make()->synchronize()->sites()->getData();
@@ -52,7 +52,7 @@ class SynchronizeSites extends Page
} }
Notification::make() Notification::make()
->body(__('Sites synchronized successfully.')) ->title(__('Sites synchronized successfully.'))
->success() ->success()
->send(); ->send();
}), }),

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\SiteResource\RelationManagers; namespace App\Filament\Resources\SiteResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\CertificateResource; use App\Filament\Resources\CertificateResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -13,12 +13,12 @@ class CertificatesRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'domain'; protected static ?string $recordTitleAttribute = 'domain';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return CertificateResource::form($form); return CertificateResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return CertificateResource::table($table); return CertificateResource::table($table);
} }

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\SiteResource\RelationManagers; namespace App\Filament\Resources\SiteResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\CronjobResource; use App\Filament\Resources\CronjobResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -13,12 +13,12 @@ class CronjobsRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'command'; protected static ?string $recordTitleAttribute = 'command';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return CronjobResource::form($form); return CronjobResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return CronjobResource::table($table); return CronjobResource::table($table);
} }

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\SiteResource\RelationManagers; namespace App\Filament\Resources\SiteResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\DatabaseResource; use App\Filament\Resources\DatabaseResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -13,12 +13,12 @@ class DatabasesRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'name'; protected static ?string $recordTitleAttribute = 'name';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return DatabaseResource::form($form); return DatabaseResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return DatabaseResource::table($table); return DatabaseResource::table($table);
} }

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\SiteResource\RelationManagers; namespace App\Filament\Resources\SiteResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\RedirectResource; use App\Filament\Resources\RedirectResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -13,12 +13,12 @@ class RedirectsRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'from'; protected static ?string $recordTitleAttribute = 'from';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return RedirectResource::form($form); return RedirectResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return RedirectResource::table($table); return RedirectResource::table($table);
} }

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\SiteResource\RelationManagers; namespace App\Filament\Resources\SiteResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\SiteSystemUserResource; use App\Filament\Resources\SiteSystemUserResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -13,12 +13,12 @@ class SystemUsersRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'user_name'; protected static ?string $recordTitleAttribute = 'user_name';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return SiteSystemUserResource::form($form); return SiteSystemUserResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return SiteSystemUserResource::table($table); return SiteSystemUserResource::table($table);
} }

View File

@@ -2,11 +2,10 @@
namespace App\Filament\Resources\SiteResource\RelationManagers; namespace App\Filament\Resources\SiteResource\RelationManagers;
use Filament\Resources\Form; use Filament\Tables;
use Filament\Resources\Table; use Filament\Forms\Form;
use Filament\Tables\Table;
use App\Filament\Resources\UserResource; use App\Filament\Resources\UserResource;
use Filament\Tables\Actions\AttachAction;
use Filament\Tables\Actions\DetachAction;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
class UsersRelationManager extends RelationManager class UsersRelationManager extends RelationManager
@@ -25,20 +24,22 @@ class UsersRelationManager extends RelationManager
return __('Users'); return __('Users');
} }
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return UserResource::form($form); return UserResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return UserResource::table($table) return UserResource::table($table)
->headerActions([ ->headerActions([
AttachAction::make() ...$table->getHeaderActions(),
Tables\Actions\AttachAction::make()
->preloadRecordSelect(), ->preloadRecordSelect(),
]) ])
->appendActions([ ->actions([
DetachAction::make(), ...$table->getActions(),
Tables\Actions\DetachAction::make(),
]); ]);
} }
} }

View File

@@ -43,7 +43,7 @@ class AvailableSitesOverview extends TableWidget
return [ return [
Action::make('synchronize_site') Action::make('synchronize_site')
->label(__('Synchronize')) ->label(__('Synchronize'))
->icon('heroicon-o-refresh') ->icon('heroicon-o-arrow-path')
->action(function (AvailableSite $record) { ->action(function (AvailableSite $record) {
app(SynchronizeSiteAction::class)->execute(ploiServerId: $record->server_id, ploiSiteId: $record->id); app(SynchronizeSiteAction::class)->execute(ploiServerId: $record->server_id, ploiSiteId: $record->id);
}), }),

View File

@@ -3,8 +3,8 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Tables; use Filament\Tables;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Models\SiteSystemUser; use App\Models\SiteSystemUser;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use App\Filament\Resources\SiteSystemUserResource\Pages; use App\Filament\Resources\SiteSystemUserResource\Pages;

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\SiteSystemUserResource\Pages; namespace App\Filament\Resources\SiteSystemUserResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\SiteSystemUserResource; use App\Filament\Resources\SiteSystemUserResource;
@@ -10,7 +10,7 @@ class ListSiteSystemUsers extends ListRecords
{ {
protected static string $resource = SiteSystemUserResource::class; protected static string $resource = SiteSystemUserResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -3,8 +3,8 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Tables; use Filament\Tables;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Laravel\Cashier\Subscription; use Laravel\Cashier\Subscription;
use App\Filament\Resources\SubscriptionResource\Pages; use App\Filament\Resources\SubscriptionResource\Pages;
@@ -13,7 +13,7 @@ class SubscriptionResource extends Resource
{ {
protected static ?string $model = Subscription::class; protected static ?string $model = Subscription::class;
protected static ?string $navigationIcon = 'heroicon-o-cash'; protected static ?string $navigationIcon = 'heroicon-o-banknotes';
protected static ?int $navigationSort = 4; protected static ?int $navigationSort = 4;
@@ -39,8 +39,9 @@ class SubscriptionResource extends Resource
->url(fn ($record) => UserResource::getUrl('edit', ['record' => $record])), ->url(fn ($record) => UserResource::getUrl('edit', ['record' => $record])),
Tables\Columns\TextColumn::make('stripe_id')->searchable(), Tables\Columns\TextColumn::make('stripe_id')->searchable(),
Tables\Columns\TextColumn::make('stripe_plan')->searchable(), Tables\Columns\TextColumn::make('stripe_plan')->searchable(),
Tables\Columns\BadgeColumn::make('stripe_status') Tables\Columns\TextColumn::make('stripe_status')
->label('Status') ->label('Status')
->badge()
->colors([ ->colors([
'success' => \Stripe\Subscription::STATUS_ACTIVE, 'success' => \Stripe\Subscription::STATUS_ACTIVE,
'warning' => \Stripe\Subscription::STATUS_PAST_DUE, 'warning' => \Stripe\Subscription::STATUS_PAST_DUE,
@@ -55,6 +56,7 @@ class SubscriptionResource extends Resource
]) ])
->actions([ ->actions([
// Tables\Actions\EditAction::make(), // Tables\Actions\EditAction::make(),
Tables\Actions\DeleteAction::make(),
]) ])
->bulkActions([ ->bulkActions([
Tables\Actions\DeleteBulkAction::make(), Tables\Actions\DeleteBulkAction::make(),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\SubscriptionResource\Pages; namespace App\Filament\Resources\SubscriptionResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use App\Filament\Resources\SubscriptionResource; use App\Filament\Resources\SubscriptionResource;
@@ -10,7 +10,7 @@ class EditSubscription extends EditRecord
{ {
protected static string $resource = SubscriptionResource::class; protected static string $resource = SubscriptionResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\DeleteAction::make(), Actions\DeleteAction::make(),

View File

@@ -2,14 +2,12 @@
namespace App\Filament\Resources; namespace App\Filament\Resources;
use Filament\Resources\Form; use Filament\Tables;
use Filament\Forms\Form;
use Filament\Tables\Table;
use App\Models\SupportTicket; use App\Models\SupportTicket;
use Filament\Resources\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Columns\BadgeColumn;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
use Filament\Tables\Filters\MultiSelectFilter;
use App\Filament\Resources\SupportTicketResource\Pages; use App\Filament\Resources\SupportTicketResource\Pages;
class SupportTicketResource extends Resource class SupportTicketResource extends Resource
@@ -18,18 +16,18 @@ class SupportTicketResource extends Resource
protected static ?string $navigationGroup = 'Support'; protected static ?string $navigationGroup = 'Support';
protected static ?string $navigationIcon = 'heroicon-o-support'; protected static ?string $navigationIcon = 'heroicon-o-lifebuoy';
protected static ?string $label = 'Ticket'; protected static ?string $label = 'Ticket';
protected static ?string $pluralLabel = 'Tickets'; protected static ?string $pluralLabel = 'Tickets';
protected static function shouldRegisterNavigation(): bool public static function shouldRegisterNavigation(): bool
{ {
return (bool) setting('support'); return (bool)setting('support');
} }
protected static function getNavigationBadge(): ?string public static function getNavigationBadge(): ?string
{ {
return static::getEloquentQuery()->count(); return static::getEloquentQuery()->count();
} }
@@ -46,32 +44,34 @@ class SupportTicketResource extends Resource
{ {
return $table return $table
->columns([ ->columns([
BadgeColumn::make('status') Tables\Columns\TextColumn::make('status')
->label(__('Status')) ->label(__('Status'))
->enum([ ->badge()
->formatStateUsing(fn (string $state) => match ($state) {
SupportTicket::STATUS_OPEN => __('Open'), SupportTicket::STATUS_OPEN => __('Open'),
SupportTicket::STATUS_CLOSED => __('Closed'), SupportTicket::STATUS_CLOSED => __('Closed'),
SupportTicket::STATUS_CUSTOMER_REPLY => __('Customer Reply'), SupportTicket::STATUS_CUSTOMER_REPLY => __('Customer Reply'),
SupportTicket::STATUS_SUPPORT_REPLY => __('Support Reply'), SupportTicket::STATUS_SUPPORT_REPLY => __('Support Reply'),
]) })
->colors([ ->colors([
'primary' => [SupportTicket::STATUS_OPEN, SupportTicket::STATUS_SUPPORT_REPLY, SupportTicket::STATUS_CUSTOMER_REPLY], 'primary' => [SupportTicket::STATUS_OPEN, SupportTicket::STATUS_SUPPORT_REPLY, SupportTicket::STATUS_CUSTOMER_REPLY],
'danger' => SupportTicket::STATUS_CLOSED, 'danger' => SupportTicket::STATUS_CLOSED,
]) ])
->wrap(false), ->wrap(false),
TextColumn::make('title') Tables\Columns\TextColumn::make('title')
->searchable() ->searchable()
->sortable(), ->sortable(),
TextColumn::make('replies_count') Tables\Columns\TextColumn::make('replies_count')
->label(__('Replies')) ->label(__('Replies'))
->getStateUsing(fn (SupportTicket $record) => $record->replies->count()), ->getStateUsing(fn (SupportTicket $record) => $record->replies->count()),
TextColumn::make('user.name') Tables\Columns\TextColumn::make('user.name')
->searchable() ->searchable()
->sortable(), ->sortable(),
]) ])
->filters([ ->filters([
MultiSelectFilter::make('status') Tables\Filters\SelectFilter::make('status')
->label(__('Status')) ->label(__('Status'))
->multiple()
->options([ ->options([
SupportTicket::STATUS_OPEN => __('Open'), SupportTicket::STATUS_OPEN => __('Open'),
SupportTicket::STATUS_CLOSED => __('Closed'), SupportTicket::STATUS_CLOSED => __('Closed'),

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\SupportTicketResource\Pages; namespace App\Filament\Resources\SupportTicketResource\Pages;
use Filament\Pages\Actions\CreateAction; use Filament\Actions;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
use App\Filament\Resources\SupportTicketResource; use App\Filament\Resources\SupportTicketResource;
@@ -10,10 +10,10 @@ class ListSupportTickets extends ListRecords
{ {
protected static string $resource = SupportTicketResource::class; protected static string $resource = SupportTicketResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
CreateAction::make(), Actions\CreateAction::make(),
]; ];
} }
} }

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\SupportTicketResource\Pages; namespace App\Filament\Resources\SupportTicketResource\Pages;
use Filament\Actions;
use App\Models\SupportTicket; use App\Models\SupportTicket;
use Filament\Pages\Actions\Action;
use Filament\Resources\Pages\Page; use Filament\Resources\Pages\Page;
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Mail; use Illuminate\Support\Facades\Mail;
@@ -31,36 +31,36 @@ class ViewSupportTicket extends Page
return __('View ticket') . ': ' . $this->record->title; return __('View ticket') . ': ' . $this->record->title;
} }
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Action::make('close') Actions\Action::make('close')
->label(__('Close')) ->label(__('Close'))
->action(function (self $livewire) { ->action(function (self $livewire) {
$livewire->record->status = SupportTicket::STATUS_CLOSED; $livewire->record->status = SupportTicket::STATUS_CLOSED;
$livewire->record->save(); $livewire->record->save();
Notification::make() Notification::make()
->body(__('Ticket closed')) ->title(__('Ticket closed'))
->success() ->success()
->send(); ->send();
$livewire->redirectRoute('filament.resources.support-tickets.view', $livewire->record); $livewire->redirect(SupportTicketResource::getUrl('view', ['record' => $livewire->getRecord()]));
}) })
->visible(fn (self $livewire) => $livewire->record->status !== SupportTicket::STATUS_CLOSED) ->visible(fn (self $livewire) => $livewire->record->status !== SupportTicket::STATUS_CLOSED)
->color('danger'), ->color('danger'),
Action::make('reopen') Actions\Action::make('reopen')
->label(__('Reopen')) ->label(__('Reopen'))
->action(function (self $livewire) { ->action(function (self $livewire) {
$livewire->record->status = SupportTicket::STATUS_OPEN; $livewire->record->status = SupportTicket::STATUS_OPEN;
$livewire->record->save(); $livewire->record->save();
Notification::make() Notification::make()
->body(__('Ticket reopened')) ->title(__('Ticket reopened'))
->success() ->success()
->send(); ->send();
$livewire->redirectRoute('filament.resources.support-tickets.view', $livewire->record); $livewire->redirect(SupportTicketResource::getUrl('view', ['record' => $livewire->getRecord()]));
}) })
->visible(fn (self $livewire) => $livewire->record->status === SupportTicket::STATUS_CLOSED) ->visible(fn (self $livewire) => $livewire->record->status === SupportTicket::STATUS_CLOSED)
->color('primary'), ->color('primary'),
@@ -100,10 +100,10 @@ class ViewSupportTicket extends Page
Mail::to($this->record->user)->send(new TicketRepliedToEmail($this->record)); Mail::to($this->record->user)->send(new TicketRepliedToEmail($this->record));
$this->form->fill(); $this->form->fill();
$this->emit('$refresh'); $this->dispatch('$refresh');
Notification::make() Notification::make()
->body(__('Reply sent')) ->title(__('Reply sent'))
->success() ->success()
->send(); ->send();
} }

View File

@@ -5,8 +5,8 @@ namespace App\Filament\Resources;
use Filament\Forms; use Filament\Forms;
use App\Models\User; use App\Models\User;
use Filament\Tables; use Filament\Tables;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use STS\FilamentImpersonate\Impersonate; use STS\FilamentImpersonate\Impersonate;
use App\Filament\Resources\UserResource\Pages; use App\Filament\Resources\UserResource\Pages;
@@ -63,7 +63,7 @@ class UserResource extends Resource
Forms\Components\Select::make('language') Forms\Components\Select::make('language')
->label(__('Language')) ->label(__('Language'))
->default('en') ->default('en')
->options(collect(languages())->mapWithKeys(fn(string $language) => [$language => $language])), ->options(collect(languages())->mapWithKeys(fn (string $language) => [$language => $language])),
Forms\Components\Textarea::make('notes') Forms\Components\Textarea::make('notes')
->label(__('Notes')) ->label(__('Notes'))
->maxLength(65535), ->maxLength(65535),
@@ -79,14 +79,6 @@ class UserResource extends Resource
public static function table(Table $table): Table public static function table(Table $table): Table
{ {
$actions = [];
if (config('core.impersonation')) {
$actions[] = Impersonate::make('impersonate')->tooltip('Login as this user (impersonate)');
}
$actions[] = Tables\Actions\EditAction::make();
return $table return $table
->columns([ ->columns([
Tables\Columns\TextColumn::make('name') Tables\Columns\TextColumn::make('name')
@@ -114,7 +106,13 @@ class UserResource extends Resource
->filters([ ->filters([
// //
]) ])
->actions($actions) ->actions([
Impersonate::make('impersonate')
->tooltip('Login as this user (impersonate)')
->visible(fn () => config('core.impersonation')),
Tables\Actions\EditAction::make(),
Tables\Actions\DeleteAction::make(),
])
->bulkActions([ ->bulkActions([
Tables\Actions\DeleteBulkAction::make(), Tables\Actions\DeleteBulkAction::make(),
]) ])

View File

@@ -2,9 +2,9 @@
namespace App\Filament\Resources\UserResource\Pages; namespace App\Filament\Resources\UserResource\Pages;
use Illuminate\Database\Eloquent\Model;
use App\Filament\Resources\UserResource; use App\Filament\Resources\UserResource;
use Filament\Resources\Pages\CreateRecord; use Filament\Resources\Pages\CreateRecord;
use Illuminate\Database\Eloquent\Model;
class CreateUser extends CreateRecord class CreateUser extends CreateRecord
{ {

View File

@@ -2,13 +2,13 @@
namespace App\Filament\Resources\UserResource\Pages; namespace App\Filament\Resources\UserResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use Filament\Forms\Components\Toggle; use Filament\Forms\Components\Toggle;
use App\Actions\User\DeleteUserAction; use App\Actions\User\DeleteUserAction;
use Illuminate\Database\Eloquent\Model;
use App\Filament\Resources\UserResource; use App\Filament\Resources\UserResource;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use Filament\Resources\Pages\EditRecord; use Filament\Resources\Pages\EditRecord;
use Illuminate\Database\Eloquent\Model;
class EditUser extends EditRecord class EditUser extends EditRecord
{ {
@@ -22,17 +22,17 @@ class EditUser extends EditRecord
return $record; return $record;
} }
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\Action::make('two_factor_authentication') Actions\Action::make('two_factor_authentication')
->label(__('Disable two-factor authentication')) ->label(__('Disable two-factor authentication'))
->color('secondary') ->color('gray')
->action(function () { ->action(function () {
$this->record->disableTwoFactorAuth(); $this->record->disableTwoFactorAuth();
Notification::make() Notification::make()
->body(__('Two-factor authentication disabled')) ->title(__('Two-factor authentication disabled'))
->success() ->success()
->send(); ->send();
}) })
@@ -50,11 +50,11 @@ class EditUser extends EditRecord
app(DeleteUserAction::class)->execute($this->getRecord(), $data['remove_all_data']); app(DeleteUserAction::class)->execute($this->getRecord(), $data['remove_all_data']);
Notification::make() Notification::make()
->body(__('User deleted')) ->title(__('User deleted'))
->success() ->success()
->send(); ->send();
$this->redirectRoute('filament.resources.users.index'); $this->redirect(UserResource::getUrl());
}) })
->color('danger'), ->color('danger'),
]; ];

View File

@@ -2,7 +2,7 @@
namespace App\Filament\Resources\UserResource\Pages; namespace App\Filament\Resources\UserResource\Pages;
use Filament\Pages\Actions; use Filament\Actions;
use App\Filament\Resources\UserResource; use App\Filament\Resources\UserResource;
use Filament\Resources\Pages\ListRecords; use Filament\Resources\Pages\ListRecords;
@@ -10,7 +10,7 @@ class ListUsers extends ListRecords
{ {
protected static string $resource = UserResource::class; protected static string $resource = UserResource::class;
protected function getActions(): array protected function getHeaderActions(): array
{ {
return [ return [
Actions\CreateAction::make(), Actions\CreateAction::make(),

View File

@@ -2,11 +2,9 @@
namespace App\Filament\Resources\UserResource\RelationManagers; namespace App\Filament\Resources\UserResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use App\Filament\Resources\ServerResource; use App\Filament\Resources\ServerResource;
use Illuminate\Database\Eloquent\Relations\Relation;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
class ServersRelationManager extends RelationManager class ServersRelationManager extends RelationManager
@@ -15,19 +13,13 @@ class ServersRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'name'; protected static ?string $recordTitleAttribute = 'name';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return ServerResource::form($form); return ServerResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return ServerResource::table($table); return ServerResource::table($table);
} }
protected function getTableQuery(): Builder|Relation
{
return parent::getTableQuery()
->withCount('sites');
}
} }

View File

@@ -2,8 +2,8 @@
namespace App\Filament\Resources\UserResource\RelationManagers; namespace App\Filament\Resources\UserResource\RelationManagers;
use Filament\Resources\Form; use Filament\Forms\Form;
use Filament\Resources\Table; use Filament\Tables\Table;
use App\Filament\Resources\SiteResource; use App\Filament\Resources\SiteResource;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
@@ -13,12 +13,12 @@ class SitesRelationManager extends RelationManager
protected static ?string $recordTitleAttribute = 'domain'; protected static ?string $recordTitleAttribute = 'domain';
public static function form(Form $form): Form public function form(Form $form): Form
{ {
return SiteResource::form($form); return SiteResource::form($form);
} }
public static function table(Table $table): Table public function table(Table $table): Table
{ {
return SiteResource::table($table); return SiteResource::table($table);
} }

View File

@@ -5,7 +5,9 @@ namespace App\Filament\Widgets;
use App\Models\Site; use App\Models\Site;
use App\Models\User; use App\Models\User;
use App\Models\Server; use App\Models\Server;
use Filament\Widgets\StatsOverviewWidget\Card; use App\Filament\Resources\SiteResource;
use App\Filament\Resources\UserResource;
use App\Filament\Resources\ServerResource;
use Filament\Widgets\StatsOverviewWidget as BaseWidget; use Filament\Widgets\StatsOverviewWidget as BaseWidget;
class StatsOverview extends BaseWidget class StatsOverview extends BaseWidget
@@ -13,14 +15,14 @@ class StatsOverview extends BaseWidget
protected function getCards(): array protected function getCards(): array
{ {
return [ return [
Card::make(__('Servers'), Server::count()) BaseWidget\Stat::make(__('Servers'), Server::count())
->url(route('filament.resources.servers.index')) ->url(ServerResource::getUrl())
->icon('heroicon-o-server'), ->icon('heroicon-o-server'),
Card::make(__('Sites'), Site::count()) BaseWidget\Stat::make(__('Sites'), Site::count())
->url(route('filament.resources.sites.index')) ->url(SiteResource::getUrl())
->icon('heroicon-o-globe-alt'), ->icon('heroicon-o-globe-alt'),
Card::make(__('Users'), User::count()) BaseWidget\Stat::make(__('Users'), User::count())
->url(route('filament.resources.users.index')) ->url(UserResource::getUrl())
->icon('heroicon-o-user'), ->icon('heroicon-o-user'),
]; ];
} }

View File

@@ -2,8 +2,9 @@
namespace App\Filament\Widgets; namespace App\Filament\Widgets;
use Filament\Tables;
use App\Models\SystemLog; use App\Models\SystemLog;
use Filament\Tables\Columns\TextColumn; use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
use Filament\Widgets\TableWidget as BaseWidget; use Filament\Widgets\TableWidget as BaseWidget;
@@ -11,27 +12,28 @@ class SystemLogs extends BaseWidget
{ {
protected int|string|array $columnSpan = 'full'; protected int|string|array $columnSpan = 'full';
protected int $defaultTableRecordsPerPageSelectOption = 10; public function table(Table $table): Table
protected function getTableQuery(): Builder
{ {
return SystemLog::query() return $table
->latest() ->query(fn (): Builder => SystemLog::query()->with('model'))
->with('model'); ->defaultSort(fn (Builder $query) => $query->latest())
} ->columns([
Tables\Columns\TextColumn::make('title')
protected function getTableColumns(): array ->label(__('Title'))
{ ->formatStateUsing(fn (SystemLog $record) => __($record->title, [
return [ 'site' => $record->model->domain ?? '-Unknown-',
TextColumn::make(__('Title')) 'database' => $record->model->name ?? '-Unknown-',
->formatStateUsing(fn (SystemLog $record) => __($record->title, [ ]))
'site' => $record->model->domain ?? '-Unknown-', ->description(fn (SystemLog $record) => __($record->description, [
'database' => $record->model->name ?? '-Unknown-', 'site' => $record->model->domain ?? '-Unknown-',
])) 'database' => $record->model->name ?? '-Unknown-',
->description(fn (SystemLog $record) => __($record->description, [ ]))
'site' => $record->model->domain ?? '-Unknown-', ->searchable()
'database' => $record->model->name ?? '-Unknown-', ->sortable(),
])), Tables\Columns\TextColumn::make('created_at')
]; ->label(__('Date'))
->dateTime()
->sortable()
]);
} }
} }

View File

@@ -22,7 +22,7 @@ class ServerController extends Controller
]); ]);
$server = app(CreateServerAction::class)->execute( $server = app(CreateServerAction::class)->execute(
ServerData::validate($data) ServerData::validateAndCreate($data)
); );
return response(content: ['data' => ServerData::from($server->refresh())->toArray()], status: 201); return response(content: ['data' => ServerData::from($server->refresh())->toArray()], status: 201);

View File

@@ -29,7 +29,7 @@ class SiteController extends Controller
]); ]);
$site = app(CreateSiteAction::class)->execute( $site = app(CreateSiteAction::class)->execute(
SiteData::validate($data) SiteData::validateAndCreate($data)
); );
$site->refresh(); $site->refresh();

View File

@@ -32,7 +32,7 @@ class UserController extends Controller
'requires_password_for_ftp' => ['nullable'], 'requires_password_for_ftp' => ['nullable'],
]); ]);
$userData = UserData::validate($data); $userData = UserData::validateAndCreate($data);
$user = User::create($userData->toArray()); $user = User::create($userData->toArray());
@@ -50,7 +50,7 @@ class UserController extends Controller
'requires_password_for_ftp' => [], 'requires_password_for_ftp' => [],
]); ]);
$userData = UserData::validate($data); $userData = UserData::validateAndCreate($data);
$user->update( $user->update(
Arr::only($userData->toArray(), array_keys($data)) Arr::only($userData->toArray(), array_keys($data))

View File

@@ -151,7 +151,7 @@ class ProfileBillingController extends Controller
$planId = $plan->stripe_plan_id; $planId = $plan->stripe_plan_id;
// Only do something if the user is not already subscribed to this plan. // Only do something if the user is not already subscribed to this plan.
if ($user->subscribedToPlan($planId, 'default')) { if ($user->subscribedToPrice($planId, 'default')) {
return redirect()->route('profile.billing.index')->with('error', 'You did not select a different plan'); return redirect()->route('profile.billing.index')->with('error', 'You did not select a different plan');
} }

View File

@@ -3,12 +3,14 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use App\Models\Server; use App\Models\Server;
use App\Models\ProviderPlan;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Jobs\Servers\DeleteServer; use App\Jobs\Servers\DeleteServer;
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
use Illuminate\Http\RedirectResponse; use Illuminate\Http\RedirectResponse;
use App\Http\Resources\ServerResource; use App\Http\Resources\ServerResource;
use App\DataTransferObjects\ServerData; use App\DataTransferObjects\ServerData;
use Illuminate\Database\Eloquent\Builder;
use App\Actions\Server\CreateServerAction; use App\Actions\Server\CreateServerAction;
use App\Http\Requests\ServerUpdateRequest; use App\Http\Requests\ServerUpdateRequest;
@@ -50,7 +52,7 @@ class ServerController extends Controller
$data['user_id'] = Auth::id(); $data['user_id'] = Auth::id();
app(CreateServerAction::class)->execute( app(CreateServerAction::class)->execute(
ServerData::validate($data) ServerData::validateAndCreate($data)
); );
return redirect()->route('servers.index'); return redirect()->route('servers.index');
@@ -101,19 +103,29 @@ class ServerController extends Controller
public function plansAndRegions(Request $request, $providerId) public function plansAndRegions(Request $request, $providerId)
{ {
$provider = $request->user()->package->providers()->findOrFail($providerId); $package = $request->user()->package;
$regions = $provider->regions() $provider = $package->providers()->findOrFail($providerId);
$regions = $provider
->regions()
->when($provider->allowed_regions, function ($query) use ($provider) { ->when($provider->allowed_regions, function ($query) use ($provider) {
return $query->whereIn('id', $provider->allowed_regions); return $query->whereIn('id', $provider->allowed_regions);
}) })
->pluck('label', 'id'); ->pluck('label', 'id');
$plans = $provider->plans() $plans = $provider
->plans()
->when($provider->allowed_plans, function ($query) use ($provider) { ->when($provider->allowed_plans, function ($query) use ($provider) {
return $query->whereIn('id', $provider->allowed_plans); return $query->whereIn('id', $provider->allowed_plans);
}) })
->pluck('label', 'id'); ->when($package->providerPlans()->whereBelongsTo($provider)->exists(), function (Builder $query) use ($provider, $package) {
return $query->whereIn('id', $package->providerPlans()->whereBelongsTo($provider)->pluck('provider_plans.id'));
})
->get()
->mapWithKeys(function (ProviderPlan $providerPlan) {
return [$providerPlan->id => $providerPlan->label ?? $providerPlan->plan_id];
});
return [ return [
'regions' => $regions, 'regions' => $regions,

View File

@@ -77,7 +77,7 @@ class SiteController extends Controller
$request->merge(['user_id' => auth()->id()]); $request->merge(['user_id' => auth()->id()]);
$site = app(CreateSiteAction::class)->execute( $site = app(CreateSiteAction::class)->execute(
SiteData::validate($request) SiteData::validateAndCreate($request)
); );
return $site return $site

View File

@@ -3,9 +3,7 @@
namespace App\Http\Middleware; namespace App\Http\Middleware;
use Closure; use Closure;
use Filament\Notifications\Notification;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Livewire\Livewire;
class Demo class Demo
{ {

View File

@@ -11,12 +11,12 @@ class HasAccessToThisGroup
public function handle(Request $request, Closure $next, $group) public function handle(Request $request, Closure $next, $group)
{ {
if ($group === 'servers') { if ($group === 'servers') {
$package = $request->user()->package ?? []; $package = $request->user()->package ?? null;
if ( if (
!Arr::get($package->server_permissions, 'create', false) && !Arr::get($package->server_permissions ?? [], 'create', false) &&
!Arr::get($package->server_permissions, 'update', false) && !Arr::get($package->server_permissions ?? [], 'update', false) &&
!Arr::get($package->server_permissions, 'delete', false) !Arr::get($package->server_permissions ?? [], 'delete', false)
) { ) {
abort(404); abort(404);
} }

View File

@@ -8,22 +8,12 @@ use Illuminate\Foundation\Http\FormRequest;
class SiteAppRequest extends FormRequest class SiteAppRequest extends FormRequest
{ {
/** public function authorize(): bool
* Determine if the user is authorized to make this request.
*
* @return bool
*/
public function authorize()
{ {
return auth()->check(); return auth()->check();
} }
/** public function rules(): array
* Get the validation rules that apply to the request.
*
* @return array
*/
public function rules()
{ {
return [ return [
'type' => [ 'type' => [
@@ -32,7 +22,6 @@ class SiteAppRequest extends FormRequest
Rule::in([ Rule::in([
Site::PROJECT_WORDPRESS, Site::PROJECT_WORDPRESS,
Site::PROJECT_NEXTCLOUD, Site::PROJECT_NEXTCLOUD,
Site::PROJECT_OCTOBERCMS
]) ])
] ]
]; ];

View File

@@ -16,18 +16,11 @@ class InstallApp implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public Site $site; public function __construct(public Site $site, public string $type = Site::PROJECT_WORDPRESS, public array $options = [])
public $type;
public $options;
public function __construct(Site $site, string $type = Site::PROJECT_WORDPRESS, array $options = [])
{ {
$this->site = $site;
$this->type = $type;
$this->options = $options;
} }
public function handle() public function handle(): void
{ {
$response = $this->getPloi() $response = $this->getPloi()
->server($this->site->server->ploi_id) ->server($this->site->server->ploi_id)

View File

@@ -14,27 +14,11 @@ class ChangePhpVersion implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $site; public function __construct(public Site $site, public $version = '8.2')
public $version;
/**
* Create a new job instance.
*
* @param Site $site
* @param string $version
*/
public function __construct(Site $site, $version = '7.4')
{ {
$this->site = $site;
$this->version = $version;
} }
/** public function handle(): void
* Execute the job.
*
* @return void
*/
public function handle()
{ {
$this->getPloi()->server($this->site->server->ploi_id)->sites($this->site->ploi_id)->phpVersion($this->version); $this->getPloi()->server($this->site->server->ploi_id)->sites($this->site->ploi_id)->phpVersion($this->version);

View File

@@ -34,6 +34,7 @@ class CreateSite implements ShouldQueue
); );
$this->site->ploi_id = $ploiSite->data->id; $this->site->ploi_id = $ploiSite->data->id;
$this->site->php_version = $ploiSite->data->php_version ?? 8.2;
$this->site->save(); $this->site->save();
// Lets fetch the status after 5 seconds // Lets fetch the status after 5 seconds

View File

@@ -4,7 +4,9 @@ namespace App\Models;
use App\Casts\PermissionCast; use App\Casts\PermissionCast;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
class Package extends Model class Package extends Model
{ {
@@ -39,19 +41,24 @@ class Package extends Model
'server_permissions' => PermissionCast::class, 'server_permissions' => PermissionCast::class,
]; ];
public function users() public function users(): HasMany
{ {
return $this->hasMany(User::class); return $this->hasMany(User::class);
} }
public function providers() public function providers(): BelongsToMany
{ {
return $this->belongsToMany(Provider::class); return $this->belongsToMany(Provider::class)->using(PackageProvider::class);
} }
protected static function booted() public function providerPlans(): BelongsToMany
{ {
static::deleting(function ($package) { return $this->belongsToMany(ProviderPlan::class);
}
protected static function booted(): void
{
static::deleting(function (self $package) {
$package->users()->update(['package_id' => null]); $package->users()->update(['package_id' => null]);
}); });
} }

View File

@@ -0,0 +1,26 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Relations\Pivot;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
class PackageProvider extends Pivot
{
protected static function booted(): void
{
static::deleting(function (self $packageProvider) {
$packageProvider->package->providerPlans()->whereBelongsTo($packageProvider->provider)->detach();
});
}
public function package(): BelongsTo
{
return $this->belongsTo(Package::class);
}
public function provider(): BelongsTo
{
return $this->belongsTo(Provider::class);
}
}

View File

@@ -22,8 +22,6 @@ class Site extends Model
const PROJECT_WORDPRESS = 'wordpress'; const PROJECT_WORDPRESS = 'wordpress';
const PROJECT_NEXTCLOUD = 'nextcloud'; const PROJECT_NEXTCLOUD = 'nextcloud';
const PROJECT_OCTOBERCMS = 'october-cms';
public $fillable = [ public $fillable = [
'ploi_id', 'ploi_id',
'domain', 'domain',

View File

@@ -3,6 +3,7 @@
namespace App\Models; namespace App\Models;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
class SystemLog extends Model class SystemLog extends Model
{ {
@@ -11,7 +12,7 @@ class SystemLog extends Model
'description' 'description'
]; ];
public function user() public function user(): BelongsTo
{ {
return $this->belongsTo(User::class); return $this->belongsTo(User::class);
} }

View File

@@ -2,6 +2,7 @@
namespace App\Models; namespace App\Models;
use Filament\Panel;
use App\Casts\Encrypted; use App\Casts\Encrypted;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
use Illuminate\Support\Str; use Illuminate\Support\Str;
@@ -69,7 +70,7 @@ class User extends Authenticatable implements HasLocalePreference, TwoFactorAuth
} }
} }
public function canAccessFilament(): bool public function canAccessPanel(Panel $panel): bool
{ {
return $this->role === self::ADMIN; return $this->role === self::ADMIN;
} }

Some files were not shown because too many files have changed in this diff Show More