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.

DateRangeSelect.php 4.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. <?php
  2. namespace App\Filament\Forms\Components;
  3. use App\Services\DateRangeService;
  4. use Filament\Forms\Components\Select;
  5. use Filament\Forms\Set;
  6. use Illuminate\Support\Carbon;
  7. class DateRangeSelect extends Select
  8. {
  9. public string $fiscalYearStartDate;
  10. public string $fiscalYearEndDate;
  11. public ?string $startDateField = null;
  12. public ?string $endDateField = null;
  13. protected function setUp(): void
  14. {
  15. parent::setUp();
  16. $company = auth()->user()->currentCompany;
  17. $this->fiscalYearStartDate = $company->locale->fiscalYearStartDate();
  18. $this->fiscalYearEndDate = $company->locale->fiscalYearEndDate();
  19. $this->options(app(DateRangeService::class)->getDateRangeOptions())
  20. ->live()
  21. ->afterStateUpdated(function ($state, Set $set) {
  22. if ($this->startDateField && $this->endDateField) {
  23. $this->updateDateRange($state, $set);
  24. }
  25. });
  26. }
  27. public function startDateField(string $fieldName): static
  28. {
  29. $this->startDateField = $fieldName;
  30. return $this;
  31. }
  32. public function endDateField(string $fieldName): static
  33. {
  34. $this->endDateField = $fieldName;
  35. return $this;
  36. }
  37. public function updateDateRange($state, Set $set): void
  38. {
  39. if ($state === null) {
  40. $set($this->startDateField, null);
  41. $set($this->endDateField, null);
  42. return;
  43. }
  44. [$type, $param1, $param2] = explode('-', $state) + [null, null, null];
  45. $this->processDateRange($type, $param1, $param2, $set);
  46. }
  47. public function processDateRange($type, $param1, $param2, Set $set): void
  48. {
  49. match ($type) {
  50. 'FY' => $this->processFiscalYear($param1, $set),
  51. 'FQ' => $this->processFiscalQuarter($param1, $param2, $set),
  52. 'Y' => $this->processCalendarYear($param1, $set),
  53. 'Q' => $this->processCalendarQuarter($param1, $param2, $set),
  54. 'M' => $this->processMonth("{$param1}-{$param2}", $set),
  55. 'Custom' => null,
  56. };
  57. }
  58. public function processFiscalYear($year, Set $set): void
  59. {
  60. $currentYear = now()->year;
  61. $diff = $currentYear - $year;
  62. $fiscalYearStart = Carbon::parse($this->fiscalYearStartDate)->subYears($diff);
  63. $fiscalYearEnd = Carbon::parse($this->fiscalYearEndDate)->subYears($diff);
  64. $this->setDateRange($fiscalYearStart, $fiscalYearEnd, $set);
  65. }
  66. public function processFiscalQuarter($quarter, $year, Set $set): void
  67. {
  68. $currentYear = now()->year;
  69. $diff = $currentYear - $year;
  70. $fiscalYearStart = Carbon::parse($this->fiscalYearStartDate)->subYears($diff);
  71. $quarterStart = $fiscalYearStart->copy()->addMonths(($quarter - 1) * 3);
  72. $quarterEnd = $quarterStart->copy()->addMonths(3)->subDay();
  73. $this->setDateRange($quarterStart, $quarterEnd, $set);
  74. }
  75. public function processCalendarYear($year, Set $set): void
  76. {
  77. $start = Carbon::createFromDate($year)->startOfYear();
  78. $end = Carbon::createFromDate($year)->endOfYear();
  79. $this->setDateRange($start, $end, $set);
  80. }
  81. public function processCalendarQuarter($quarter, $year, Set $set): void
  82. {
  83. $month = ($quarter - 1) * 3 + 1;
  84. $start = Carbon::createFromDate($year, $month, 1);
  85. $end = Carbon::createFromDate($year, $month, 1)->endOfQuarter();
  86. $this->setDateRange($start, $end, $set);
  87. }
  88. public function processMonth($yearMonth, Set $set): void
  89. {
  90. $start = Carbon::parse($yearMonth)->startOfMonth();
  91. $end = Carbon::parse($yearMonth)->endOfMonth();
  92. $this->setDateRange($start, $end, $set);
  93. }
  94. public function setDateRange(Carbon $start, Carbon $end, Set $set): void
  95. {
  96. $set($this->startDateField, $start->startOfDay()->toDateString());
  97. $set($this->endDateField, $end->isFuture() ? now()->endOfDay()->toDateTimeString() : $end->endOfDay()->toDateTimeString());
  98. }
  99. }