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.

BillOverview.php 3.1KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. <?php
  2. namespace App\Filament\Company\Resources\Purchases\VendorResource\Widgets;
  3. use App\Enums\Accounting\BillStatus;
  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 BillOverview extends EnhancedStatsOverviewWidget
  11. {
  12. public ?Model $record = null;
  13. protected function getStats(): array
  14. {
  15. $unpaidBills = $this->record->bills()
  16. ->whereIn('status', [BillStatus::Open, BillStatus::Partial, BillStatus::Overdue]);
  17. $amountToPay = $unpaidBills->get()->sumMoneyInDefaultCurrency('amount_due');
  18. $amountOverdue = $unpaidBills
  19. ->clone()
  20. ->where('status', BillStatus::Overdue)
  21. ->get()
  22. ->sumMoneyInDefaultCurrency('amount_due');
  23. $amountDueWithin7Days = $unpaidBills
  24. ->clone()
  25. ->whereBetween('due_date', [company_today(), company_today()->addWeek()])
  26. ->get()
  27. ->sumMoneyInDefaultCurrency('amount_due');
  28. $driver = DB::getDriverName();
  29. $query = $this->record->bills()
  30. ->whereNotNull('paid_at');
  31. if ($driver === 'pgsql') {
  32. $query->selectRaw('AVG(EXTRACT(EPOCH FROM (paid_at - date)) / 86400) as avg_days');
  33. } elseif ($driver === 'sqlite') {
  34. $query->selectRaw('AVG(julianday(paid_at) - julianday(date)) as avg_days');
  35. } else {
  36. $query->selectRaw('AVG(TIMESTAMPDIFF(DAY, date, paid_at)) as avg_days');
  37. }
  38. $averagePaymentTime = $query
  39. ->groupBy('company_id')
  40. ->reorder()
  41. ->value('avg_days');
  42. $averagePaymentTimeFormatted = Number::format($averagePaymentTime ?? 0, maxPrecision: 1);
  43. $lastMonthTotal = $this->record->bills()
  44. ->where('status', BillStatus::Paid)
  45. ->whereBetween('date', [
  46. company_today()->subMonth()->startOfMonth(),
  47. company_today()->subMonth()->endOfMonth(),
  48. ])
  49. ->get()
  50. ->sumMoneyInDefaultCurrency('amount_paid');
  51. return [
  52. EnhancedStatsOverviewWidget\EnhancedStat::make('Total To Pay', CurrencyConverter::formatCentsToMoney($amountToPay))
  53. ->suffix(CurrencyAccessor::getDefaultCurrency())
  54. ->description('Includes ' . CurrencyConverter::formatCentsToMoney($amountOverdue) . ' overdue'),
  55. EnhancedStatsOverviewWidget\EnhancedStat::make('Due Within 7 Days', CurrencyConverter::formatCentsToMoney($amountDueWithin7Days))
  56. ->suffix(CurrencyAccessor::getDefaultCurrency()),
  57. EnhancedStatsOverviewWidget\EnhancedStat::make('Average Payment Time', $averagePaymentTimeFormatted)
  58. ->suffix('days'),
  59. EnhancedStatsOverviewWidget\EnhancedStat::make('Paid Last Month', CurrencyConverter::formatCentsToMoney($lastMonthTotal))
  60. ->suffix(CurrencyAccessor::getDefaultCurrency()),
  61. ];
  62. }
  63. }