feat(add services on membership page)

This commit is contained in:
2026-03-16 15:20:49 +01:00
parent c3b64e4bb9
commit 1790214fff
11 changed files with 434 additions and 77 deletions

View File

@@ -14,7 +14,8 @@
"mcp__laravel-boost__tinker", "mcp__laravel-boost__tinker",
"Bash(php artisan make:job:*)", "Bash(php artisan make:job:*)",
"mcp__laravel-boost__last-error", "mcp__laravel-boost__last-error",
"Bash(php artisan view:clear:*)" "Bash(php artisan view:clear:*)",
"mcp__laravel-boost__database-query"
] ]
}, },
"enableAllProjectMcpServers": true, "enableAllProjectMcpServers": true,

View File

@@ -8,6 +8,7 @@ use App\Services\Dolibarr\DolibarrService;
use Carbon\Carbon; use Carbon\Carbon;
use Illuminate\Console\Command; use Illuminate\Console\Command;
use Illuminate\Http\Client\ConnectionException; use Illuminate\Http\Client\ConnectionException;
use function Laravel\Prompts\progress; use function Laravel\Prompts\progress;
class SyncDolibarrMembers extends Command class SyncDolibarrMembers extends Command
@@ -26,6 +27,7 @@ class SyncDolibarrMembers extends Command
/** /**
* Execute the console command. * Execute the console command.
*
* @throws ConnectionException * @throws ConnectionException
*/ */
public function handle(): void public function handle(): void
@@ -50,7 +52,7 @@ class SyncDolibarrMembers extends Command
$memberStatuses = [ $memberStatuses = [
'-2' => 'excluded', '-2' => 'excluded',
'0' => 'cancelled', '0' => 'cancelled',
'1' => 'valid' '1' => 'valid',
]; ];
foreach ($doliMembers as $member) { foreach ($doliMembers as $member) {
@@ -63,8 +65,8 @@ class SyncDolibarrMembers extends Command
'nature' => 'physical', 'nature' => 'physical',
'member_type' => $member['type'], 'member_type' => $member['type'],
'group_id' => null, 'group_id' => null,
'lastname' => $member['firstname'], 'lastname' => $member['lastname'],
'firstname' => $member['lastname'], 'firstname' => $member['firstname'],
'email' => $member['email'] ?: null, 'email' => $member['email'] ?: null,
'retzien_email' => '', 'retzien_email' => '',
'company' => $member['societe'], 'company' => $member['societe'],
@@ -109,7 +111,7 @@ class SyncDolibarrMembers extends Command
'payment_status' => 'paid', 'payment_status' => 'paid',
'note_public' => $membership['note_public'], 'note_public' => $membership['note_public'],
'note_private' => $membership['note_private'], 'note_private' => $membership['note_private'],
'dolibarr_user_id' => $member['id'] 'dolibarr_user_id' => $member['id'],
] ]
); );
@@ -139,6 +141,7 @@ class SyncDolibarrMembers extends Command
/** /**
* Convert timestamp to date format safely * Convert timestamp to date format safely
*
* @todo: export this in a service or repo * @todo: export this in a service or repo
*/ */
private function toDate($timestamp): ?string private function toDate($timestamp): ?string

View File

