| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377 | <?php
namespace App\Filament\Company\Clusters\Settings\Resources;
use App\Enums\Accounting\AdjustmentCategory;
use App\Enums\Accounting\AdjustmentComputation;
use App\Enums\Accounting\AdjustmentScope;
use App\Enums\Accounting\AdjustmentStatus;
use App\Enums\Accounting\AdjustmentType;
use App\Filament\Company\Clusters\Settings;
use App\Filament\Company\Clusters\Settings\Resources\AdjustmentResource\Pages;
use App\Models\Accounting\Adjustment;
use App\Services\CompanySettingsService;
use Filament\Forms;
use Filament\Forms\Form;
use Filament\Notifications\Notification;
use Filament\Resources\Resource;
use Filament\Tables;
use Filament\Tables\Filters\Indicator;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Collection;
class AdjustmentResource extends Resource
{
    protected static ?string $model = Adjustment::class;
    protected static ?string $cluster = Settings::class;
    public static function form(Form $form): Form
    {
        return $form
            ->schema([
                Forms\Components\Section::make('General')
                    ->schema([
                        Forms\Components\TextInput::make('name')
                            ->autofocus()
                            ->required()
                            ->maxLength(255),
                        Forms\Components\Textarea::make('description')
                            ->label('Description'),
                    ]),
                Forms\Components\Section::make('Configuration')
                    ->schema([
                        Forms\Components\Select::make('category')
                            ->localizeLabel()
                            ->options(AdjustmentCategory::class)
                            ->default(AdjustmentCategory::Tax)
                            ->live()
                            ->required(),
                        Forms\Components\Select::make('type')
                            ->localizeLabel()
                            ->options(AdjustmentType::class)
                            ->default(AdjustmentType::Sales)
                            ->live()
                            ->required(),
                        Forms\Components\Checkbox::make('recoverable')
                            ->label('Recoverable')
                            ->default(false)
                            ->helperText('When enabled, tax is tracked separately as claimable from the government. Non-recoverable taxes are treated as part of the expense.')
                            ->visible(fn (Forms\Get $get) => AdjustmentCategory::parse($get('category'))->isTax() && AdjustmentType::parse($get('type'))->isPurchase()),
                    ])
                    ->columns()
                    ->visibleOn('create'),
                Forms\Components\Section::make('Adjustment Details')
                    ->schema([
                        Forms\Components\Select::make('computation')
                            ->localizeLabel()
                            ->options(AdjustmentComputation::class)
                            ->default(AdjustmentComputation::Percentage)
                            ->live()
                            ->required(),
                        Forms\Components\TextInput::make('rate')
                            ->localizeLabel()
                            ->rate(static fn (Forms\Get $get) => $get('computation'))
                            ->required(),
                        Forms\Components\Select::make('scope')
                            ->localizeLabel()
                            ->options(AdjustmentScope::class),
                    ])
                    ->columns(),
                Forms\Components\Section::make('Dates')
                    ->schema([
                        Forms\Components\DateTimePicker::make('start_date')
                            ->timezone(CompanySettingsService::getDefaultTimezone()),
                        Forms\Components\DateTimePicker::make('end_date')
                            ->timezone(CompanySettingsService::getDefaultTimezone())
                            ->after('start_date'),
                    ])
                    ->columns()
                    ->visible(fn (Forms\Get $get) => AdjustmentCategory::parse($get('category'))->isDiscount()),
            ]);
    }
    public static function table(Table $table): Table
    {
        return $table
            ->columns([
                Tables\Columns\TextColumn::make('name')
                    ->label('Name')
                    ->sortable(),
                Tables\Columns\TextColumn::make('status')
                    ->badge(),
                Tables\Columns\TextColumn::make('category')
                    ->searchable(),
                Tables\Columns\TextColumn::make('type')
                    ->searchable(),
                Tables\Columns\TextColumn::make('rate')
                    ->localizeLabel()
                    ->rate(static fn (Adjustment $record) => $record->computation->value)
                    ->searchable()
                    ->sortable(),
                Tables\Columns\TextColumn::make('paused_until')
                    ->label('Auto-Resume Date')
                    ->dateTime()
                    ->sortable()
                    ->toggleable(isToggledHiddenByDefault: true),
                Tables\Columns\TextColumn::make('start_date')
                    ->dateTime()
                    ->sortable()
                    ->toggleable(isToggledHiddenByDefault: true),
                Tables\Columns\TextColumn::make('end_date')
                    ->dateTime()
                    ->sortable()
                    ->toggleable(isToggledHiddenByDefault: true),
            ])
            ->filters([
                Tables\Filters\SelectFilter::make('status')
                    ->label('Status')
                    ->native(false)
                    ->default('unarchived')
                    ->options(
                        collect(AdjustmentStatus::cases())
                            ->mapWithKeys(fn (AdjustmentStatus $status) => [$status->value => $status->getLabel()])
                            ->merge([
                                'unarchived' => 'Unarchived',
                            ])
                            ->toArray()
                    )
                    ->indicateUsing(function (Tables\Filters\SelectFilter $filter, array $state) {
                        if (blank($state['value'] ?? null)) {
                            return [];
                        }
                        $label = collect($filter->getOptions())
                            ->mapWithKeys(fn (string | array $label, string $value): array => is_array($label) ? $label : [$value => $label])
                            ->get($state['value']);
                        if (blank($label)) {
                            return [];
                        }
                        $indicator = $filter->getIndicator();
                        if (! $indicator instanceof Indicator) {
                            if ($state['value'] === 'unarchived') {
                                $indicator = $label;
                            } else {
                                $indicator = Indicator::make("{$indicator}: {$label}");
                            }
                        }
                        return [$indicator];
                    })
                    ->query(function (Builder $query, array $data): Builder {
                        if (blank($data['value'] ?? null)) {
                            return $query;
                        }
                        if ($data['value'] !== 'unarchived') {
                            return $query->where('status', $data['value']);
                        } else {
                            return $query->where('status', '!=', AdjustmentStatus::Archived->value);
                        }
                    }),
                Tables\Filters\SelectFilter::make('category')
                    ->label('Category')
                    ->native(false)
                    ->options(AdjustmentCategory::class),
                Tables\Filters\SelectFilter::make('type')
                    ->label('Type')
                    ->native(false)
                    ->options(AdjustmentType::class),
                Tables\Filters\SelectFilter::make('computation')
                    ->label('Computation')
                    ->native(false)
                    ->options(AdjustmentComputation::class),
            ])
            ->actions([
                Tables\Actions\ActionGroup::make([
                    Tables\Actions\EditAction::make(),
                    Tables\Actions\Action::make('pause')
                        ->label('Pause')
                        ->icon('heroicon-m-pause')
                        ->form([
                            Forms\Components\DateTimePicker::make('paused_until')
                                ->label('Auto-resume date')
                                ->timezone(CompanySettingsService::getDefaultTimezone())
                                ->helperText('When should this adjustment automatically resume? Leave empty to keep paused indefinitely.')
                                ->after('now'),
                            Forms\Components\Textarea::make('status_reason')
                                ->label('Reason for pausing')
                                ->maxLength(255),
                        ])
                        ->databaseTransaction()
                        ->successNotificationTitle('Adjustment paused')
                        ->failureNotificationTitle('Failed to pause adjustment')
                        ->visible(fn (Adjustment $record) => $record->canBePaused())
                        ->action(function (Adjustment $record, array $data, Tables\Actions\Action $action) {
                            $pausedUntil = $data['paused_until'] ?? null;
                            $reason = $data['status_reason'] ?? null;
                            $record->pause($reason, $pausedUntil);
                            $action->success();
                        }),
                    Tables\Actions\Action::make('resume')
                        ->label('Resume')
                        ->icon('heroicon-m-play')
                        ->requiresConfirmation()
                        ->databaseTransaction()
                        ->successNotificationTitle('Adjustment resumed')
                        ->failureNotificationTitle('Failed to resume adjustment')
                        ->visible(fn (Adjustment $record) => $record->canBeResumed())
                        ->action(function (Adjustment $record, Tables\Actions\Action $action) {
                            $record->resume();
                            $action->success();
                        }),
                    Tables\Actions\Action::make('archive')
                        ->label('Archive')
                        ->icon('heroicon-m-archive-box')
                        ->color('danger')
                        ->form([
                            Forms\Components\Textarea::make('status_reason')
                                ->label('Reason for archiving')
                                ->maxLength(255),
                        ])
                        ->databaseTransaction()
                        ->successNotificationTitle('Adjustment archived')
                        ->failureNotificationTitle('Failed to archive adjustment')
                        ->visible(fn (Adjustment $record) => $record->canBeArchived())
                        ->action(function (Adjustment $record, array $data, Tables\Actions\Action $action) {
                            $reason = $data['status_reason'] ?? null;
                            $record->archive($reason);
                            $action->success();
                        }),
                ]),
            ])
            ->bulkActions([
                Tables\Actions\BulkActionGroup::make([
                    Tables\Actions\BulkAction::make('pause')
                        ->label('Pause')
                        ->icon('heroicon-m-pause')
                        ->form([
                            Forms\Components\DateTimePicker::make('paused_until')
                                ->label('Auto-resume date')
                                ->timezone(CompanySettingsService::getDefaultTimezone())
                                ->helperText('When should these adjustments automatically resume? Leave empty to keep paused indefinitely.')
                                ->after('now'),
                            Forms\Components\Textarea::make('status_reason')
                                ->label('Reason for pausing')
                                ->maxLength(255),
                        ])
                        ->databaseTransaction()
                        ->successNotificationTitle('Adjustments paused')
                        ->failureNotificationTitle('Failed to pause adjustments')
                        ->beforeFormFilled(function (Collection $records, Tables\Actions\BulkAction $action) {
                            $isInvalid = $records->contains(fn (Adjustment $record) => ! $record->canBePaused());
                            if ($isInvalid) {
                                Notification::make()
                                    ->title('Pause failed')
                                    ->body('Only adjustments that are currently active can be paused. Please adjust your selection and try again.')
                                    ->persistent()
                                    ->danger()
                                    ->send();
                                $action->cancel(true);
                            }
                        })
                        ->deselectRecordsAfterCompletion()
                        ->action(function (Collection $records, array $data, Tables\Actions\BulkAction $action) {
                            $pausedUntil = $data['paused_until'] ?? null;
                            $reason = $data['status_reason'] ?? null;
                            $records->each(function (Adjustment $record) use ($reason, $pausedUntil) {
                                $record->pause($reason, $pausedUntil);
                            });
                            $action->success();
                        }),
                    Tables\Actions\BulkAction::make('resume')
                        ->label('Resume')
                        ->icon('heroicon-m-play')
                        ->databaseTransaction()
                        ->requiresConfirmation()
                        ->successNotificationTitle('Adjustments resumed')
                        ->failureNotificationTitle('Failed to resume adjustments')
                        ->before(function (Collection $records, Tables\Actions\BulkAction $action) {
                            $isInvalid = $records->contains(fn (Adjustment $record) => ! $record->canBeResumed());
                            if ($isInvalid) {
                                Notification::make()
                                    ->title('Resume failed')
                                    ->body('Only adjustments that are currently paused can be resumed. Please adjust your selection and try again.')
                                    ->persistent()
                                    ->danger()
                                    ->send();
                                $action->cancel(true);
                            }
                        })
                        ->deselectRecordsAfterCompletion()
                        ->action(function (Collection $records, Tables\Actions\BulkAction $action) {
                            $records->each(function (Adjustment $record) {
                                $record->resume();
                            });
                            $action->success();
                        }),
                    Tables\Actions\BulkAction::make('archive')
                        ->label('Archive')
                        ->icon('heroicon-m-archive-box')
                        ->color('danger')
                        ->form([
                            Forms\Components\Textarea::make('status_reason')
                                ->label('Reason for archiving')
                                ->maxLength(255),
                        ])
                        ->databaseTransaction()
                        ->successNotificationTitle('Adjustments archived')
                        ->failureNotificationTitle('Failed to archive adjustments')
                        ->beforeFormFilled(function (Collection $records, Tables\Actions\BulkAction $action) {
                            $isInvalid = $records->contains(fn (Adjustment $record) => ! $record->canBeArchived());
                            if ($isInvalid) {
                                Notification::make()
                                    ->title('Archive failed')
                                    ->body('Only adjustments that are currently active or paused can be archived. Please adjust your selection and try again.')
                                    ->persistent()
                                    ->danger()
                                    ->send();
                                $action->cancel(true);
                            }
                        })
                        ->deselectRecordsAfterCompletion()
                        ->action(function (Collection $records, array $data, Tables\Actions\BulkAction $action) {
                            $reason = $data['status_reason'] ?? null;
                            $records->each(function (Adjustment $record) use ($reason) {
                                $record->archive($reason);
                            });
                            $action->success();
                        }),
                ]),
            ]);
    }
    public static function getRelations(): array
    {
        return [
            //
        ];
    }
    public static function getPages(): array
    {
        return [
            'index' => Pages\ListAdjustments::route('/'),
            'create' => Pages\CreateAdjustment::route('/create'),
            'edit' => Pages\EditAdjustment::route('/{record}/edit'),
        ];
    }
}
 |