| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * RecurringRepository.php | 
					
						
							| 
									
										
										
										
											2020-02-16 14:00:57 +01:00
										 |  |  |  * Copyright (c) 2019 james@firefly-iii.org | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This file is part of Firefly III (https://github.com/firefly-iii). | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * 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. | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * GNU Affero General Public License for more details. | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * 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/>. | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | declare(strict_types=1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Repositories\Recurring; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  | use FireflyIII\Factory\RecurrenceFactory; | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  | use FireflyIII\Helpers\Collector\GroupCollectorInterface; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | use FireflyIII\Models\Note; | 
					
						
							|  |  |  | use FireflyIII\Models\Preference; | 
					
						
							|  |  |  | use FireflyIII\Models\Recurrence; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  | use FireflyIII\Models\RecurrenceMeta; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | use FireflyIII\Models\RecurrenceRepetition; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  | use FireflyIII\Models\RecurrenceTransaction; | 
					
						
							|  |  |  | use FireflyIII\Models\RecurrenceTransactionMeta; | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  | use FireflyIII\Models\TransactionJournal; | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  | use FireflyIII\Models\TransactionJournalMeta; | 
					
						
							| 
									
										
										
										
											2018-06-23 08:19:29 +02:00
										 |  |  | use FireflyIII\Services\Internal\Destroy\RecurrenceDestroyService; | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  | use FireflyIII\Services\Internal\Update\RecurrenceUpdateService; | 
					
						
							| 
									
										
										
										
											2019-01-27 11:00:28 +01:00
										 |  |  | use FireflyIII\Support\Repositories\Recurring\CalculateRangeOccurrences; | 
					
						
							|  |  |  | use FireflyIII\Support\Repositories\Recurring\CalculateXOccurrences; | 
					
						
							| 
									
										
										
										
											2019-09-26 19:16:17 +02:00
										 |  |  | use FireflyIII\Support\Repositories\Recurring\CalculateXOccurrencesSince; | 
					
						
							| 
									
										
										
										
											2019-01-27 11:00:28 +01:00
										 |  |  | use FireflyIII\Support\Repositories\Recurring\FiltersWeekends; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | use FireflyIII\User; | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  | use Illuminate\Pagination\LengthAwarePaginator; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  | use Log; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class RecurringRepository | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class RecurringRepository implements RecurringRepositoryInterface | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-26 19:16:17 +02:00
										 |  |  |     use CalculateRangeOccurrences, CalculateXOccurrences, CalculateXOccurrencesSince, FiltersWeekends; | 
					
						
							| 
									
										
										
										
											2021-03-05 07:03:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-23 19:11:25 +02:00
										 |  |  |     private User $user; | 
					
						
							| 
									
										
										
										
											2021-04-07 07:28:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-23 08:19:29 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Destroy a recurring transaction. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Recurrence $recurrence | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function destroy(Recurrence $recurrence): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var RecurrenceDestroyService $service */ | 
					
						
							|  |  |  |         $service = app(RecurrenceDestroyService::class); | 
					
						
							|  |  |  |         $service->destroy($recurrence); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @inheritDoc | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function destroyAll(): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->user->recurrences()->delete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns all of the user's recurring transactions. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     public function get(): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->user->recurrences() | 
					
						
							|  |  |  |                           ->with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions']) | 
					
						
							|  |  |  |                           ->orderBy('active', 'DESC') | 
					
						
							| 
									
										
										
										
											2018-07-01 21:32:48 +02:00
										 |  |  |                           ->orderBy('transaction_type_id', 'ASC') | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |                           ->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. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |      * @param RecurrenceTransaction $recTransaction | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return null|int | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |     public function getBudget(RecurrenceTransaction $recTransaction): ?int | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $return = 0; | 
					
						
							|  |  |  |         /** @var RecurrenceTransactionMeta $meta */ | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         foreach ($recTransaction->recurrenceTransactionMeta as $meta) { | 
					
						
							| 
									
										
										
										
											2018-07-22 18:50:27 +02:00
										 |  |  |             if ('budget_id' === $meta->name) { | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |                 $return = (int)$meta->value; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-22 18:50:27 +02:00
										 |  |  |         return 0 === $return ? null : $return; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the category from a recurring transaction transaction. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |      * @param RecurrenceTransaction $recTransaction | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return null|string | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-05-15 12:58:19 +02:00
										 |  |  |     public function getCategoryName(RecurrenceTransaction $recTransaction): ?string | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |         $return = ''; | 
					
						
							|  |  |  |         /** @var RecurrenceTransactionMeta $meta */ | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         foreach ($recTransaction->recurrenceTransactionMeta as $meta) { | 
					
						
							| 
									
										
										
										
											2018-07-22 18:50:27 +02:00
										 |  |  |             if ('category_name' === $meta->name) { | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |                 $return = (string)$meta->value; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-22 18:50:27 +02:00
										 |  |  |         return '' === $return ? null : $return; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 12:58:19 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the category from a recurring transaction transaction. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param RecurrenceTransaction $recTransaction | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return null|int | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getCategoryId(RecurrenceTransaction $recTransaction): ?int | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $return = ''; | 
					
						
							|  |  |  |         /** @var RecurrenceTransactionMeta $meta */ | 
					
						
							|  |  |  |         foreach ($recTransaction->recurrenceTransactionMeta as $meta) { | 
					
						
							|  |  |  |             if ('category_id' === $meta->name) { | 
					
						
							|  |  |  |                 $return = (int)$meta->value; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return '' === $return ? null : $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns the journals created for this recurrence, possibly limited by time. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * @param Recurrence  $recurrence | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |      * @param Carbon|null $start | 
					
						
							|  |  |  |      * @param Carbon|null $end | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-08-07 19:29:53 +02:00
										 |  |  |      * @return int | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-08-07 19:29:53 +02:00
										 |  |  |     public function getJournalCount(Recurrence $recurrence, Carbon $start = null, Carbon $end = null): int | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $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')); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 07:28:43 +02:00
										 |  |  |         return $query->count(['transaction_journals.id']); | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:20:54 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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 ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Generate events in the date range. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param RecurrenceRepetition $repetition | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * @param Carbon               $start | 
					
						
							|  |  |  |      * @param Carbon               $end | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							| 
									
										
										
										
											2019-08-17 10:47:29 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function getOccurrencesInRange(RecurrenceRepetition $repetition, Carbon $start, Carbon $end): array | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-07-23 21:49:15 +02:00
										 |  |  |         $occurrences = []; | 
					
						
							|  |  |  |         $mutator     = clone $start; | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |         $mutator->startOfDay(); | 
					
						
							| 
									
										
										
										
											2018-07-23 21:49:15 +02:00
										 |  |  |         $skipMod = $repetition->repetition_skip + 1; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |         Log::debug(sprintf('Calculating occurrences for rep type "%s"', $repetition->repetition_type)); | 
					
						
							| 
									
										
										
										
											2018-06-27 05:37:56 +02:00
										 |  |  |         Log::debug(sprintf('Mutator is now: %s', $mutator->format('Y-m-d'))); | 
					
						
							| 
									
										
										
										
											2018-07-23 21:49:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         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); | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-07 07:28:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 18:49:33 +02:00
										 |  |  |         // filter out all the weekend days:
 | 
					
						
							| 
									
										
										
										
											2020-10-23 19:11:25 +02:00
										 |  |  |         return $this->filterWeekends($repetition, $occurrences); | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |      * @param RecurrenceTransaction $transaction | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |      * @return int|null | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |     public function getPiggyBank(RecurrenceTransaction $transaction): ?int | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |         $meta = $transaction->recurrenceTransactionMeta; | 
					
						
							|  |  |  |         /** @var RecurrenceTransactionMeta $metaEntry */ | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |         foreach ($meta as $metaEntry) { | 
					
						
							|  |  |  |             if ('piggy_bank_id' === $metaEntry->name) { | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |                 return (int)$metaEntry->value; | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the tags from the recurring transaction. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |      * @param RecurrenceTransaction $transaction | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |     public function getTags(RecurrenceTransaction $transaction): array | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $tags = []; | 
					
						
							|  |  |  |         /** @var RecurrenceMeta $meta */ | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |         foreach ($transaction->recurrenceTransactionMeta as $meta) { | 
					
						
							| 
									
										
										
										
											2018-07-22 18:50:27 +02:00
										 |  |  |             if ('tags' === $meta->name && '' !== $meta->value) { | 
					
						
							| 
									
										
										
										
											2019-11-14 17:33:04 +01:00
										 |  |  |                 $tags = json_decode($meta->value, true, 512, JSON_THROW_ON_ERROR); | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $tags; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Recurrence $recurrence | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * @param int        $page | 
					
						
							|  |  |  |      * @param int        $pageSize | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return LengthAwarePaginator | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |     public function getTransactionPaginator(Recurrence $recurrence, int $page, int $pageSize): LengthAwarePaginator | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $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) { | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |             $search[] = (int)$journalId; | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         /** @var GroupCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(GroupCollectorInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |         $collector->setUser($recurrence->user); | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         $collector->withCategoryInformation()->withBudgetInformation()->setLimit($pageSize)->setPage($page) | 
					
						
							|  |  |  |                   ->withAccountInformation(); | 
					
						
							|  |  |  |         $collector->setJournalIds($search); | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         return $collector->getPaginatedGroups(); | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Recurrence $recurrence | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  |     public function getTransactions(Recurrence $recurrence): Collection | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $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      = []; | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         foreach ($journalMeta as $journalId) { | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |             $search[] = (int)$journalId; | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-07 07:28:43 +02:00
										 |  |  |         if (0 === count($search)) { | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |             return new Collection; | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         /** @var GroupCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(GroupCollectorInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         $collector->setUser($recurrence->user); | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         $collector->withCategoryInformation()->withBudgetInformation()->withAccountInformation(); | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         // filter on specific journals.
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         $collector->setJournalIds($search); | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  |         return $collector->getGroups(); | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Calculate the next X iterations starting on the date given in $date. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param RecurrenceRepetition $repetition | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * @param Carbon               $date | 
					
						
							|  |  |  |      * @param int                  $count | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |     public function getXOccurrences(RecurrenceRepetition $repetition, Carbon $date, int $count): array | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-07-23 21:49:15 +02:00
										 |  |  |         $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); | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-23 21:49:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |         // filter out all the weekend days:
 | 
					
						
							| 
									
										
										
										
											2020-10-23 19:11:25 +02:00
										 |  |  |         return $this->filterWeekends($repetition, $occurrences); | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $this->filterMaxDate($repeatUntil, $occurrences); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Parse the repetition in a string that is user readable. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param RecurrenceRepetition $repetition | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return string | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function repetitionDescription(RecurrenceRepetition $repetition): string | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-10-10 21:02:56 +02:00
										 |  |  |         Log::debug('Now in repetitionDescription()'); | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |         /** @var Preference $pref */ | 
					
						
							|  |  |  |         $pref     = app('preferences')->getForUser($this->user, 'language', config('firefly.default_language', 'en_US')); | 
					
						
							|  |  |  |         $language = $pref->data; | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         if ('daily' === $repetition->repetition_type) { | 
					
						
							|  |  |  |             return (string)trans('firefly.recurring_daily', [], $language); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ('weekly' === $repetition->repetition_type) { | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |             $dayOfWeek = trans(sprintf('config.dow_%s', $repetition->repetition_moment), [], $language); | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |             if ($repetition->repetition_skip > 0) { | 
					
						
							|  |  |  |                 return (string)trans('firefly.recurring_weekly_skip', ['weekday' => $dayOfWeek, 'skip' => $repetition->repetition_skip + 1], $language); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |             return (string)trans('firefly.recurring_weekly', ['weekday' => $dayOfWeek], $language); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ('monthly' === $repetition->repetition_type) { | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |             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 | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         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) { | 
					
						
							|  |  |  |             //
 | 
					
						
							| 
									
										
										
										
											2018-08-04 17:30:06 +02:00
										 |  |  |             $today       = Carbon::now()->endOfYear(); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |             $repDate     = Carbon::createFromFormat('Y-m-d', $repetition->repetition_moment); | 
					
						
							|  |  |  |             $diffInYears = $today->diffInYears($repDate); | 
					
						
							|  |  |  |             $repDate->addYears($diffInYears); // technically not necessary.
 | 
					
						
							| 
									
										
										
										
											2018-07-27 04:46:21 +02:00
										 |  |  |             $string = $repDate->formatLocalized((string)trans('config.month_and_day_no_year')); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return (string)trans('firefly.recurring_yearly', ['date' => $string], $language); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ''; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @inheritDoc | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function searchRecurrence(string $query, int $limit): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $search = $this->user->recurrences(); | 
					
						
							|  |  |  |         if ('' !== $query) { | 
					
						
							|  |  |  |             $search->where('recurrences.title', 'LIKE', sprintf('%%%s%%', $query)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $search | 
					
						
							|  |  |  |             ->orderBy('recurrences.title', 'ASC'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $search->take($limit)->get(['id', 'title', 'description']); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Set user for in repository. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param User $user | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function setUser(User $user): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->user = $user; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param array $data | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Recurrence | 
					
						
							| 
									
										
										
										
											2019-06-29 19:47:40 +02:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function store(array $data): Recurrence | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-06-29 19:47:40 +02:00
										 |  |  |         /** @var RecurrenceFactory $factory */ | 
					
						
							|  |  |  |         $factory = app(RecurrenceFactory::class); | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |         $factory->setUser($this->user); | 
					
						
							| 
									
										
										
										
											2019-06-29 19:47:40 +02:00
										 |  |  |         $result = $factory->create($data); | 
					
						
							|  |  |  |         if (null === $result) { | 
					
						
							|  |  |  |             throw new FireflyException($factory->getErrors()->first()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-29 19:47:40 +02:00
										 |  |  |         return $result; | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @inheritDoc | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function totalTransactions(Recurrence $recurrence, RecurrenceRepetition $repetition): int | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // if repeat = null just return 0.
 | 
					
						
							|  |  |  |         if (null === $recurrence->repeat_until && 0 === (int)$recurrence->repetitions) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // expect X transactions then stop. Return that number
 | 
					
						
							|  |  |  |         if (null === $recurrence->repeat_until && 0 !== (int)$recurrence->repetitions) { | 
					
						
							|  |  |  |             return (int)$recurrence->repetitions; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // need to calculate, this depends on the repetition:
 | 
					
						
							|  |  |  |         if (null !== $recurrence->repeat_until && 0 === (int)$recurrence->repetitions) { | 
					
						
							|  |  |  |             $occurrences = $this->getOccurrencesInRange($repetition, $recurrence->first_date ?? today(), $recurrence->repeat_until); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return count($occurrences); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Update a recurring transaction. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Recurrence $recurrence | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |      * @param array      $data | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return Recurrence | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function update(Recurrence $recurrence, array $data): Recurrence | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var RecurrenceUpdateService $service */ | 
					
						
							|  |  |  |         $service = app(RecurrenceUpdateService::class); | 
					
						
							| 
									
										
										
										
											2021-03-21 09:15:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  |         return $service->update($recurrence, $data); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-09-26 19:16:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-04 10:26:01 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @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; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-22 20:32:02 +02:00
										 |  |  | } |