@@ -2,6 +2,7 @@
namespace App\Filament\Actions; namespace App\Filament\Actions;
use App\Models\Membership;
use Filament\Actions\Action; use Filament\Actions\Action;
use Illuminate\Support\Facades\Bus; use Illuminate\Support\Facades\Bus;
use App\Models\Member; use App\Models\Member;
@@ -10,45 +11,56 @@ class ServiceToggleAction extends Action
{ {
protected string $serviceIdentifier; protected string $serviceIdentifier;
/*
* Create a new action instance.
*/
public static function forService(string $serviceIdentifier): static public static function forService(string $serviceIdentifier): static
{ {
return static::make('toggle_' . $serviceIdentifier) return static::make('toggle_' . $serviceIdentifier)
->configureForService($serviceIdentifier); ->configureForService($serviceIdentifier);
} }
/**
* Configure the action for a specific service.
*/
protected function configureForService(string $serviceIdentifier): static protected function configureForService(string $serviceIdentifier): static
{ {
$this->serviceIdentifier = $serviceIdentifier; $this->serviceIdentifier = $serviceIdentifier;
return $this return $this
->label('Service actif') ->label('Service actif')
->icon(fn (Member $record) => ->icon(fn (Member|Membership $record) =>
$record->hasService($serviceIdentifier) $this->getMember($record)?->hasService($serviceIdentifier)
? 'heroicon-o-check-circle' ? 'heroicon-o-check-circle'
: 'heroicon-o-x-circle' : 'heroicon-o-x-circle'
) )
->color(fn (Member $record) => ->color(fn (Member|Membership $record) =>
$record->hasService($serviceIdentifier) $this->getMember($record)?->hasService($serviceIdentifier)
? 'success' ? 'success'
: 'gray' : 'gray'
) )
->requiresConfirmation() ->requiresConfirmation()
->modalHeading(fn (Member $record) => ->modalHeading(fn (Member|Membership $record) =>
$record->hasService($serviceIdentifier) $this->getMember($record)?->hasService($serviceIdentifier)
? 'Désactiver le service' ? 'Désactiver le service'
: 'Activer le service' : 'Activer le service'
) )
->modalDescription(fn (Member $record) => ->modalDescription(fn (Member|Membership $record) =>
$record->hasService($serviceIdentifier) $this->getMember($record)?->hasService($serviceIdentifier)
? 'Êtes-vous sûr·e de vouloir désactiver ce service pour ce membre ?' ? 'Êtes-vous sûr·e de vouloir désactiver ce service pour ce membre ?'
: 'Êtes-vous sûr·e de vouloir activer ce service pour ce membre ?' : 'Êtes-vous sûr·e de vouloir activer ce service pour ce membre ?'
) )
->modalSubmitActionLabel(fn (Member $record) => ->modalSubmitActionLabel(fn (Member|Membership $record) =>
$record->hasService($serviceIdentifier) $this->getMember($record)?->hasService($serviceIdentifier)
? 'Désactiver' ? 'Désactiver'
: 'Activer' : 'Activer'
) )
->action(function (Member $record) use ($serviceIdentifier) { ->action(function (Member|Membership $record) use ($serviceIdentifier) {
$member = $this->getMember($record);
if (!$member) {
return;
}
// @todo à discuter // @todo à discuter
/* if ($record->hasService($serviceIdentifier)) { /* if ($record->hasService($serviceIdentifier)) {
@@ -62,4 +74,12 @@ class ServiceToggleAction extends Action
}*/ }*/
}); });
} }
/**
* Get the member associated with the given record.
*/
protected function getMember(Member|Membership $record): ?Member
{
return $record instanceof Member ? $record : $record->member;
}
} }

View File

