您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

InvoiceOverview.php 2.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  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\Number;
  9. class InvoiceOverview extends EnhancedStatsOverviewWidget
  10. {
  11. public ?Model $record = null;
  12. protected function getStats(): array
  13. {
  14. $unpaidInvoices = $this->record->invoices()->unpaid();
  15. $amountUnpaid = $unpaidInvoices->get()->sumMoneyInDefaultCurrency('amount_due');
  16. $amountOverdue = $unpaidInvoices->clone()
  17. ->where('status', InvoiceStatus::Overdue)
  18. ->get()
  19. ->sumMoneyInDefaultCurrency('amount_due');
  20. $amountDueWithin30Days = $unpaidInvoices->clone()
  21. ->whereBetween('due_date', [today(), today()->addMonth()])
  22. ->get()
  23. ->sumMoneyInDefaultCurrency('amount_due');
  24. $validInvoices = $this->record->invoices()
  25. ->whereNotIn('status', [
  26. InvoiceStatus::Void,
  27. InvoiceStatus::Draft,
  28. ]);
  29. $totalValidInvoiceAmount = $validInvoices->get()->sumMoneyInDefaultCurrency('total');
  30. $totalValidInvoiceCount = $validInvoices->count();
  31. $averageInvoiceTotal = $totalValidInvoiceCount > 0
  32. ? (int) round($totalValidInvoiceAmount / $totalValidInvoiceCount)
  33. : 0;
  34. $averagePaymentTime = $this->record->invoices()
  35. ->whereNotNull('paid_at')
  36. ->selectRaw('AVG(TIMESTAMPDIFF(DAY, date, paid_at)) as avg_days')
  37. ->value('avg_days');
  38. $averagePaymentTimeFormatted = Number::format($averagePaymentTime ?? 0, maxPrecision: 1);
  39. return [
  40. EnhancedStatsOverviewWidget\EnhancedStat::make('Total Unpaid', CurrencyConverter::formatCentsToMoney($amountUnpaid))
  41. ->suffix(CurrencyAccessor::getDefaultCurrency())
  42. ->description('Includes ' . CurrencyConverter::formatCentsToMoney($amountOverdue) . ' overdue'),
  43. EnhancedStatsOverviewWidget\EnhancedStat::make('Due Within 30 Days', CurrencyConverter::formatCentsToMoney($amountDueWithin30Days))
  44. ->suffix(CurrencyAccessor::getDefaultCurrency()),
  45. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Payment Time', $averagePaymentTimeFormatted)
  46. ->suffix('days'),
  47. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Invoice Total', CurrencyConverter::formatCentsToMoney($averageInvoiceTotal))
  48. ->suffix(CurrencyAccessor::getDefaultCurrency())
  49. ->description('Excludes draft and voided invoices'),
  50. ];
  51. }
  52. }