Faza 6: Activity log + Kanban + Payroll + cleanup
══════ Activity log (Spatie) ══════ - spatie/laravel-activitylog v5 instalat - Migration cu company_id pentru tenant scoping - Trait Auditable (App\Models\Concerns\Auditable): - LogOptions cu logFillable + logOnlyDirty + dontSubmitEmptyLogs - tapActivity auto-fill company_id + causer - Descrieri RO (creat/modificat/șters/restaurat) - Aplicat pe: Client, Vehicle, Lead, Deal, WorkOrder, Payment, Expense - ActivityResource (group Admin → Jurnal activitate) - Listă read-only, scope pe tenant, filtre by description/today ══════ Kanban Work Orders ══════ - Custom Filament page la /app/kanban (group Service) - 6 coloane (new → diagnosis → agreement → in_work → awaiting_parts → ready) - Drag-drop nativ HTML5 cu wire:click moveCard() - Cards arată: număr fișă, client, auto, plate, master, total - Link 'Deschide' direct la editare WO ══════ Payroll (Salarii) ══════ Schema: - employee_profiles: user_id, position, base_salary, works_pct, parts_pct - payroll_runs: period (YYYY-MM), base, works_revenue/pct, parts_margin/pct, bonus, fines, advance, total auto-calculat - payroll_adjustments: bonus/fine/advance per period PayrollCalculator service: - compute($userId, $period) — calculează auto: - Manopere finalizate de mecanic în luna respectivă (sum total) - Marja pieselor montate de el (sell-buy * qty) - Bonus + fines + advance from adjustments - Total = base + works% + parts% + bonus - fines - advance Resources Filament (group Finanțe): - EmployeeProfileResource: profil cu % comisioane - PayrollRunResource: salarii cu action 'Calculează luna curentă' (toți userii) + per-row 'Recalculează'; Sum summary pe total - PayrollAdjustmentResource: gestionare bonus/penalizări/avansuri ══════ Cleanup ══════ - Șterse toate /__debug, /__seed, /__try-login, /__force-login, /__whoami, /__coolify-check (security) - Routes/web.php conține doar / redirect, /manifest.json, /sw.js Total Filament tenant routes: 92.
This commit is contained in:
@@ -0,0 +1,74 @@
|
||||
<?php
|
||||
|
||||
use Illuminate\Database\Migrations\Migration;
|
||||
use Illuminate\Database\Schema\Blueprint;
|
||||
use Illuminate\Support\Facades\Schema;
|
||||
|
||||
return new class extends Migration
|
||||
{
|
||||
public function up(): void
|
||||
{
|
||||
Schema::create('employee_profiles', function (Blueprint $t) {
|
||||
$t->id();
|
||||
$t->foreignId('company_id')->constrained()->cascadeOnDelete();
|
||||
$t->foreignId('user_id')->constrained()->cascadeOnDelete();
|
||||
$t->string('position')->nullable(); // Mecanic / Maistru / Recepție / Magazioner
|
||||
$t->decimal('base_salary', 10, 2)->default(0);
|
||||
$t->decimal('works_pct', 5, 2)->default(0); // % din venitul manoperelor finalizate
|
||||
$t->decimal('parts_pct', 5, 2)->default(0); // % din marja pieselor vândute
|
||||
$t->date('hire_date')->nullable();
|
||||
$t->text('notes')->nullable();
|
||||
$t->timestamps();
|
||||
$t->softDeletes();
|
||||
|
||||
$t->unique(['company_id', 'user_id']);
|
||||
});
|
||||
|
||||
Schema::create('payroll_runs', function (Blueprint $t) {
|
||||
$t->id();
|
||||
$t->foreignId('company_id')->constrained()->cascadeOnDelete();
|
||||
$t->foreignId('user_id')->constrained()->cascadeOnDelete();
|
||||
$t->string('period', 7); // YYYY-MM
|
||||
$t->decimal('base', 10, 2)->default(0);
|
||||
$t->decimal('works_revenue', 12, 2)->default(0);
|
||||
$t->decimal('works_pct_amount', 10, 2)->default(0);
|
||||
$t->decimal('parts_margin', 12, 2)->default(0);
|
||||
$t->decimal('parts_pct_amount', 10, 2)->default(0);
|
||||
$t->decimal('bonus', 10, 2)->default(0);
|
||||
$t->decimal('fines', 10, 2)->default(0);
|
||||
$t->decimal('advance', 10, 2)->default(0);
|
||||
$t->decimal('total', 10, 2)->default(0);
|
||||
$t->boolean('paid')->default(false);
|
||||
$t->date('paid_at')->nullable();
|
||||
$t->text('notes')->nullable();
|
||||
$t->timestamps();
|
||||
|
||||
$t->unique(['company_id', 'user_id', 'period']);
|
||||
$t->index(['company_id', 'period']);
|
||||
});
|
||||
|
||||
Schema::create('payroll_adjustments', function (Blueprint $t) {
|
||||
$t->id();
|
||||
$t->foreignId('company_id')->constrained()->cascadeOnDelete();
|
||||
$t->foreignId('user_id')->constrained()->cascadeOnDelete();
|
||||
$t->string('type'); // bonus / fine / advance
|
||||
$t->decimal('amount', 10, 2);
|
||||
$t->string('period', 7)->nullable(); // YYYY-MM (la care se aplică)
|
||||
$t->date('date')->default(now());
|
||||
$t->string('reason')->nullable();
|
||||
$t->boolean('applied')->default(false); // marcat true după payroll run
|
||||
$t->timestamps();
|
||||
$t->softDeletes();
|
||||
|
||||
$t->index(['company_id', 'user_id', 'period']);
|
||||
$t->index(['company_id', 'type']);
|
||||
});
|
||||
}
|
||||
|
||||
public function down(): void
|
||||
{
|
||||
Schema::dropIfExists('payroll_adjustments');
|
||||
Schema::dropIfExists('payroll_runs');
|
||||
Schema::dropIfExists('employee_profiles');
|
||||
}
|
||||
};
|
||||
Reference in New Issue
Block a user