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 4.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. <?php
  2. namespace App\Filament\Company\Resources\Sales\InvoiceResource\Widgets;
  3. use App\Enums\Accounting\InvoiceStatus;
  4. use App\Filament\Company\Resources\Sales\InvoiceResource\Pages\ListInvoices;
  5. use App\Filament\Widgets\EnhancedStatsOverviewWidget;
  6. use App\Utilities\Currency\CurrencyAccessor;
  7. use App\Utilities\Currency\CurrencyConverter;
  8. use Filament\Widgets\Concerns\InteractsWithPageTable;
  9. use Illuminate\Support\Facades\DB;
  10. use Illuminate\Support\Number;
  11. class InvoiceOverview extends EnhancedStatsOverviewWidget
  12. {
  13. use InteractsWithPageTable;
  14. protected function getTablePage(): string
  15. {
  16. return ListInvoices::class;
  17. }
  18. protected function getStats(): array
  19. {
  20. $activeTab = $this->activeTab;
  21. if ($activeTab === 'draft') {
  22. $draftInvoices = $this->getPageTableQuery();
  23. $totalDraftCount = $draftInvoices->count();
  24. $totalDraftAmount = $draftInvoices->get()->sumMoneyInDefaultCurrency('total');
  25. $averageDraftTotal = $totalDraftCount > 0
  26. ? (int) round($totalDraftAmount / $totalDraftCount)
  27. : 0;
  28. return [
  29. EnhancedStatsOverviewWidget\EnhancedStat::make('Total Unpaid', '-'),
  30. EnhancedStatsOverviewWidget\EnhancedStat::make('Due Within 30 Days', '-'),
  31. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Payment Time', '-'),
  32. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Invoice Total', CurrencyConverter::formatCentsToMoney($averageDraftTotal))
  33. ->suffix(CurrencyAccessor::getDefaultCurrency()),
  34. ];
  35. }
  36. $unpaidInvoices = $this->getPageTableQuery()->unpaid();
  37. $amountUnpaid = $unpaidInvoices->get()->sumMoneyInDefaultCurrency('amount_due');
  38. $amountOverdue = $unpaidInvoices
  39. ->clone()
  40. ->where('status', InvoiceStatus::Overdue)
  41. ->get()
  42. ->sumMoneyInDefaultCurrency('amount_due');
  43. $amountDueWithin30Days = $unpaidInvoices
  44. ->clone()
  45. ->whereBetween('due_date', [company_today(), company_today()->addMonth()])
  46. ->get()
  47. ->sumMoneyInDefaultCurrency('amount_due');
  48. $validInvoices = $this->getPageTableQuery()
  49. ->whereNotIn('status', [
  50. InvoiceStatus::Void,
  51. InvoiceStatus::Draft,
  52. ]);
  53. $totalValidInvoiceAmount = $validInvoices->get()->sumMoneyInDefaultCurrency('total');
  54. $totalValidInvoiceCount = $validInvoices->count();
  55. $averageInvoiceTotal = $totalValidInvoiceCount > 0
  56. ? (int) round($totalValidInvoiceAmount / $totalValidInvoiceCount)
  57. : 0;
  58. $averagePaymentTimeFormatted = '-';
  59. $averagePaymentTimeSuffix = null;
  60. if ($activeTab !== 'unpaid') {
  61. $driver = DB::getDriverName();
  62. $query = $this->getPageTableQuery()
  63. ->whereNotNull('paid_at');
  64. if ($driver === 'pgsql') {
  65. $query->selectRaw('AVG(EXTRACT(EPOCH FROM (paid_at - approved_at)) / 86400) as avg_days');
  66. } elseif ($driver === 'sqlite') {
  67. $query->selectRaw('AVG(julianday(paid_at) - julianday(approved_at)) as avg_days');
  68. } else {
  69. $query->selectRaw('AVG(TIMESTAMPDIFF(DAY, approved_at, paid_at)) as avg_days');
  70. }
  71. $averagePaymentTime = $query
  72. ->groupBy('company_id')
  73. ->reorder()
  74. ->value('avg_days');
  75. $averagePaymentTimeFormatted = Number::format($averagePaymentTime ?? 0, maxPrecision: 1);
  76. $averagePaymentTimeSuffix = 'days';
  77. }
  78. return [
  79. EnhancedStatsOverviewWidget\EnhancedStat::make('Total Unpaid', CurrencyConverter::formatCentsToMoney($amountUnpaid))
  80. ->suffix(CurrencyAccessor::getDefaultCurrency())
  81. ->description('Includes ' . CurrencyConverter::formatCentsToMoney($amountOverdue) . ' overdue'),
  82. EnhancedStatsOverviewWidget\EnhancedStat::make('Due Within 30 Days', CurrencyConverter::formatCentsToMoney($amountDueWithin30Days))
  83. ->suffix(CurrencyAccessor::getDefaultCurrency()),
  84. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Payment Time', $averagePaymentTimeFormatted)
  85. ->suffix($averagePaymentTimeSuffix),
  86. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Invoice Total', CurrencyConverter::formatCentsToMoney($averageInvoiceTotal))
  87. ->suffix(CurrencyAccessor::getDefaultCurrency())
  88. ->description($activeTab === 'all' ? 'Excludes draft and voided invoices' : null),
  89. ];
  90. }
  91. }