| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2024-11-25 04:18:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * 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; | 
					
						
							| 
									
										
										
										
											2025-02-23 12:28:27 +01:00
										 |  |  | use FireflyIII\Support\Repositories\UserGroup\UserGroupInterface; | 
					
						
							|  |  |  | use FireflyIII\Support\Repositories\UserGroup\UserGroupTrait; | 
					
						
							| 
									
										
										
										
											2022-12-29 15:42:02 +01:00
										 |  |  | use Illuminate\Database\Eloquent\Builder; | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  | use Illuminate\Pagination\LengthAwarePaginator; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2024-01-05 10:55:46 +01:00
										 |  |  | use Illuminate\Support\Facades\Log; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class RecurringRepository | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2025-02-23 12:28:27 +01:00
										 |  |  | class RecurringRepository implements RecurringRepositoryInterface, UserGroupInterface | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |     use CalculateRangeOccurrences; | 
					
						
							|  |  |  |     use CalculateXOccurrences; | 
					
						
							|  |  |  |     use CalculateXOccurrencesSince; | 
					
						
							|  |  |  |     use FiltersWeekends; | 
					
						
							| 
									
										
										
										
											2021-03-05 07:03:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-23 12:28:27 +01:00
										 |  |  |     use UserGroupTrait; | 
					
						
							| 
									
										
										
										
											2021-04-07 07:28:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |     public function createdPreviously(Recurrence $recurrence, Carbon $date): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // if not, loop set and try to read the recurrence_date. If it matches start or end, return it as well.
 | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |         $set | 
					
						
							| 
									
										
										
										
											2023-12-21 05:07:26 +01:00
										 |  |  |             = TransactionJournalMeta::where(static function (Builder $q1) use ($recurrence): void { | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |                 $q1->where('name', 'recurrence_id'); | 
					
						
							|  |  |  |                 $q1->where('data', json_encode((string) $recurrence->id)); | 
					
						
							|  |  |  |             })->get(['journal_meta.transaction_journal_id']); | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // there are X journals made for this recurrence. Any of them meant for today?
 | 
					
						
							|  |  |  |         foreach ($set as $journalMeta) { | 
					
						
							| 
									
										
										
										
											2023-12-21 05:07:26 +01:00
										 |  |  |             $count = TransactionJournalMeta::where(static function (Builder $q2) use ($date): void { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 $string = (string) $date; | 
					
						
							| 
									
										
										
										
											2023-10-29 06:33:43 +01:00
										 |  |  |                 app('log')->debug(sprintf('Search for date: %s', json_encode($string))); | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |                 $q2->where('name', 'recurrence_date'); | 
					
						
							|  |  |  |                 $q2->where('data', json_encode($string)); | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |                 ->where('transaction_journal_id', $journalMeta->transaction_journal_id) | 
					
						
							|  |  |  |                 ->count() | 
					
						
							|  |  |  |             ; | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |             if ($count > 0) { | 
					
						
							| 
									
										
										
										
											2023-10-29 06:33:43 +01:00
										 |  |  |                 app('log')->debug(sprintf('Looks like journal #%d was already created', $journalMeta->transaction_journal_id)); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2025-02-01 15:39:34 +01:00
										 |  |  |      * Returns all the user's recurring transactions. | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function get(): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->user->recurrences() | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions']) | 
					
						
							|  |  |  |             ->orderBy('active', 'DESC') | 
					
						
							|  |  |  |             ->orderBy('transaction_type_id', 'ASC') | 
					
						
							|  |  |  |             ->orderBy('title', 'ASC') | 
					
						
							|  |  |  |             ->get() | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-23 08:19:29 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Destroy a recurring transaction. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function destroy(Recurrence $recurrence): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var RecurrenceDestroyService $service */ | 
					
						
							|  |  |  |         $service = app(RecurrenceDestroyService::class); | 
					
						
							|  |  |  |         $service->destroy($recurrence); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     public function destroyAll(): void | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-01-05 10:55:46 +01:00
										 |  |  |         Log::channel('audit')->info('Delete all recurring transactions through destroyAll'); | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |         $this->user->recurrences()->delete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get ALL recurring transactions. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getAll(): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // grab ALL recurring transactions:
 | 
					
						
							|  |  |  |         return Recurrence::with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions']) | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->orderBy('active', 'DESC') | 
					
						
							|  |  |  |             ->orderBy('title', 'ASC') | 
					
						
							|  |  |  |             ->get() | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-29 14:59:58 +02:00
										 |  |  |     public function getBillId(RecurrenceTransaction $recTransaction): ?int | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $return = null; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-29 14:59:58 +02:00
										 |  |  |         /** @var RecurrenceTransactionMeta $meta */ | 
					
						
							|  |  |  |         foreach ($recTransaction->recurrenceTransactionMeta as $meta) { | 
					
						
							|  |  |  |             if ('bill_id' === $meta->name) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 $return = (int) $meta->value; | 
					
						
							| 
									
										
										
										
											2022-03-29 14:59:58 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the budget ID from a recurring transaction transaction. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |     public function getBudget(RecurrenceTransaction $recTransaction): ?int | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $return = 0; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |         /** @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) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 $return = (int) $meta->value; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-09-18 10:26:12 +02:00
										 |  |  |     public function getCategoryId(RecurrenceTransaction $recTransaction): ?int | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |         $return = ''; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |         /** @var RecurrenceTransactionMeta $meta */ | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         foreach ($recTransaction->recurrenceTransactionMeta as $meta) { | 
					
						
							| 
									
										
										
										
											2021-09-18 10:26:12 +02:00
										 |  |  |             if ('category_id' === $meta->name) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 $return = (int) $meta->value; | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-09-18 10:26:12 +02:00
										 |  |  |     public function getCategoryName(RecurrenceTransaction $recTransaction): ?string | 
					
						
							| 
									
										
										
										
											2021-05-15 12:58:19 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $return = ''; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 12:58:19 +02:00
										 |  |  |         /** @var RecurrenceTransactionMeta $meta */ | 
					
						
							|  |  |  |         foreach ($recTransaction->recurrenceTransactionMeta as $meta) { | 
					
						
							| 
									
										
										
										
											2021-09-18 10:26:12 +02:00
										 |  |  |             if ('category_name' === $meta->name) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 $return = (string) $meta->value; | 
					
						
							| 
									
										
										
										
											2021-05-15 12:58:19 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return '' === $return ? null : $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns the journals created for this recurrence, possibly limited by time. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2024-03-18 20:25:30 +01:00
										 |  |  |     public function getJournalCount(Recurrence $recurrence, ?Carbon $start = null, ?Carbon $end = null): int | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         Log::debug(sprintf('Now in getJournalCount(#%d, "%s", "%s")', $recurrence->id, $start?->format('Y-m-d H:i:s'), $end?->format('Y-m-d H:i:s'))); | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |         $query = TransactionJournal::leftJoin('journal_meta', 'journal_meta.transaction_journal_id', '=', 'transaction_journals.id') | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->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.'"') | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2018-06-26 21:17:50 +02:00
										 |  |  |         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')); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         $count = $query->count('transaction_journals.id'); | 
					
						
							|  |  |  |         Log::debug(sprintf('Count is %d', $count)); | 
					
						
							| 
									
										
										
										
											2024-11-03 09:01:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         return $count; | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getJournalIds(Recurrence $recurrence): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return TransactionJournalMeta::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id') | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->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-12-07 20:20:54 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the notes. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getNoteText(Recurrence $recurrence): string | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  |         /** @var null|Note $note */ | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |         $note = $recurrence->notes()->first(); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         return (string) $note?->text; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +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; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |         /** @var RecurrenceTransactionMeta $metaEntry */ | 
					
						
							| 
									
										
										
										
											2019-08-22 17:09:08 +02:00
										 |  |  |         foreach ($meta as $metaEntry) { | 
					
						
							|  |  |  |             if ('piggy_bank_id' === $metaEntry->name) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +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. | 
					
						
							| 
									
										
										
										
											2023-12-22 20:12:38 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-10-31 07:45:55 +01:00
										 |  |  |     public function getTags(RecurrenceTransaction $transaction): array | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $tags = []; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 18:42:23 +02:00
										 |  |  |         /** @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-03 05:32:35 +02:00
										 |  |  |     public function getTransactionPaginator(Recurrence $recurrence, int $page, int $pageSize): LengthAwarePaginator | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |         $journalMeta = TransactionJournalMeta::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id') | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->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() | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |         $search      = []; | 
					
						
							|  |  |  |         foreach ($journalMeta as $journalId) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             $search[] = (int) $journalId; | 
					
						
							| 
									
										
										
										
											2018-07-02 20:39:45 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         /** @var GroupCollectorInterface $collector */ | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |         $collector   = app(GroupCollectorInterface::class); | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->withAccountInformation() | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         $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
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  |     public function getTransactions(Recurrence $recurrence): Collection | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |         $journalMeta = TransactionJournalMeta::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id') | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->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() | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         $search      = []; | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         foreach ($journalMeta as $journalId) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             $search[] = (int) $journalId; | 
					
						
							| 
									
										
										
										
											2018-07-03 05:32:35 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-11-04 05:11:05 +01:00
										 |  |  |         if (0 === count($search)) { | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |             return new Collection(); | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         /** @var GroupCollectorInterface $collector */ | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |         $collector   = app(GroupCollectorInterface::class); | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getXOccurrencesSince(RecurrenceRepetition $repetition, Carbon $date, Carbon $afterDate, int $count): array | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-10-29 06:33:43 +01:00
										 |  |  |         app('log')->debug('Now in getXOccurrencesSince()'); | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |         $skipMod     = $repetition->repetition_skip + 1; | 
					
						
							|  |  |  |         $occurrences = []; | 
					
						
							| 
									
										
										
										
											2024-03-03 13:45:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // to fix #8616, take a few days from both dates, then filter the list to make sure no entries
 | 
					
						
							|  |  |  |         // from today or before are saved.
 | 
					
						
							|  |  |  |         $date->subDays(4); | 
					
						
							|  |  |  |         $afterDate->subDays(4); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |         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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-22 20:11:09 +01:00
										 |  |  |     private function filterMaxDate(?Carbon $max, array $occurrences): array | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-07-06 15:50:20 +02:00
										 |  |  |         $filtered = []; | 
					
						
							| 
									
										
										
										
											2024-02-22 20:11:09 +01:00
										 |  |  |         if (null === $max) { | 
					
						
							| 
									
										
										
										
											2024-07-06 15:50:20 +02:00
										 |  |  |             foreach ($occurrences as $date) { | 
					
						
							|  |  |  |                 if ($date->gt(today())) { | 
					
						
							|  |  |  |                     $filtered[] = $date; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-07-08 05:07:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-06 15:50:20 +02:00
										 |  |  |             return $filtered; | 
					
						
							| 
									
										
										
										
											2024-02-22 20:11:09 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         foreach ($occurrences as $date) { | 
					
						
							| 
									
										
										
										
											2024-03-03 13:45:05 +01:00
										 |  |  |             if ($date->lte($max) && $date->gt(today())) { | 
					
						
							| 
									
										
										
										
											2024-02-22 20:11:09 +01:00
										 |  |  |                 $filtered[] = $date; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $filtered; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Parse the repetition in a string that is user readable. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2021-05-24 08:50:17 +02:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function repetitionDescription(RecurrenceRepetition $repetition): string | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-10-29 06:33:43 +01:00
										 |  |  |         app('log')->debug('Now in repetitionDescription()'); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2023-11-28 05:05:42 +01:00
										 |  |  |         if (is_array($language)) { | 
					
						
							|  |  |  |             $language = 'en_US'; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         $language = (string) $language; | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         if ('daily' === $repetition->repetition_type) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             return (string) trans('firefly.recurring_daily', [], $language); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         if ('weekly' === $repetition->repetition_type) { | 
					
						
							|  |  |  |             $dayOfWeek = trans(sprintf('config.dow_%s', $repetition->repetition_moment), [], $language); | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |             if ($repetition->repetition_skip > 0) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 return (string) trans('firefly.recurring_weekly_skip', ['weekday' => $dayOfWeek, 'skip' => $repetition->repetition_skip + 1], $language); | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             return (string) trans('firefly.recurring_weekly', ['weekday' => $dayOfWeek], $language); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         if ('monthly' === $repetition->repetition_type) { | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |             if ($repetition->repetition_skip > 0) { | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |                 return (string) trans( | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |                     'firefly.recurring_monthly_skip', | 
					
						
							|  |  |  |                     ['dayOfMonth' => $repetition->repetition_moment, 'skip' => $repetition->repetition_skip + 1], | 
					
						
							|  |  |  |                     $language | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |                 ); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             return (string) trans( | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |                 'firefly.recurring_monthly', | 
					
						
							|  |  |  |                 ['dayOfMonth' => $repetition->repetition_moment, 'skip' => $repetition->repetition_skip - 1], | 
					
						
							|  |  |  |                 $language | 
					
						
							| 
									
										
										
										
											2019-08-27 05:57:28 +02:00
										 |  |  |             ); | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         if ('ndom' === $repetition->repetition_type) { | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             $parts     = explode(',', $repetition->repetition_moment); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |             // first part is number of week, second is weekday.
 | 
					
						
							|  |  |  |             $dayOfWeek = trans(sprintf('config.dow_%s', $parts[1]), [], $language); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             return (string) trans('firefly.recurring_ndom', ['weekday' => $dayOfWeek, 'dayOfMonth' => $parts[0]], $language); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         if ('yearly' === $repetition->repetition_type) { | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             $today       = today(config('app.timezone'))->endOfYear(); | 
					
						
							|  |  |  |             $repDate     = Carbon::createFromFormat('Y-m-d', $repetition->repetition_moment); | 
					
						
							| 
									
										
										
										
											2024-04-02 15:40:33 +02:00
										 |  |  |             if (null === $repDate) { | 
					
						
							| 
									
										
										
										
											2023-11-28 05:05:42 +01:00
										 |  |  |                 $repDate = clone $today; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             $diffInYears = (int) $today->diffInYears($repDate, true); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |             $repDate->addYears($diffInYears); // technically not necessary.
 | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             $string      = $repDate->isoFormat((string) trans('config.month_and_day_no_year_js')); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |             return (string) trans('firefly.recurring_yearly', ['date' => $string], $language); | 
					
						
							| 
									
										
										
										
											2018-07-25 19:43:02 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ''; | 
					
						
							| 
									
										
										
										
											2018-06-10 16:59:03 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     public function searchRecurrence(string $query, int $limit): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $search = $this->user->recurrences(); | 
					
						
							|  |  |  |         if ('' !== $query) { | 
					
						
							| 
									
										
										
										
											2024-10-10 06:30:05 +02:00
										 |  |  |             $search->whereLike('recurrences.title', sprintf('%%%s%%', $query)); | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         $search | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |             ->orderBy('recurrences.title', 'ASC') | 
					
						
							|  |  |  |         ; | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $search->take($limit)->get(['id', 'title', 'description']); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2019-06-29 19:47:40 +02:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2023-12-22 20:12:38 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-04 07:18:03 +01:00
										 |  |  |         return $factory->create($data); | 
					
						
							| 
									
										
										
										
											2018-06-16 21:47:51 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-21 18:57:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |     public function totalTransactions(Recurrence $recurrence, RecurrenceRepetition $repetition): int | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // if repeat = null just return 0.
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         if (null === $recurrence->repeat_until && 0 === (int) $recurrence->repetitions) { | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // expect X transactions then stop. Return that number
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         if (null === $recurrence->repeat_until && 0 !== (int) $recurrence->repetitions) { | 
					
						
							|  |  |  |             return (int) $recurrence->repetitions; | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // need to calculate, this depends on the repetition:
 | 
					
						
							| 
									
										
										
										
											2024-11-03 08:16:46 +01:00
										 |  |  |         if (null !== $recurrence->repeat_until && 0 === (int) $recurrence->repetitions) { | 
					
						
							| 
									
										
										
										
											2021-03-12 06:20:01 +01:00
										 |  |  |             $occurrences = $this->getOccurrencesInRange($repetition, $recurrence->first_date ?? today(), $recurrence->repeat_until); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return count($occurrences); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-30 15:41:29 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |      * Generate events in the date range. | 
					
						
							| 
									
										
										
										
											2021-10-30 15:41:29 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |     public function getOccurrencesInRange(RecurrenceRepetition $repetition, Carbon $start, Carbon $end): array | 
					
						
							| 
									
										
										
										
											2021-10-30 15:41:29 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |         $occurrences = []; | 
					
						
							|  |  |  |         $mutator     = clone $start; | 
					
						
							|  |  |  |         $mutator->startOfDay(); | 
					
						
							| 
									
										
										
										
											2025-03-14 17:45:16 +01:00
										 |  |  |         $skipMod     = $repetition->repetition_skip + 1; | 
					
						
							| 
									
										
										
										
											2023-10-29 06:33:43 +01:00
										 |  |  |         app('log')->debug(sprintf('Calculating occurrences for rep type "%s"', $repetition->repetition_type)); | 
					
						
							|  |  |  |         app('log')->debug(sprintf('Mutator is now: %s', $mutator->format('Y-m-d'))); | 
					
						
							| 
									
										
										
										
											2021-10-30 15:41:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +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); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // filter out all the weekend days:
 | 
					
						
							|  |  |  |         return $this->filterWeekends($repetition, $occurrences); | 
					
						
							| 
									
										
										
										
											2021-10-30 15:41:29 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |      * Update a recurring transaction. | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |     public function update(Recurrence $recurrence, array $data): Recurrence | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |         /** @var RecurrenceUpdateService $service */ | 
					
						
							|  |  |  |         $service = app(RecurrenceUpdateService::class); | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-21 12:34:58 +02:00
										 |  |  |         return $service->update($recurrence, $data); | 
					
						
							| 
									
										
										
										
											2023-05-29 13:56:55 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-22 20:32:02 +02:00
										 |  |  | } |