| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195 | <?php
namespace App\Filament\Company\Resources\Common;
use App\Enums\Accounting\AccountCategory;
use App\Enums\Accounting\AccountType;
use App\Enums\Common\OfferingType;
use App\Filament\Company\Resources\Common\OfferingResource\Pages;
use App\Models\Accounting\Account;
use App\Models\Common\Offering;
use App\Utilities\Currency\CurrencyAccessor;
use Filament\Forms;
use Filament\Forms\Form;
use Filament\Resources\Resource;
use Filament\Tables;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Support\Str;
use JaOcero\RadioDeck\Forms\Components\RadioDeck;
class OfferingResource extends Resource
{
    protected static ?string $model = Offering::class;
    protected static ?string $modelLabel = 'Offering';
    protected static ?string $navigationIcon = 'heroicon-o-square-3-stack-3d';
    public static function form(Form $form): Form
    {
        return $form
            ->schema([
                Forms\Components\Section::make('General')
                    ->schema([
                        RadioDeck::make('type')
                            ->options(OfferingType::class)
                            ->default(OfferingType::Product)
                            ->icons(OfferingType::class)
                            ->color('primary')
                            ->columns()
                            ->required(),
                        Forms\Components\TextInput::make('name')
                            ->autofocus()
                            ->required()
                            ->columnStart(1)
                            ->maxLength(255),
                        Forms\Components\TextInput::make('price')
                            ->required()
                            ->money(),
                        Forms\Components\Textarea::make('description')
                            ->label('Description')
                            ->columnSpan(2)
                            ->rows(3),
                        Forms\Components\CheckboxList::make('attributes')
                            ->options([
                                'Sellable' => 'Sellable',
                                'Purchasable' => 'Purchasable',
                            ])
                            ->hiddenLabel()
                            ->required()
                            ->live()
                            ->bulkToggleable()
                            ->validationMessages([
                                'required' => 'The offering must be either sellable or purchasable.',
                            ]),
                    ])->columns(),
                // Sellable Section
                Forms\Components\Section::make('Sale Information')
                    ->schema([
                        Forms\Components\Select::make('income_account_id')
                            ->label('Income Account')
                            ->options(Account::query()
                                ->where('category', AccountCategory::Revenue)
                                ->where('type', AccountType::OperatingRevenue)
                                ->pluck('name', 'id')
                                ->toArray())
                            ->searchable()
                            ->preload()
                            ->required()
                            ->validationMessages([
                                'required' => 'The income account is required for sellable offerings.',
                            ]),
                        Forms\Components\Select::make('salesTaxes')
                            ->label('Sales Tax')
                            ->relationship('salesTaxes', 'name')
                            ->preload()
                            ->multiple(),
                        Forms\Components\Select::make('salesDiscounts')
                            ->label('Sales Discount')
                            ->relationship('salesDiscounts', 'name')
                            ->preload()
                            ->multiple(),
                    ])
                    ->columns()
                    ->visible(fn (Forms\Get $get) => in_array('Sellable', $get('attributes') ?? [])),
                // Purchasable Section
                Forms\Components\Section::make('Purchase Information')
                    ->schema([
                        Forms\Components\Select::make('expense_account_id')
                            ->label('Expense Account')
                            ->options(Account::query()
                                ->where('category', AccountCategory::Expense)
                                ->where('type', AccountType::OperatingExpense)
                                ->orderBy('name')
                                ->pluck('name', 'id')
                                ->toArray())
                            ->searchable()
                            ->preload()
                            ->required()
                            ->validationMessages([
                                'required' => 'The expense account is required for purchasable offerings.',
                            ]),
                        Forms\Components\Select::make('purchaseTaxes')
                            ->label('Purchase Tax')
                            ->relationship('purchaseTaxes', 'name')
                            ->preload()
                            ->multiple(),
                        Forms\Components\Select::make('purchaseDiscounts')
                            ->label('Purchase Discount')
                            ->relationship('purchaseDiscounts', 'name')
                            ->preload()
                            ->multiple(),
                    ])
                    ->columns()
                    ->visible(fn (Forms\Get $get) => in_array('Purchasable', $get('attributes') ?? [])),
            ])->columns();
    }
    public static function table(Table $table): Table
    {
        return $table
            ->modifyQueryUsing(function (Builder $query) {
                $query->selectRaw("
                        *,
                        CONCAT_WS(' & ',
                            CASE WHEN sellable THEN 'Sellable' END,
                            CASE WHEN purchasable THEN 'Purchasable' END
                        ) AS attributes
                    ");
            })
            ->columns([
                Tables\Columns\TextColumn::make('name')
                    ->label('Name'),
                Tables\Columns\TextColumn::make('attributes')
                    ->label('Attributes')
                    ->badge(),
                Tables\Columns\TextColumn::make('type')
                    ->searchable(),
                Tables\Columns\TextColumn::make('price')
                    ->currency(CurrencyAccessor::getDefaultCurrency(), true)
                    ->sortable()
                    ->description(function (Offering $record) {
                        $adjustments = $record->adjustments()
                            ->pluck('name')
                            ->join(', ');
                        if (empty($adjustments)) {
                            return null;
                        }
                        $adjustmentsList = Str::of($adjustments)->limit(40);
                        return "+ {$adjustmentsList}";
                    }),
            ])
            ->filters([
                //
            ])
            ->actions([
                Tables\Actions\EditAction::make(),
            ])
            ->bulkActions([
                Tables\Actions\BulkActionGroup::make([
                    Tables\Actions\DeleteBulkAction::make(),
                ]),
            ]);
    }
    public static function getRelations(): array
    {
        return [
            //
        ];
    }
    public static function getPages(): array
    {
        return [
            'index' => Pages\ListOfferings::route('/'),
            'create' => Pages\CreateOffering::route('/create'),
            'edit' => Pages\EditOffering::route('/{record}/edit'),
        ];
    }
}
 |