@@ -11,6 +11,7 @@ use Filament\Actions\EditAction;
use Filament\Resources\RelationManagers\RelationManager; use Filament\Resources\RelationManagers\RelationManager;
use Filament\Tables\Columns\TextColumn; use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table; use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Model;
class MembershipsRelationManager extends RelationManager class MembershipsRelationManager extends RelationManager
{ {
@@ -18,6 +19,13 @@ class MembershipsRelationManager extends RelationManager
protected static ?string $title = 'Adhésions'; protected static ?string $title = 'Adhésions';
public static function getTitle(Model $ownerRecord, string $pageClass): string
{
$fullName = $ownerRecord->getFullNameAttribute();
return $fullName ? 'Adhésions de '.$fullName : 'Adhésions';
}
public function table(Table $table): Table public function table(Table $table): Table
{ {
return $table return $table

View File

@@ -2,28 +2,46 @@
namespace App\Filament\Resources\Memberships\Schemas; namespace App\Filament\Resources\Memberships\Schemas;
use App\Enums\IspconfigType;
use App\Filament\Actions\ServiceToggleAction;
use App\Models\Membership; use App\Models\Membership;
use App\Models\Service;
use Filament\Actions\Action; use Filament\Actions\Action;
use Filament\Forms\Components\CheckboxList;
use Filament\Forms\Components\DatePicker; use Filament\Forms\Components\DatePicker;
use Filament\Forms\Components\Select; use Filament\Forms\Components\Select;
use Filament\Forms\Components\TextInput; use Filament\Forms\Components\TextInput;
use Filament\Infolists\Infolist; use Filament\Infolists\Components\RepeatableEntry;
use Filament\Infolists\Components\TextEntry; use Filament\Infolists\Components\TextEntry;
use Filament\Infolists\Components\ViewEntry;
use Filament\Schemas\Components\Grid; use Filament\Schemas\Components\Grid;
use Filament\Schemas\Components\Section; use Filament\Schemas\Components\Section;
use Filament\Schemas\Components\Tabs;
use Filament\Schemas\Schema; use Filament\Schemas\Schema;
use Filament\Support\Icons\Heroicon;
class MembershipForm class MembershipForm
{ {
public static function configure(Schema $schema): Schema public static function configure(Schema $schema): Schema
{ {
return $schema return $schema
->components ([ ->components([
Grid::make() Grid::make()
->schema([ ->schema([
/*
|--------------------------------------------------------------------------
| Colonne principale
|--------------------------------------------------------------------------
*/
Grid::make(1) Grid::make(1)
->schema([
Tabs::make('MembershipTabs')
->tabs([
/*
|--------------------------------------------------------------------------
| TAB : Informations générales
|--------------------------------------------------------------------------
*/
Tabs\Tab::make('Informations générales')
->icon(Heroicon::OutlinedInformationCircle)
->schema([ ->schema([
Section::make('Adhérent') Section::make('Adhérent')
->headerActions([ ->headerActions([
@@ -40,7 +58,7 @@ class MembershipForm
TextEntry::make('author.name') TextEntry::make('author.name')
->label(Membership::getAttributeLabel('admin_id')), ->label(Membership::getAttributeLabel('admin_id')),
TextEntry::make('created_at') TextEntry::make('created_at')
->label(Membership::getAttributeLabel('created_at')) ->label(Membership::getAttributeLabel('created_at')),
]) ])
->columns(2), ->columns(2),
@@ -64,27 +82,156 @@ class MembershipForm
->default(0.0), ->default(0.0),
]) ])
->columns(2), ->columns(2),
]),
Section::make('Services') /*
|--------------------------------------------------------------------------
| TAB : Services/Modules
|--------------------------------------------------------------------------
*/
Tabs\Tab::make('Modules')
->icon(Heroicon::OutlinedPuzzlePiece)
->schema([ ->schema([
CheckboxList::make('services') /*
->label('Services activés') | Messageries ISPConfig (lecture seule)
->helperText('Sélectionne les services que ce membre peut utiliser.') */
->options(Service::all()->pluck('name', 'id')) Section::make('Messagerie ISPConfig')
->relationship('services', 'name') ->afterHeader([
->columns(2) ServiceToggleAction::forService('mail'),
]) ])
/*->schema(function () { ->collapsible()
return Service::all()->map(function ($service) { ->schema([
return Toggle::make("services_sync.{$service->id}") RepeatableEntry::make('ispconfig_mails')
->label($service->name) ->label('Données ISPConfig Mail')
->default(false) ->state(fn(?Membership $record) => $record?->member?->ispconfigs()
->helperText("Active ou désactive le service {$service->name}"); ->where('type', IspconfigType::MAIL)
})->toArray(); ->get()
})*/ )
->schema([
TextEntry::make('email')
->label('Adresse email'),
TextEntry::make('ispconfig_service_user_id')
->label('ID ISPConfig'),
TextEntry::make('data.mailuser.quota')
->label('Quota'),
TextEntry::make('data.mailuser.domain')
->label('Domaine')
->default('retzien.fr'),
ViewEntry::make('data')
->label('JSON')
->view('filament.components.json-viewer')
->viewData(fn($state) => [
'data' => $state,
])
->columnSpanFull(),
])
->columns(2),
])
->visible(fn(?Membership $record) => $record?->member?->ispconfigs()
->where('type', IspconfigType::MAIL)
->exists() ?? false
),
/*
| Hébergements web ISPConfig
*/
Section::make('Hébergements Web')
->afterHeader([
ServiceToggleAction::forService('webhosting'),
])
->collapsible()
->schema([
RepeatableEntry::make('ispconfigs_web')
->label('Données ISPConfig Web')
->state(fn(?Membership $record) => $record?->member?->ispconfigs()
->where('type', IspconfigType::WEB)
->get()
->map(fn($ispconfig) => $ispconfig->toArray())
->all()
)
->schema([
TextEntry::make('data.domain_id')
->label('ID ISPConfig'),
TextEntry::make('data.domain')
->label('Domaine'),
TextEntry::make('data.active')
->label('État')
->formatStateUsing(fn($state) => $state === 'y' ? 'Activé' : 'Désactivé'
),
ViewEntry::make('data')
->label('JSON')
->view('filament.components.json-viewer')
->viewData(fn($state) => [
'data' => $state,
])
->columnSpanFull(),
])
->columns(3),
])
->visible(fn(?Membership $record) => $record?->member?->ispconfigs()
->where('type', IspconfigType::WEB)
->exists() ?? false
),
/*
| Compte(s) NextCloud (lecture seule)
*/
Section::make('NextCloud')
->afterHeader([
ServiceToggleAction::forService('nextcloud'),
])
->collapsible()
->schema([
RepeatableEntry::make('nextcloud_accounts')
->label('Données NextCloud')
->state(fn(?Membership $record) => $record?->member?->nextcloudAccounts()
->get()
->map(fn($nextcloudAccount) => $nextcloudAccount->toArray())
->all()
)
->schema([
TextEntry::make('nextcloud_user_id')
->label('Id Nextcloud'),
TextEntry::make('data.displayname')
->label('Nom de l\'utilisateur'),
TextEntry::make('data.enabled')
->label('État')
->formatStateUsing(fn($state) => $state == 'true' ? 'Activé' : 'Désactivé'
),
ViewEntry::make('data')
->label('JSON')
->view('filament.components.json-viewer')
->viewData(fn($state) => [
'data' => $state,
])
->columnSpanFull(),
])
->columns(3),
])
->visible(fn(?Membership $record) => $record?->member?->nextcloudAccounts()
->exists() ?? false
),
]),
])
->contained(false),
]) ])
->columnSpan(3), ->columnSpan(3),
/*
|--------------------------------------------------------------------------
| Colonne latérale
|--------------------------------------------------------------------------
*/
Grid::make(1) Grid::make(1)
->schema([ ->schema([
Section::make('Statut') Section::make('Statut')
@@ -99,13 +246,13 @@ class MembershipForm
->required(), ->required(),
DatePicker::make('end_date') DatePicker::make('end_date')
->label(Membership::getAttributeLabel('end_date')), ->label(Membership::getAttributeLabel('end_date')),
]) ])
->extraAttributes(['class' => 'sticky top-4 h-fit']),
]) ])
->columnSpan(1), ->columnSpan(1),
]) ])
->columns(4) ->columns(4)
->columnSpanFull() ->columnSpanFull(),
]); ]);
} }
} }

