Faza 3.2: Service modules — Norme-ore, Tehnicieni, Fișe lucru
Schema:
- users + specialization, color, hourly_rate (pentru maistri)
- labors: catalog manopere standard cu category/ore/preț (RO+RU)
- work_orders: nr unique per tenant, status workflow (9 stări),
pay_status (3 stări), client/vehicle/master/deal/appointment refs,
complaint/diagnosis/recommendations, total auto-calculat
- wo_works: manopere per fișă, recalc auto la save/delete
- wo_parts: piese per fișă (free-text deocamdată), discount/total auto
Filament resources (group Service):
- LaborResource: CRUD + grupare pe categorie + filter active
- WorkOrderResource: form complex în 4 secțiuni (antet, diagnostic, plată)
+ 2 RelationManagers (Works, Parts)
- MasterResource: vedere User filtrată role=mechanic, edit specializare/
culoare calendar/tarif oră
Conversie auto: la adaugare manoperă din catalog Labor,
form populează numele + ore + preț/oră derivat (price/hours).
Number generator pentru WO: format WO-{YY}-{NNNN} per tenant per an,
calculat în CreateWorkOrder via WorkOrder::generateNumber().
Seed extins:
- 3 mecanici (Vasile/Andrei/Nicolae) cu culori + specializări
- 10 manopere standard din prototipul AutoCRM.html
- 1 fișă demo (BMW X5 plăcuțe Brembo) cu 1 manoperă + 1 piesă, total auto
This commit is contained in:
@@ -0,0 +1,84 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources;
|
||||
|
||||
use App\Filament\Tenant\Resources\LaborResource\Pages;
|
||||
use App\Models\Tenant\Labor;
|
||||
use Filament\Actions;
|
||||
use Filament\Forms;
|
||||
use Filament\Resources\Resource;
|
||||
use Filament\Schemas;
|
||||
use Filament\Schemas\Schema;
|
||||
use Filament\Tables;
|
||||
use Filament\Tables\Table;
|
||||
|
||||
class LaborResource extends Resource
|
||||
{
|
||||
protected static ?string $model = Labor::class;
|
||||
|
||||
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-clock';
|
||||
|
||||
protected static ?string $navigationLabel = 'Norme-ore';
|
||||
|
||||
protected static string|\UnitEnum|null $navigationGroup = 'Service';
|
||||
|
||||
protected static ?string $modelLabel = 'normă';
|
||||
|
||||
protected static ?string $pluralModelLabel = 'norme-ore';
|
||||
|
||||
protected static ?int $navigationSort = 32;
|
||||
|
||||
public static function form(Schema $schema): Schema
|
||||
{
|
||||
return $schema->components([
|
||||
Schemas\Components\Section::make('Manoperă')
|
||||
->columns(2)
|
||||
->schema([
|
||||
Forms\Components\Select::make('category')
|
||||
->label('Categorie')
|
||||
->options(array_combine(Labor::CATEGORIES, Labor::CATEGORIES))
|
||||
->required()
|
||||
->searchable(),
|
||||
Forms\Components\TextInput::make('code')->label('Cod')->maxLength(32),
|
||||
Forms\Components\TextInput::make('name_ro')->label('Nume (RO)')->required()->maxLength(160),
|
||||
Forms\Components\TextInput::make('name_ru')->label('Nume (RU)')->maxLength(160),
|
||||
Forms\Components\TextInput::make('hours')->label('Ore')->numeric()->default(1)->required(),
|
||||
Forms\Components\TextInput::make('price')->label('Preț (MDL)')->numeric()->default(0),
|
||||
Forms\Components\Toggle::make('is_active')->label('Activă')->default(true),
|
||||
]),
|
||||
Forms\Components\Textarea::make('notes')->label('Observații')->columnSpanFull()->rows(2),
|
||||
]);
|
||||
}
|
||||
|
||||
public static function table(Table $table): Table
|
||||
{
|
||||
return $table
|
||||
->columns([
|
||||
Tables\Columns\TextColumn::make('category')->label('Categorie')->badge()->sortable(),
|
||||
Tables\Columns\TextColumn::make('name_ro')->label('Manoperă')->searchable()->sortable(),
|
||||
Tables\Columns\TextColumn::make('hours')->label('Ore')->numeric(decimalPlaces: 2)->alignRight(),
|
||||
Tables\Columns\TextColumn::make('price')->label('Preț')->money('MDL')->alignRight(),
|
||||
Tables\Columns\IconColumn::make('is_active')->label('Activă')->boolean(),
|
||||
])
|
||||
->filters([
|
||||
Tables\Filters\SelectFilter::make('category')
|
||||
->options(array_combine(Labor::CATEGORIES, Labor::CATEGORIES)),
|
||||
Tables\Filters\TernaryFilter::make('is_active')->label('Doar active'),
|
||||
])
|
||||
->actions([
|
||||
Actions\EditAction::make(),
|
||||
Actions\DeleteAction::make(),
|
||||
])
|
||||
->defaultSort('category')
|
||||
->defaultGroup('category');
|
||||
}
|
||||
|
||||
public static function getPages(): array
|
||||
{
|
||||
return [
|
||||
'index' => Pages\ListLabors::route('/'),
|
||||
'create' => Pages\CreateLabor::route('/create'),
|
||||
'edit' => Pages\EditLabor::route('/{record}/edit'),
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\LaborResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\LaborResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\CreateRecord;
|
||||
|
||||
class CreateLabor extends CreateRecord
|
||||
{
|
||||
protected static string $resource = LaborResource::class;
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\LaborResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\LaborResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\EditRecord;
|
||||
|
||||
class EditLabor extends EditRecord
|
||||
{
|
||||
protected static string $resource = LaborResource::class;
|
||||
|
||||
protected function getHeaderActions(): array { return [Actions\DeleteAction::make()]; }
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\LaborResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\LaborResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\ListRecords;
|
||||
|
||||
class ListLabors extends ListRecords
|
||||
{
|
||||
protected static string $resource = LaborResource::class;
|
||||
|
||||
protected function getHeaderActions(): array { return [Actions\CreateAction::make()]; }
|
||||
}
|
||||
@@ -0,0 +1,112 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources;
|
||||
|
||||
use App\Filament\Tenant\Resources\MasterResource\Pages;
|
||||
use App\Models\Tenant\User;
|
||||
use Filament\Actions;
|
||||
use Filament\Forms;
|
||||
use Filament\Resources\Resource;
|
||||
use Filament\Schemas;
|
||||
use Filament\Schemas\Schema;
|
||||
use Filament\Tables;
|
||||
use Filament\Tables\Table;
|
||||
use Illuminate\Database\Eloquent\Builder;
|
||||
use Illuminate\Support\Facades\Hash;
|
||||
|
||||
/**
|
||||
* Tehnicieni — vedere filtrată peste users (role=mechanic).
|
||||
*/
|
||||
class MasterResource extends Resource
|
||||
{
|
||||
protected static ?string $model = User::class;
|
||||
|
||||
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-wrench';
|
||||
|
||||
protected static ?string $navigationLabel = 'Tehnicieni';
|
||||
|
||||
protected static string|\UnitEnum|null $navigationGroup = 'Service';
|
||||
|
||||
protected static ?string $modelLabel = 'tehnician';
|
||||
|
||||
protected static ?string $pluralModelLabel = 'tehnicieni';
|
||||
|
||||
protected static ?int $navigationSort = 33;
|
||||
|
||||
public static function getEloquentQuery(): Builder
|
||||
{
|
||||
return parent::getEloquentQuery()->where('role', 'mechanic');
|
||||
}
|
||||
|
||||
public static function form(Schema $schema): Schema
|
||||
{
|
||||
return $schema->components([
|
||||
Schemas\Components\Section::make('Date personale')
|
||||
->columns(2)
|
||||
->schema([
|
||||
Forms\Components\TextInput::make('name')->label('Nume')->required()->maxLength(120),
|
||||
Forms\Components\TextInput::make('phone')->label('Telefon')->tel()->maxLength(40),
|
||||
Forms\Components\TextInput::make('email')->label('Email')->email()->maxLength(120),
|
||||
Forms\Components\Select::make('status')
|
||||
->options(['active' => 'Activ', 'inactive' => 'Inactiv', 'blocked' => 'Blocat'])
|
||||
->default('active')
|
||||
->required(),
|
||||
]),
|
||||
Schemas\Components\Section::make('Profesie')
|
||||
->columns(2)
|
||||
->schema([
|
||||
Forms\Components\TextInput::make('specialization')
|
||||
->label('Specializare')
|
||||
->placeholder('Motor / Frâne / Electrică ...')
|
||||
->maxLength(120),
|
||||
Forms\Components\ColorPicker::make('color')->label('Culoare în calendar'),
|
||||
Forms\Components\TextInput::make('hourly_rate')->label('Tarif/oră')->numeric(),
|
||||
Forms\Components\Hidden::make('role')->default('mechanic'),
|
||||
]),
|
||||
Schemas\Components\Section::make('Acces în aplicație (opțional)')
|
||||
->columns(1)
|
||||
->collapsed()
|
||||
->schema([
|
||||
Forms\Components\TextInput::make('password')
|
||||
->label('Parolă (lasă gol pentru a nu schimba)')
|
||||
->password()
|
||||
->minLength(6)
|
||||
->dehydrated(fn ($state) => filled($state))
|
||||
->dehydrateStateUsing(fn ($state) => Hash::make($state)),
|
||||
]),
|
||||
]);
|
||||
}
|
||||
|
||||
public static function table(Table $table): Table
|
||||
{
|
||||
return $table
|
||||
->columns([
|
||||
Tables\Columns\ColorColumn::make('color')->label(''),
|
||||
Tables\Columns\TextColumn::make('name')->searchable()->sortable(),
|
||||
Tables\Columns\TextColumn::make('specialization')->label('Specializare')->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('phone')->copyable()->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('hourly_rate')->label('Tarif/h')->money('MDL')->alignRight()->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('status')
|
||||
->badge()
|
||||
->colors([
|
||||
'success' => ['active'],
|
||||
'warning' => ['inactive'],
|
||||
'danger' => ['blocked'],
|
||||
]),
|
||||
])
|
||||
->actions([
|
||||
Actions\EditAction::make(),
|
||||
Actions\DeleteAction::make(),
|
||||
])
|
||||
->defaultSort('name');
|
||||
}
|
||||
|
||||
public static function getPages(): array
|
||||
{
|
||||
return [
|
||||
'index' => Pages\ListMasters::route('/'),
|
||||
'create' => Pages\CreateMaster::route('/create'),
|
||||
'edit' => Pages\EditMaster::route('/{record}/edit'),
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\MasterResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\MasterResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\CreateRecord;
|
||||
|
||||
class CreateMaster extends CreateRecord
|
||||
{
|
||||
protected static string $resource = MasterResource::class;
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\MasterResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\MasterResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\EditRecord;
|
||||
|
||||
class EditMaster extends EditRecord
|
||||
{
|
||||
protected static string $resource = MasterResource::class;
|
||||
|
||||
protected function getHeaderActions(): array { return [Actions\DeleteAction::make()]; }
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\MasterResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\MasterResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\ListRecords;
|
||||
|
||||
class ListMasters extends ListRecords
|
||||
{
|
||||
protected static string $resource = MasterResource::class;
|
||||
|
||||
protected function getHeaderActions(): array { return [Actions\CreateAction::make()->label('Nou tehnician')]; }
|
||||
}
|
||||
@@ -0,0 +1,159 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources;
|
||||
|
||||
use App\Filament\Tenant\Resources\WorkOrderResource\Pages;
|
||||
use App\Filament\Tenant\Resources\WorkOrderResource\RelationManagers;
|
||||
use App\Models\Tenant\Client;
|
||||
use App\Models\Tenant\User;
|
||||
use App\Models\Tenant\Vehicle;
|
||||
use App\Models\Tenant\WorkOrder;
|
||||
use App\Tenancy\TenantManager;
|
||||
use Filament\Actions;
|
||||
use Filament\Forms;
|
||||
use Filament\Resources\Resource;
|
||||
use Filament\Schemas;
|
||||
use Filament\Schemas\Components\Utilities\Get;
|
||||
use Filament\Schemas\Schema;
|
||||
use Filament\Tables;
|
||||
use Filament\Tables\Table;
|
||||
|
||||
class WorkOrderResource extends Resource
|
||||
{
|
||||
protected static ?string $model = WorkOrder::class;
|
||||
|
||||
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-wrench-screwdriver';
|
||||
|
||||
protected static ?string $navigationLabel = 'Fișe lucru';
|
||||
|
||||
protected static string|\UnitEnum|null $navigationGroup = 'Service';
|
||||
|
||||
protected static ?string $modelLabel = 'fișă';
|
||||
|
||||
protected static ?string $pluralModelLabel = 'fișe lucru';
|
||||
|
||||
protected static ?int $navigationSort = 30;
|
||||
|
||||
public static function form(Schema $schema): Schema
|
||||
{
|
||||
return $schema->components([
|
||||
Schemas\Components\Section::make('Antet')
|
||||
->columns(3)
|
||||
->schema([
|
||||
Forms\Components\TextInput::make('number')
|
||||
->label('Nr.')
|
||||
->disabled()
|
||||
->dehydrated(false)
|
||||
->placeholder('Generat automat'),
|
||||
Forms\Components\DatePicker::make('opened_at')
|
||||
->label('Deschis')
|
||||
->default(today())
|
||||
->required(),
|
||||
Forms\Components\Select::make('status')
|
||||
->options(WorkOrder::STATUSES)
|
||||
->default('new')
|
||||
->required(),
|
||||
Forms\Components\Select::make('client_id')
|
||||
->label('Client')
|
||||
->options(fn () => Client::pluck('name', 'id'))
|
||||
->searchable()
|
||||
->live()
|
||||
->required(),
|
||||
Forms\Components\Select::make('vehicle_id')
|
||||
->label('Auto')
|
||||
->options(fn (Get $get) => $get('client_id')
|
||||
? Vehicle::where('client_id', $get('client_id'))
|
||||
->get()
|
||||
->mapWithKeys(fn ($v) => [$v->id => "{$v->make} {$v->model} {$v->plate}"])
|
||||
->toArray()
|
||||
: [])
|
||||
->searchable(),
|
||||
Forms\Components\Select::make('master_id')
|
||||
->label('Maistru')
|
||||
->options(fn () => User::where('status', 'active')->pluck('name', 'id'))
|
||||
->searchable(),
|
||||
Forms\Components\TextInput::make('mileage_in')->label('Km la intrare')->numeric(),
|
||||
Forms\Components\TextInput::make('mileage_out')->label('Km la ieșire')->numeric(),
|
||||
]),
|
||||
Schemas\Components\Section::make('Diagnostic')
|
||||
->collapsible()
|
||||
->schema([
|
||||
Forms\Components\Textarea::make('complaint')->label('Plângere client')->rows(2)->columnSpanFull(),
|
||||
Forms\Components\Textarea::make('diagnosis')->label('Diagnostic')->rows(3)->columnSpanFull(),
|
||||
Forms\Components\Textarea::make('recommendations')->label('Recomandări')->rows(2)->columnSpanFull(),
|
||||
]),
|
||||
Schemas\Components\Section::make('Plată & total')
|
||||
->columns(3)
|
||||
->schema([
|
||||
Forms\Components\Select::make('pay_status')
|
||||
->options(WorkOrder::PAY_STATUSES)
|
||||
->default('unpaid')
|
||||
->required(),
|
||||
Forms\Components\TextInput::make('discount_pct')->label('Discount %')->numeric()->default(0),
|
||||
Forms\Components\TextInput::make('total')->label('Total')->numeric()->disabled()->dehydrated(false),
|
||||
Forms\Components\Toggle::make('approved')->label('Aprobat de client'),
|
||||
Forms\Components\DatePicker::make('closed_at')->label('Închis'),
|
||||
]),
|
||||
]);
|
||||
}
|
||||
|
||||
public static function table(Table $table): Table
|
||||
{
|
||||
return $table
|
||||
->columns([
|
||||
Tables\Columns\TextColumn::make('number')->label('Nr.')->searchable()->sortable(),
|
||||
Tables\Columns\TextColumn::make('opened_at')->label('Deschis')->date('d.m.Y')->sortable(),
|
||||
Tables\Columns\TextColumn::make('client.name')->label('Client')->searchable(),
|
||||
Tables\Columns\TextColumn::make('vehicle.plate')->label('Auto')->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('master.name')->label('Maistru')->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('status')
|
||||
->formatStateUsing(fn ($state) => WorkOrder::STATUSES[$state] ?? $state)
|
||||
->badge()
|
||||
->colors([
|
||||
'gray' => ['new'],
|
||||
'info' => ['diagnosis', 'agreement', 'approved'],
|
||||
'warning' => ['in_work', 'awaiting_parts'],
|
||||
'success' => ['ready', 'done'],
|
||||
'danger' => ['cancelled'],
|
||||
]),
|
||||
Tables\Columns\TextColumn::make('pay_status')
|
||||
->formatStateUsing(fn ($state) => WorkOrder::PAY_STATUSES[$state] ?? $state)
|
||||
->badge()
|
||||
->colors([
|
||||
'danger' => ['unpaid'],
|
||||
'warning' => ['partial'],
|
||||
'success' => ['paid'],
|
||||
]),
|
||||
Tables\Columns\TextColumn::make('total')->money('MDL')->alignRight()->sortable(),
|
||||
])
|
||||
->filters([
|
||||
Tables\Filters\SelectFilter::make('status')->options(WorkOrder::STATUSES),
|
||||
Tables\Filters\SelectFilter::make('pay_status')->options(WorkOrder::PAY_STATUSES),
|
||||
Tables\Filters\SelectFilter::make('master_id')
|
||||
->label('Maistru')
|
||||
->options(fn () => User::pluck('name', 'id')),
|
||||
])
|
||||
->actions([
|
||||
Actions\EditAction::make(),
|
||||
Actions\DeleteAction::make(),
|
||||
])
|
||||
->defaultSort('opened_at', 'desc');
|
||||
}
|
||||
|
||||
public static function getRelations(): array
|
||||
{
|
||||
return [
|
||||
RelationManagers\WorksRelationManager::class,
|
||||
RelationManagers\PartsRelationManager::class,
|
||||
];
|
||||
}
|
||||
|
||||
public static function getPages(): array
|
||||
{
|
||||
return [
|
||||
'index' => Pages\ListWorkOrders::route('/'),
|
||||
'create' => Pages\CreateWorkOrder::route('/create'),
|
||||
'edit' => Pages\EditWorkOrder::route('/{record}/edit'),
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,20 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\WorkOrderResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\WorkOrderResource;
|
||||
use App\Models\Tenant\WorkOrder;
|
||||
use App\Tenancy\TenantManager;
|
||||
use Filament\Resources\Pages\CreateRecord;
|
||||
|
||||
class CreateWorkOrder extends CreateRecord
|
||||
{
|
||||
protected static string $resource = WorkOrderResource::class;
|
||||
|
||||
protected function mutateFormDataBeforeCreate(array $data): array
|
||||
{
|
||||
$companyId = app(TenantManager::class)->currentId();
|
||||
$data['number'] = WorkOrder::generateNumber($companyId);
|
||||
return $data;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,17 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\WorkOrderResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\WorkOrderResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\EditRecord;
|
||||
|
||||
class EditWorkOrder extends EditRecord
|
||||
{
|
||||
protected static string $resource = WorkOrderResource::class;
|
||||
|
||||
protected function getHeaderActions(): array
|
||||
{
|
||||
return [Actions\DeleteAction::make()];
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,17 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\WorkOrderResource\Pages;
|
||||
|
||||
use App\Filament\Tenant\Resources\WorkOrderResource;
|
||||
use Filament\Actions;
|
||||
use Filament\Resources\Pages\ListRecords;
|
||||
|
||||
class ListWorkOrders extends ListRecords
|
||||
{
|
||||
protected static string $resource = WorkOrderResource::class;
|
||||
|
||||
protected function getHeaderActions(): array
|
||||
{
|
||||
return [Actions\CreateAction::make()];
|
||||
}
|
||||
}
|
||||
+67
@@ -0,0 +1,67 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\WorkOrderResource\RelationManagers;
|
||||
|
||||
use App\Models\Tenant\WorkOrderPart;
|
||||
use Filament\Actions;
|
||||
use Filament\Forms;
|
||||
use Filament\Resources\RelationManagers\RelationManager;
|
||||
use Filament\Schemas\Schema;
|
||||
use Filament\Tables;
|
||||
use Filament\Tables\Table;
|
||||
|
||||
class PartsRelationManager extends RelationManager
|
||||
{
|
||||
protected static string $relationship = 'parts';
|
||||
|
||||
protected static ?string $title = 'Piese';
|
||||
|
||||
public function form(Schema $schema): Schema
|
||||
{
|
||||
return $schema->components([
|
||||
Forms\Components\TextInput::make('name')->label('Denumire')->required()->columnSpanFull(),
|
||||
Forms\Components\TextInput::make('article')->label('Cod articol')->maxLength(64),
|
||||
Forms\Components\TextInput::make('brand')->label('Brand')->maxLength(64),
|
||||
Forms\Components\TextInput::make('qty')->label('Cantitate')->numeric()->default(1)->required(),
|
||||
Forms\Components\TextInput::make('unit')->label('UM')->maxLength(16)->default('buc'),
|
||||
Forms\Components\TextInput::make('buy_price')->label('Preț achiziție')->numeric()->default(0),
|
||||
Forms\Components\TextInput::make('sell_price')->label('Preț vânzare')->numeric()->required(),
|
||||
Forms\Components\TextInput::make('discount_pct')->label('Discount %')->numeric()->default(0),
|
||||
Forms\Components\Select::make('status')
|
||||
->options(WorkOrderPart::STATUSES)
|
||||
->default('needed')
|
||||
->required(),
|
||||
Forms\Components\Textarea::make('notes')->label('Observații')->columnSpanFull()->rows(2),
|
||||
]);
|
||||
}
|
||||
|
||||
public function table(Table $table): Table
|
||||
{
|
||||
return $table
|
||||
->recordTitleAttribute('name')
|
||||
->columns([
|
||||
Tables\Columns\TextColumn::make('name')->label('Piesă')->wrap(),
|
||||
Tables\Columns\TextColumn::make('article')->label('Cod')->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('brand')->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('qty')->label('Cant.')->alignRight(),
|
||||
Tables\Columns\TextColumn::make('sell_price')->label('Preț')->money('MDL')->alignRight(),
|
||||
Tables\Columns\TextColumn::make('total')->money('MDL')->alignRight(),
|
||||
Tables\Columns\TextColumn::make('status')
|
||||
->formatStateUsing(fn ($s) => WorkOrderPart::STATUSES[$s] ?? $s)
|
||||
->badge()
|
||||
->colors([
|
||||
'gray' => ['needed'],
|
||||
'warning' => ['ordered'],
|
||||
'info' => ['delivered'],
|
||||
'success' => ['installed'],
|
||||
]),
|
||||
])
|
||||
->headerActions([
|
||||
Actions\CreateAction::make(),
|
||||
])
|
||||
->actions([
|
||||
Actions\EditAction::make(),
|
||||
Actions\DeleteAction::make(),
|
||||
]);
|
||||
}
|
||||
}
|
||||
+79
@@ -0,0 +1,79 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Resources\WorkOrderResource\RelationManagers;
|
||||
|
||||
use App\Models\Tenant\Labor;
|
||||
use App\Models\Tenant\User;
|
||||
use App\Models\Tenant\WorkOrderWork;
|
||||
use Filament\Actions;
|
||||
use Filament\Forms;
|
||||
use Filament\Resources\RelationManagers\RelationManager;
|
||||
use Filament\Schemas\Components\Utilities\Set;
|
||||
use Filament\Schemas\Schema;
|
||||
use Filament\Tables;
|
||||
use Filament\Tables\Table;
|
||||
|
||||
class WorksRelationManager extends RelationManager
|
||||
{
|
||||
protected static string $relationship = 'works';
|
||||
|
||||
protected static ?string $title = 'Manopere';
|
||||
|
||||
public function form(Schema $schema): Schema
|
||||
{
|
||||
return $schema->components([
|
||||
Forms\Components\Select::make('labor_id')
|
||||
->label('Catalog manoperă')
|
||||
->options(fn () => Labor::where('is_active', true)
|
||||
->get()
|
||||
->mapWithKeys(fn ($l) => [$l->id => "[{$l->category}] {$l->name_ro} ({$l->hours}h)"])
|
||||
->toArray())
|
||||
->searchable()
|
||||
->live()
|
||||
->afterStateUpdated(function ($state, Set $set) {
|
||||
if ($state && $labor = Labor::find($state)) {
|
||||
$set('name', $labor->name_ro);
|
||||
$set('hours', $labor->hours);
|
||||
$set('price_per_hour', $labor->hours > 0 ? round($labor->price / max((float) $labor->hours, 1), 2) : 0);
|
||||
}
|
||||
})
|
||||
->columnSpanFull(),
|
||||
Forms\Components\TextInput::make('name')->label('Nume')->required()->columnSpanFull(),
|
||||
Forms\Components\TextInput::make('hours')->label('Ore')->numeric()->default(1)->required(),
|
||||
Forms\Components\TextInput::make('price_per_hour')->label('Preț/h')->numeric()->required(),
|
||||
Forms\Components\Select::make('master_id')
|
||||
->label('Maistru')
|
||||
->options(fn () => User::pluck('name', 'id'))
|
||||
->searchable(),
|
||||
Forms\Components\Select::make('status')
|
||||
->options(WorkOrderWork::STATUSES)
|
||||
->default('todo')
|
||||
->required(),
|
||||
Forms\Components\Textarea::make('notes')->label('Notițe')->columnSpanFull()->rows(2),
|
||||
]);
|
||||
}
|
||||
|
||||
public function table(Table $table): Table
|
||||
{
|
||||
return $table
|
||||
->recordTitleAttribute('name')
|
||||
->columns([
|
||||
Tables\Columns\TextColumn::make('name')->label('Manoperă')->wrap(),
|
||||
Tables\Columns\TextColumn::make('hours')->label('Ore')->numeric(decimalPlaces: 2)->alignRight(),
|
||||
Tables\Columns\TextColumn::make('price_per_hour')->label('Preț/h')->money('MDL')->alignRight(),
|
||||
Tables\Columns\TextColumn::make('total')->label('Total')->money('MDL')->alignRight(),
|
||||
Tables\Columns\TextColumn::make('master.name')->label('Maistru')->placeholder('—'),
|
||||
Tables\Columns\TextColumn::make('status')
|
||||
->formatStateUsing(fn ($s) => WorkOrderWork::STATUSES[$s] ?? $s)
|
||||
->badge()
|
||||
->colors(['gray' => ['todo'], 'warning' => ['in_progress'], 'success' => ['done']]),
|
||||
])
|
||||
->headerActions([
|
||||
Actions\CreateAction::make(),
|
||||
])
|
||||
->actions([
|
||||
Actions\EditAction::make(),
|
||||
Actions\DeleteAction::make(),
|
||||
]);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user