mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-30 06:50:49 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			573 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			573 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * RecurringRepository.php
 | |
|  * Copyright (c) 2019 james@firefly-iii.org
 | |
|  *
 | |
|  * This file is part of Firefly III (https://github.com/firefly-iii).
 | |
|  *
 | |
|  * This program is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Affero General Public License as
 | |
|  * published by the Free Software Foundation, either version 3 of the
 | |
|  * License, or (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Affero General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Affero General Public License
 | |
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| declare(strict_types=1);
 | |
| 
 | |
| namespace FireflyIII\Repositories\Recurring;
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use FireflyIII\Exceptions\FireflyException;
 | |
| use FireflyIII\Factory\RecurrenceFactory;
 | |
| use FireflyIII\Helpers\Collector\GroupCollectorInterface;
 | |
| use FireflyIII\Models\Note;
 | |
| use FireflyIII\Models\Preference;
 | |
| use FireflyIII\Models\Recurrence;
 | |
| use FireflyIII\Models\RecurrenceMeta;
 | |
| use FireflyIII\Models\RecurrenceRepetition;
 | |
| use FireflyIII\Models\RecurrenceTransaction;
 | |
| use FireflyIII\Models\RecurrenceTransactionMeta;
 | |
| use FireflyIII\Models\TransactionJournal;
 | |
| use FireflyIII\Models\TransactionJournalMeta;
 | |
| use FireflyIII\Services\Internal\Destroy\RecurrenceDestroyService;
 | |
| use FireflyIII\Services\Internal\Update\RecurrenceUpdateService;
 | |
| use FireflyIII\Support\Repositories\Recurring\CalculateRangeOccurrences;
 | |
| use FireflyIII\Support\Repositories\Recurring\CalculateXOccurrences;
 | |
| use FireflyIII\Support\Repositories\Recurring\CalculateXOccurrencesSince;
 | |
| use FireflyIII\Support\Repositories\Recurring\FiltersWeekends;
 | |
| use FireflyIII\User;
 | |
| use Illuminate\Pagination\LengthAwarePaginator;
 | |
| use Illuminate\Support\Collection;
 | |
| use Log;
 | |
| 
 | |
| /**
 | |
|  * Class RecurringRepository
 | |
|  */
 | |
| class RecurringRepository implements RecurringRepositoryInterface
 | |
| {
 | |
|     use CalculateRangeOccurrences, CalculateXOccurrences, CalculateXOccurrencesSince, FiltersWeekends;
 | |
|     /** @var User */
 | |
|     private $user;
 | |
| 
 | |
|     /**
 | |
|      * Constructor.
 | |
|      */
 | |
|     public function __construct()
 | |
|     {
 | |
|         if ('testing' === config('app.env')) {
 | |
|             Log::warning(sprintf('%s should not be instantiated in the TEST environment!', get_class($this)));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Destroy a recurring transaction.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      */
 | |
|     public function destroy(Recurrence $recurrence): void
 | |
|     {
 | |
|         /** @var RecurrenceDestroyService $service */
 | |
|         $service = app(RecurrenceDestroyService::class);
 | |
|         $service->destroy($recurrence);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns all of the user's recurring transactions.
 | |
|      *
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function get(): Collection
 | |
|     {
 | |
|         return $this->user->recurrences()
 | |
|                           ->with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions'])
 | |
|                           ->orderBy('active', 'DESC')
 | |
|                           ->orderBy('transaction_type_id', 'ASC')
 | |
|                           ->orderBy('title', 'ASC')
 | |
|                           ->get();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get ALL recurring transactions.
 | |
|      *
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function getAll(): Collection
 | |
|     {
 | |
|         // grab ALL recurring transactions:
 | |
|         return Recurrence
 | |
|             ::with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions'])
 | |
|             ->orderBy('active', 'DESC')
 | |
|             ->orderBy('title', 'ASC')
 | |
|             ->get();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the budget ID from a recurring transaction transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $recTransaction
 | |
|      *
 | |
|      * @return null|int
 | |
|      */
 | |
|     public function getBudget(RecurrenceTransaction $recTransaction): ?int
 | |
|     {
 | |
|         $return = 0;
 | |
|         /** @var RecurrenceTransactionMeta $meta */
 | |
|         foreach ($recTransaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('budget_id' === $meta->name) {
 | |
|                 $return = (int)$meta->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return 0 === $return ? null : $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the category from a recurring transaction transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $recTransaction
 | |
|      *
 | |
|      * @return null|string
 | |
|      */
 | |
|     public function getCategory(RecurrenceTransaction $recTransaction): ?string
 | |
|     {
 | |
|         $return = '';
 | |
|         /** @var RecurrenceTransactionMeta $meta */
 | |
|         foreach ($recTransaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('category_name' === $meta->name) {
 | |
|                 $return = (string)$meta->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return '' === $return ? null : $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the journals created for this recurrence, possibly limited by time.
 | |
|      *
 | |
|      * @param Recurrence  $recurrence
 | |
|      * @param Carbon|null $start
 | |
|      * @param Carbon|null $end
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     public function getJournalCount(Recurrence $recurrence, Carbon $start = null, Carbon $end = null): int
 | |
|     {
 | |
|         $query = TransactionJournal
 | |
|             ::leftJoin('journal_meta', 'journal_meta.transaction_journal_id', '=', 'transaction_journals.id')
 | |
|             ->where('transaction_journals.user_id', $recurrence->user_id)
 | |
|             ->whereNull('transaction_journals.deleted_at')
 | |
|             ->where('journal_meta.name', 'recurrence_id')
 | |
|             ->where('journal_meta.data', '"' . $recurrence->id . '"');
 | |
|         if (null !== $start) {
 | |
|             $query->where('transaction_journals.date', '>=', $start->format('Y-m-d 00:00:00'));
 | |
|         }
 | |
|         if (null !== $end) {
 | |
|             $query->where('transaction_journals.date', '<=', $end->format('Y-m-d 00:00:00'));
 | |
|         }
 | |
| 
 | |
|         return $query->get(['transaction_journals.*'])->count();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get journal ID's for journals created by this recurring transaction.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function getJournalIds(Recurrence $recurrence): array
 | |
|     {
 | |
|         return TransactionJournalMeta::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
 | |
|                                      ->where('transaction_journals.user_id', $this->user->id)
 | |
|                                      ->where('journal_meta.name', '=', 'recurrence_id')
 | |
|                                      ->where('journal_meta.data', '=', json_encode((string)$recurrence->id))
 | |
|                                      ->get(['journal_meta.transaction_journal_id'])->pluck('transaction_journal_id')->toArray();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the notes.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getNoteText(Recurrence $recurrence): string
 | |
|     {
 | |
|         /** @var Note $note */
 | |
|         $note = $recurrence->notes()->first();
 | |
|         if (null !== $note) {
 | |
|             return (string)$note->text;
 | |
|         }
 | |
| 
 | |
|         return '';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Generate events in the date range.
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      * @param Carbon               $start
 | |
|      * @param Carbon               $end
 | |
|      *
 | |
|      * @return array
 | |
|      *
 | |
|      */
 | |
|     public function getOccurrencesInRange(RecurrenceRepetition $repetition, Carbon $start, Carbon $end): array
 | |
|     {
 | |
|         $occurrences = [];
 | |
|         $mutator     = clone $start;
 | |
|         $mutator->startOfDay();
 | |
|         $skipMod = $repetition->repetition_skip + 1;
 | |
|         Log::debug(sprintf('Calculating occurrences for rep type "%s"', $repetition->repetition_type));
 | |
|         Log::debug(sprintf('Mutator is now: %s', $mutator->format('Y-m-d')));
 | |
| 
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getDailyInRange($mutator, $end, $skipMod);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getWeeklyInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getMonthlyInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getNdomInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getYearlyInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
| 
 | |
| 
 | |
|         // filter out all the weekend days:
 | |
|         $occurrences = $this->filterWeekends($repetition, $occurrences);
 | |
| 
 | |
|         return $occurrences;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param RecurrenceTransaction $transaction
 | |
|      *
 | |
|      * @return int|null
 | |
|      */
 | |
|     public function getPiggyBank(RecurrenceTransaction $transaction): ?int
 | |
|     {
 | |
|         $meta = $transaction->recurrenceTransactionMeta;
 | |
|         /** @var RecurrenceTransactionMeta $metaEntry */
 | |
|         foreach ($meta as $metaEntry) {
 | |
|             if ('piggy_bank_id' === $metaEntry->name) {
 | |
|                 return (int)$metaEntry->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the tags from the recurring transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $transaction
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function getTags(RecurrenceTransaction $transaction): array
 | |
|     {
 | |
|         $tags = [];
 | |
|         /** @var RecurrenceMeta $meta */
 | |
|         foreach ($transaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('tags' === $meta->name && '' !== $meta->value) {
 | |
|                 $tags = json_decode($meta->value, true, 512, JSON_THROW_ON_ERROR);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $tags;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Recurrence $recurrence
 | |
|      * @param int        $page
 | |
|      * @param int        $pageSize
 | |
|      *
 | |
|      * @return LengthAwarePaginator
 | |
|      */
 | |
|     public function getTransactionPaginator(Recurrence $recurrence, int $page, int $pageSize): LengthAwarePaginator
 | |
|     {
 | |
|         $journalMeta = TransactionJournalMeta
 | |
|             ::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
 | |
|             ->whereNull('transaction_journals.deleted_at')
 | |
|             ->where('transaction_journals.user_id', $this->user->id)
 | |
|             ->where('name', 'recurrence_id')
 | |
|             ->where('data', json_encode((string)$recurrence->id))
 | |
|             ->get()->pluck('transaction_journal_id')->toArray();
 | |
|         $search      = [];
 | |
|         foreach ($journalMeta as $journalId) {
 | |
|             $search[] = (int)$journalId;
 | |
|         }
 | |
|         /** @var GroupCollectorInterface $collector */
 | |
|         $collector = app(GroupCollectorInterface::class);
 | |
| 
 | |
|         $collector->setUser($recurrence->user);
 | |
|         $collector->withCategoryInformation()->withBudgetInformation()->setLimit($pageSize)->setPage($page)
 | |
|                   ->withAccountInformation();
 | |
|         $collector->setJournalIds($search);
 | |
| 
 | |
|         return $collector->getPaginatedGroups();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Recurrence $recurrence
 | |
|      *
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function getTransactions(Recurrence $recurrence): Collection
 | |
|     {
 | |
|         $journalMeta = TransactionJournalMeta
 | |
|             ::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
 | |
|             ->whereNull('transaction_journals.deleted_at')
 | |
|             ->where('transaction_journals.user_id', $this->user->id)
 | |
|             ->where('name', 'recurrence_id')
 | |
|             ->where('data', json_encode((string)$recurrence->id))
 | |
|             ->get()->pluck('transaction_journal_id')->toArray();
 | |
|         $search      = [];
 | |
| 
 | |
|         foreach ($journalMeta as $journalId) {
 | |
|             $search[] = (int)$journalId;
 | |
|         }
 | |
|         if (0 === count($search)) {
 | |
| 
 | |
|             return new Collection;
 | |
|         }
 | |
| 
 | |
|         /** @var GroupCollectorInterface $collector */
 | |
|         $collector = app(GroupCollectorInterface::class);
 | |
| 
 | |
|         $collector->setUser($recurrence->user);
 | |
|         $collector->withCategoryInformation()->withBudgetInformation()->withAccountInformation();
 | |
|         // filter on specific journals.
 | |
|         $collector->setJournalIds($search);
 | |
| 
 | |
|         return $collector->getGroups();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Calculate the next X iterations starting on the date given in $date.
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      * @param Carbon               $date
 | |
|      * @param int                  $count
 | |
|      *
 | |
|      * @return array
 | |
|      *
 | |
|      */
 | |
|     public function getXOccurrences(RecurrenceRepetition $repetition, Carbon $date, int $count): array
 | |
|     {
 | |
|         $skipMod     = $repetition->repetition_skip + 1;
 | |
|         $occurrences = [];
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXDailyOccurrences($date, $count, $skipMod);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXWeeklyOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXMonthlyOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXNDomOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXYearlyOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
| 
 | |
|         // filter out all the weekend days:
 | |
|         $occurrences = $this->filterWeekends($repetition, $occurrences);
 | |
| 
 | |
|         return $occurrences;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Parse the repetition in a string that is user readable.
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      *
 | |
|      * @return string
 | |
|      *
 | |
|      */
 | |
|     public function repetitionDescription(RecurrenceRepetition $repetition): string
 | |
|     {
 | |
|         Log::debug('Now in repetitionDescription()');
 | |
|         /** @var Preference $pref */
 | |
|         $pref     = app('preferences')->getForUser($this->user, 'language', config('firefly.default_language', 'en_US'));
 | |
|         $language = $pref->data;
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             return (string)trans('firefly.recurring_daily', [], $language);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
| 
 | |
|             $dayOfWeek = trans(sprintf('config.dow_%s', $repetition->repetition_moment), [], $language);
 | |
|             if ($repetition->repetition_skip > 0) {
 | |
|                 return (string)trans('firefly.recurring_weekly_skip', ['weekday' => $dayOfWeek, 'skip' => $repetition->repetition_skip + 1], $language);
 | |
|             }
 | |
| 
 | |
|             return (string)trans('firefly.recurring_weekly', ['weekday' => $dayOfWeek], $language);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             if ($repetition->repetition_skip > 0) {
 | |
|                 return (string)trans(
 | |
|                     'firefly.recurring_monthly_skip', ['dayOfMonth' => $repetition->repetition_moment, 'skip' => $repetition->repetition_skip + 1], $language
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             return (string)trans(
 | |
|                 'firefly.recurring_monthly', ['dayOfMonth' => $repetition->repetition_moment, 'skip' => $repetition->repetition_skip - 1], $language
 | |
|             );
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $parts = explode(',', $repetition->repetition_moment);
 | |
|             // first part is number of week, second is weekday.
 | |
|             $dayOfWeek = trans(sprintf('config.dow_%s', $parts[1]), [], $language);
 | |
| 
 | |
|             return (string)trans('firefly.recurring_ndom', ['weekday' => $dayOfWeek, 'dayOfMonth' => $parts[0]], $language);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             //
 | |
|             $today       = Carbon::now()->endOfYear();
 | |
|             $repDate     = Carbon::createFromFormat('Y-m-d', $repetition->repetition_moment);
 | |
|             $diffInYears = $today->diffInYears($repDate);
 | |
|             $repDate->addYears($diffInYears); // technically not necessary.
 | |
|             $string = $repDate->formatLocalized((string)trans('config.month_and_day_no_year'));
 | |
| 
 | |
|             return (string)trans('firefly.recurring_yearly', ['date' => $string], $language);
 | |
|         }
 | |
| 
 | |
|         return '';
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set user for in repository.
 | |
|      *
 | |
|      * @param User $user
 | |
|      */
 | |
|     public function setUser(User $user): void
 | |
|     {
 | |
|         $this->user = $user;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param array $data
 | |
|      *
 | |
|      * @return Recurrence
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function store(array $data): Recurrence
 | |
|     {
 | |
|         /** @var RecurrenceFactory $factory */
 | |
|         $factory = app(RecurrenceFactory::class);
 | |
|         $factory->setUser($this->user);
 | |
|         $result = $factory->create($data);
 | |
|         if (null === $result) {
 | |
|             throw new FireflyException($factory->getErrors()->first());
 | |
|         }
 | |
| 
 | |
|         return $result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Update a recurring transaction.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      * @param array      $data
 | |
|      *
 | |
|      * @return Recurrence
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function update(Recurrence $recurrence, array $data): Recurrence
 | |
|     {
 | |
|         /** @var RecurrenceUpdateService $service */
 | |
|         $service = app(RecurrenceUpdateService::class);
 | |
| 
 | |
|         return $service->update($recurrence, $data);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Calculate the next X iterations starting on the date given in $date.
 | |
|      * Returns an array of Carbon objects.
 | |
|      *
 | |
|      * Only returns them of they are after $afterDate
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      * @param Carbon               $date
 | |
|      * @param Carbon               $afterDate
 | |
|      * @param int                  $count
 | |
|      *
 | |
|      * @return array
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function getXOccurrencesSince(RecurrenceRepetition $repetition, Carbon $date, Carbon $afterDate, int $count): array
 | |
|     {
 | |
|         Log::debug('Now in getXOccurrencesSince()');
 | |
|         $skipMod     = $repetition->repetition_skip + 1;
 | |
|         $occurrences = [];
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXDailyOccurrencesSince($date, $afterDate, $count, $skipMod);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXWeeklyOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXMonthlyOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXNDomOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXYearlyOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
| 
 | |
|         // filter out all the weekend days:
 | |
|         $occurrences = $this->filterWeekends($repetition, $occurrences);
 | |
| 
 | |
|         // filter out everything if "repeat_until" is set.
 | |
|         $repeatUntil = $repetition->recurrence->repeat_until;
 | |
|         $occurrences = $this->filterMaxDate($repeatUntil, $occurrences);
 | |
| 
 | |
|         return $occurrences;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Carbon|null $max
 | |
|      * @param array       $occurrences
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     private function filterMaxDate(?Carbon $max, array $occurrences): array
 | |
|     {
 | |
|         if (null === $max) {
 | |
|             return $occurrences;
 | |
|         }
 | |
|         $filtered = [];
 | |
|         foreach ($occurrences as $date) {
 | |
|             if ($date->lte($max)) {
 | |
|                 $filtered[] = $date;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $filtered;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      */
 | |
|     public function destroyAll(): void
 | |
|     {
 | |
|         $this->user->recurrences()->delete();
 | |
|     }
 | |
| }
 |