View File

@@ -103,6 +103,9 @@ class Member extends Model
'phone2', 'phone2',
'public_membership', 'public_membership',
'website_url', 'website_url',
'member_type',
'retzien_email',
'created_at',
]; ];
public static function getAttributeLabel(string $attribute): string public static function getAttributeLabel(string $attribute): string

View File

@@ -23,6 +23,7 @@ return [
'paid' => 'Payé', 'paid' => 'Payé',
'unpaid' => 'Impayé', 'unpaid' => 'Impayé',
'partial' => 'Paiement partiel', 'partial' => 'Paiement partiel',
'services' => 'Services',
'created_at' => 'Créée le', 'created_at' => 'Créée le',
'updated_at' => 'Mise à jour le', 'updated_at' => 'Mise à jour le',
'subscription' => [ 'subscription' => [

View File

@@ -0,0 +1,83 @@
import { queryParams, type RouteQueryOptions, type RouteDefinition, type RouteFormDefinition } from './../../../../wayfinder'
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
const Synchronisations = (options?: RouteQueryOptions): RouteDefinition<'get'> => ({
url: Synchronisations.url(options),
method: 'get',
})
Synchronisations.definition = {
methods: ["get","head"],
url: '/admin/synchronisations',
} satisfies RouteDefinition<["get","head"]>
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
Synchronisations.url = (options?: RouteQueryOptions) => {
return Synchronisations.definition.url + queryParams(options)
}
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
Synchronisations.get = (options?: RouteQueryOptions): RouteDefinition<'get'> => ({
url: Synchronisations.url(options),
method: 'get',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
Synchronisations.head = (options?: RouteQueryOptions): RouteDefinition<'head'> => ({
url: Synchronisations.url(options),
method: 'head',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
const SynchronisationsForm = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
action: Synchronisations.url(options),
method: 'get',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
SynchronisationsForm.get = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
action: Synchronisations.url(options),
method: 'get',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
SynchronisationsForm.head = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
action: Synchronisations.url({
[options?.mergeQuery ? 'mergeQuery' : 'query']: {
_method: 'HEAD',
...(options?.query ?? options?.mergeQuery ?? {}),
}
}),
method: 'get',
})
Synchronisations.form = SynchronisationsForm
export default Synchronisations

View File

@@ -0,0 +1,7 @@
import Synchronisations from './Synchronisations'
const Pages = {
Synchronisations: Object.assign(Synchronisations, Synchronisations),
}
export default Pages

View File

@@ -1,6 +1,8 @@
import Pages from './Pages'
import Resources from './Resources' import Resources from './Resources'
const Filament = { const Filament = {
Pages: Object.assign(Pages, Pages),
Resources: Object.assign(Resources, Resources), Resources: Object.assign(Resources, Resources),
} }

View File

@@ -80,8 +80,90 @@ dashboardForm.head = (options?: RouteQueryOptions): RouteFormDefinition<'get'> =
dashboard.form = dashboardForm dashboard.form = dashboardForm
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
export const synchronisations = (options?: RouteQueryOptions): RouteDefinition<'get'> => ({
url: synchronisations.url(options),
method: 'get',
})
synchronisations.definition = {
methods: ["get","head"],
url: '/admin/synchronisations',
} satisfies RouteDefinition<["get","head"]>
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
synchronisations.url = (options?: RouteQueryOptions) => {
return synchronisations.definition.url + queryParams(options)
}
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
synchronisations.get = (options?: RouteQueryOptions): RouteDefinition<'get'> => ({
url: synchronisations.url(options),
method: 'get',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
synchronisations.head = (options?: RouteQueryOptions): RouteDefinition<'head'> => ({
url: synchronisations.url(options),
method: 'head',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
const synchronisationsForm = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
action: synchronisations.url(options),
method: 'get',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
synchronisationsForm.get = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
action: synchronisations.url(options),
method: 'get',
})
/**
* @see \App\Filament\Pages\Synchronisations::__invoke
* @see app/Filament/Pages/Synchronisations.php:7
* @route '/admin/synchronisations'
*/
synchronisationsForm.head = (options?: RouteQueryOptions): RouteFormDefinition<'get'> => ({
action: synchronisations.url({
[options?.mergeQuery ? 'mergeQuery' : 'query']: {
_method: 'HEAD',
...(options?.query ?? options?.mergeQuery ?? {}),
}
}),
method: 'get',
})
synchronisations.form = synchronisationsForm
const pages = { const pages = {
dashboard: Object.assign(dashboard, dashboard), dashboard: Object.assign(dashboard, dashboard),
synchronisations: Object.assign(synchronisations, synchronisations),
} }
export default pages export default pages