You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

InvoiceOverview.php 3.2KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. <?php
  2. namespace App\Filament\Company\Resources\Sales\ClientResource\Widgets;
  3. use App\Enums\Accounting\InvoiceStatus;
  4. use App\Filament\Widgets\EnhancedStatsOverviewWidget;
  5. use App\Utilities\Currency\CurrencyAccessor;
  6. use App\Utilities\Currency\CurrencyConverter;
  7. use Illuminate\Database\Eloquent\Model;
  8. use Illuminate\Support\Facades\DB;
  9. use Illuminate\Support\Number;
  10. class InvoiceOverview extends EnhancedStatsOverviewWidget
  11. {
  12. public ?Model $record = null;
  13. protected function getStats(): array
  14. {
  15. $unpaidInvoices = $this->record->invoices()->unpaid();
  16. $amountUnpaid = $unpaidInvoices->get()->sumMoneyInDefaultCurrency('amount_due');
  17. $amountOverdue = $unpaidInvoices->clone()
  18. ->where('status', InvoiceStatus::Overdue)
  19. ->get()
  20. ->sumMoneyInDefaultCurrency('amount_due');
  21. $amountDueWithin30Days = $unpaidInvoices->clone()
  22. ->whereBetween('due_date', [company_today(), company_today()->addMonth()])
  23. ->get()
  24. ->sumMoneyInDefaultCurrency('amount_due');
  25. $validInvoices = $this->record->invoices()
  26. ->whereNotIn('status', [
  27. InvoiceStatus::Void,
  28. InvoiceStatus::Draft,
  29. ]);
  30. $totalValidInvoiceAmount = $validInvoices->get()->sumMoneyInDefaultCurrency('total');
  31. $totalValidInvoiceCount = $validInvoices->count();
  32. $averageInvoiceTotal = $totalValidInvoiceCount > 0
  33. ? (int) round($totalValidInvoiceAmount / $totalValidInvoiceCount)
  34. : 0;
  35. $driver = DB::getDriverName();
  36. $query = $this->record->invoices()
  37. ->whereNotNull('paid_at');
  38. if ($driver === 'pgsql') {
  39. $query->selectRaw('AVG(EXTRACT(EPOCH FROM (paid_at - date)) / 86400) as avg_days');
  40. } elseif ($driver === 'sqlite') {
  41. $query->selectRaw('AVG(julianday(paid_at) - julianday(date)) as avg_days');
  42. } else {
  43. $query->selectRaw('AVG(TIMESTAMPDIFF(DAY, date, paid_at)) as avg_days');
  44. }
  45. $averagePaymentTime = $query
  46. ->groupBy('company_id')
  47. ->reorder()
  48. ->value('avg_days');
  49. $averagePaymentTimeFormatted = Number::format($averagePaymentTime ?? 0, maxPrecision: 1);
  50. return [
  51. EnhancedStatsOverviewWidget\EnhancedStat::make('Total Unpaid', CurrencyConverter::formatCentsToMoney($amountUnpaid))
  52. ->suffix(CurrencyAccessor::getDefaultCurrency())
  53. ->description('Includes ' . CurrencyConverter::formatCentsToMoney($amountOverdue) . ' overdue'),
  54. EnhancedStatsOverviewWidget\EnhancedStat::make('Due Within 30 Days', CurrencyConverter::formatCentsToMoney($amountDueWithin30Days))
  55. ->suffix(CurrencyAccessor::getDefaultCurrency()),
  56. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Payment Time', $averagePaymentTimeFormatted)
  57. ->suffix('days'),
  58. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Invoice Total', CurrencyConverter::formatCentsToMoney($averageInvoiceTotal))
  59. ->suffix(CurrencyAccessor::getDefaultCurrency())
  60. ->description('Excludes draft and voided invoices'),
  61. ];
  62. }
  63. }