| 
									
										
										
										
											2022-03-29 14:58:06 +02:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2024-11-25 04:18:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-29 14:58:06 +02:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |  * BudgetReportController.php | 
					
						
							| 
									
										
										
										
											2020-01-31 07:32:04 +01:00
										 |  |  |  * Copyright (c) 2020 james@firefly-iii.org | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This file is part of Firefly III (https://github.com/firefly-iii). | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01: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. | 
					
						
							| 
									
										
										
										
											2017-10-21 08:40:00 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2017-10-21 08:40:00 +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. | 
					
						
							| 
									
										
										
										
											2017-10-21 08:40:00 +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/>. | 
					
						
							| 
									
										
										
										
											2020-01-23 19:38:39 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-25 13:41:17 +01:00
										 |  |  | declare(strict_types=1); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Controllers\Chart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use FireflyIII\Generator\Chart\Basic\GeneratorInterface; | 
					
						
							|  |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							|  |  |  | use FireflyIII\Models\Budget; | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | use FireflyIII\Repositories\Budget\OperationsRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | use FireflyIII\Support\Http\Controllers\AugumentData; | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  | use FireflyIII\Support\Http\Controllers\TransactionCalculation; | 
					
						
							| 
									
										
										
										
											2018-07-08 12:28:42 +02:00
										 |  |  | use Illuminate\Http\JsonResponse; | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Separate controller because many helper functions are shared. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Class BudgetReportController | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class BudgetReportController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:19 +01:00
										 |  |  |     use AugumentData; | 
					
						
							|  |  |  |     use TransactionCalculation; | 
					
						
							| 
									
										
										
										
											2021-03-28 11:46:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 08:06:24 +02:00
										 |  |  |     /** @var GeneratorInterface Chart generation methods. */ | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     private $generator; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |     /** @var OperationsRepositoryInterface */ | 
					
						
							|  |  |  |     private $opsRepository; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2018-07-21 08:06:24 +02:00
										 |  |  |      * BudgetReportController constructor. | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							|  |  |  |             function ($request, $next) { | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                 $this->generator     = app(GeneratorInterface::class); | 
					
						
							|  |  |  |                 $this->opsRepository = app(OperationsRepositoryInterface::class); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |      * Chart that groups the expenses by budget. | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |     public function budgetExpense(Collection $accounts, Collection $budgets, Carbon $start, Carbon $end): JsonResponse | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |         $result = []; | 
					
						
							|  |  |  |         $spent  = $this->opsRepository->listExpenses($start, $end, $accounts, $budgets); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // loop expenses.
 | 
					
						
							|  |  |  |         foreach ($spent as $currency) { | 
					
						
							|  |  |  |             foreach ($currency['budgets'] as $budget) { | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |                 $title = sprintf('%s (%s)', $budget['name'], $currency['currency_name']); | 
					
						
							| 
									
										
										
										
											2023-12-10 06:45:59 +01:00
										 |  |  |                 $result[$title] ??= [ | 
					
						
							| 
									
										
										
										
											2022-12-29 19:41:57 +01:00
										 |  |  |                     'amount'          => '0', | 
					
						
							|  |  |  |                     'currency_symbol' => $currency['currency_symbol'], | 
					
						
							|  |  |  |                     'currency_code'   => $currency['currency_code'], | 
					
						
							|  |  |  |                 ]; | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                 foreach ($budget['transaction_journals'] as $journal) { | 
					
						
							|  |  |  |                     $amount                   = app('steam')->positive($journal['amount']); | 
					
						
							| 
									
										
										
										
											2025-05-04 13:47:00 +02:00
										 |  |  |                     $result[$title]['amount'] = bcadd($result[$title]['amount'], (string) $amount); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |         $data   = $this->generator->multiCurrencyPieChart($result); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 20:30:09 +01:00
										 |  |  |         return response()->json($data); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2018-07-21 08:06:24 +02:00
										 |  |  |      * Chart that groups the expenses by budget. | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |     public function categoryExpense(Collection $accounts, Collection $budgets, Carbon $start, Carbon $end): JsonResponse | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |         $result = []; | 
					
						
							|  |  |  |         $spent  = $this->opsRepository->listExpenses($start, $end, $accounts, $budgets); | 
					
						
							|  |  |  |         // loop expenses.
 | 
					
						
							|  |  |  |         foreach ($spent as $currency) { | 
					
						
							|  |  |  |             foreach ($currency['budgets'] as $budget) { | 
					
						
							|  |  |  |                 foreach ($budget['transaction_journals'] as $journal) { | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |                     $categoryName             = $journal['category_name'] ?? trans('firefly.no_category'); | 
					
						
							|  |  |  |                     $title                    = sprintf('%s (%s)', $categoryName, $currency['currency_name']); | 
					
						
							| 
									
										
										
										
											2023-12-10 06:45:59 +01:00
										 |  |  |                     $result[$title] ??= [ | 
					
						
							| 
									
										
										
										
											2022-12-29 19:41:57 +01:00
										 |  |  |                         'amount'          => '0', | 
					
						
							|  |  |  |                         'currency_symbol' => $currency['currency_symbol'], | 
					
						
							|  |  |  |                         'currency_code'   => $currency['currency_code'], | 
					
						
							|  |  |  |                     ]; | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     $amount                   = app('steam')->positive($journal['amount']); | 
					
						
							| 
									
										
										
										
											2025-05-04 13:47:00 +02:00
										 |  |  |                     $result[$title]['amount'] = bcadd($result[$title]['amount'], (string) $amount); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |         $data   = $this->generator->multiCurrencyPieChart($result); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 20:30:09 +01:00
										 |  |  |         return response()->json($data); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Chart that groups expenses by the account. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function destinationAccountExpense(Collection $accounts, Collection $budgets, Carbon $start, Carbon $end): JsonResponse | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $result = []; | 
					
						
							|  |  |  |         $spent  = $this->opsRepository->listExpenses($start, $end, $accounts, $budgets); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // loop expenses.
 | 
					
						
							|  |  |  |         foreach ($spent as $currency) { | 
					
						
							|  |  |  |             foreach ($currency['budgets'] as $budget) { | 
					
						
							|  |  |  |                 foreach ($budget['transaction_journals'] as $journal) { | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |                     $title                    = sprintf('%s (%s)', $journal['destination_account_name'], $currency['currency_name']); | 
					
						
							| 
									
										
										
										
											2023-12-10 06:45:59 +01:00
										 |  |  |                     $result[$title] ??= [ | 
					
						
							| 
									
										
										
										
											2022-12-29 19:41:57 +01:00
										 |  |  |                         'amount'          => '0', | 
					
						
							|  |  |  |                         'currency_symbol' => $currency['currency_symbol'], | 
					
						
							|  |  |  |                         'currency_code'   => $currency['currency_code'], | 
					
						
							|  |  |  |                     ]; | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     $amount                   = app('steam')->positive($journal['amount']); | 
					
						
							| 
									
										
										
										
											2025-05-04 13:47:00 +02:00
										 |  |  |                     $result[$title]['amount'] = bcadd($result[$title]['amount'], (string) $amount); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |         $data   = $this->generator->multiCurrencyPieChart($result); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return response()->json($data); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-16 21:38:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2018-07-21 08:06:24 +02:00
										 |  |  |      * Main overview of a budget in the budget report. | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function mainChart(Collection $accounts, Budget $budget, Carbon $start, Carbon $end): JsonResponse | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $chartData = []; | 
					
						
							|  |  |  |         $spent     = $this->opsRepository->listExpenses($start, $end, $accounts, new Collection([$budget])); | 
					
						
							|  |  |  |         $format    = app('navigation')->preferredCarbonLocalizedFormat($start, $end); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // loop expenses.
 | 
					
						
							|  |  |  |         foreach ($spent as $currency) { | 
					
						
							|  |  |  |             // add things to chart Data for each currency:
 | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |             $spentKey = sprintf('%d-spent', $currency['currency_id']); | 
					
						
							| 
									
										
										
										
											2023-12-10 06:45:59 +01:00
										 |  |  |             $chartData[$spentKey] ??= [ | 
					
						
							| 
									
										
										
										
											2022-12-29 19:41:57 +01:00
										 |  |  |                 'label'           => sprintf( | 
					
						
							|  |  |  |                     '%s (%s)', | 
					
						
							| 
									
										
										
										
											2024-12-22 08:43:12 +01:00
										 |  |  |                     (string) trans('firefly.spent_in_specific_budget', ['budget' => $budget->name]), | 
					
						
							| 
									
										
										
										
											2022-12-29 19:41:57 +01:00
										 |  |  |                     $currency['currency_name'] | 
					
						
							|  |  |  |                 ), | 
					
						
							|  |  |  |                 'type'            => 'bar', | 
					
						
							|  |  |  |                 'currency_symbol' => $currency['currency_symbol'], | 
					
						
							|  |  |  |                 'currency_code'   => $currency['currency_code'], | 
					
						
							|  |  |  |                 'currency_id'     => $currency['currency_id'], | 
					
						
							|  |  |  |                 'entries'         => $this->makeEntries($start, $end), | 
					
						
							|  |  |  |             ]; | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             foreach ($currency['budgets'] as $currentBudget) { | 
					
						
							|  |  |  |                 foreach ($currentBudget['transaction_journals'] as $journal) { | 
					
						
							| 
									
										
										
										
											2022-03-27 20:24:13 +02:00
										 |  |  |                     $key                                   = $journal['date']->isoFormat($format); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                     $amount                                = app('steam')->positive($journal['amount']); | 
					
						
							| 
									
										
										
										
											2023-12-10 06:45:59 +01:00
										 |  |  |                     $chartData[$spentKey]['entries'][$key] ??= '0'; | 
					
						
							| 
									
										
										
										
											2025-05-04 13:47:00 +02:00
										 |  |  |                     $chartData[$spentKey]['entries'][$key] = bcadd($chartData[$spentKey]['entries'][$key], (string) $amount); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |         $data      = $this->generator->multiSet($chartData); | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return response()->json($data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-22 20:11:09 +01:00
										 |  |  |     private function makeEntries(Carbon $start, Carbon $end): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $return         = []; | 
					
						
							|  |  |  |         $format         = app('navigation')->preferredCarbonLocalizedFormat($start, $end); | 
					
						
							|  |  |  |         $preferredRange = app('navigation')->preferredRangeFormat($start, $end); | 
					
						
							|  |  |  |         $currentStart   = clone $start; | 
					
						
							|  |  |  |         while ($currentStart <= $end) { | 
					
						
							|  |  |  |             $currentEnd   = app('navigation')->endOfPeriod($currentStart, $preferredRange); | 
					
						
							|  |  |  |             $key          = $currentStart->isoFormat($format); | 
					
						
							|  |  |  |             $return[$key] = '0'; | 
					
						
							|  |  |  |             $currentStart = clone $currentEnd; | 
					
						
							|  |  |  |             $currentStart->addDay()->startOfDay(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Chart that groups expenses by the account. | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |     public function sourceAccountExpense(Collection $accounts, Collection $budgets, Carbon $start, Carbon $end): JsonResponse | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  |         $result = []; | 
					
						
							|  |  |  |         $spent  = $this->opsRepository->listExpenses($start, $end, $accounts, $budgets); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // loop expenses.
 | 
					
						
							|  |  |  |         foreach ($spent as $currency) { | 
					
						
							|  |  |  |             foreach ($currency['budgets'] as $budget) { | 
					
						
							|  |  |  |                 foreach ($budget['transaction_journals'] as $journal) { | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |                     $title                    = sprintf('%s (%s)', $journal['source_account_name'], $currency['currency_name']); | 
					
						
							| 
									
										
										
										
											2023-12-10 06:45:59 +01:00
										 |  |  |                     $result[$title] ??= [ | 
					
						
							| 
									
										
										
										
											2022-12-29 19:41:57 +01:00
										 |  |  |                         'amount'          => '0', | 
					
						
							|  |  |  |                         'currency_symbol' => $currency['currency_symbol'], | 
					
						
							|  |  |  |                         'currency_code'   => $currency['currency_code'], | 
					
						
							|  |  |  |                     ]; | 
					
						
							| 
									
										
										
										
											2019-09-02 20:30:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     $amount                   = app('steam')->positive($journal['amount']); | 
					
						
							| 
									
										
										
										
											2025-05-04 13:47:00 +02:00
										 |  |  |                     $result[$title]['amount'] = bcadd($result[$title]['amount'], (string) $amount); | 
					
						
							| 
									
										
										
										
											2016-12-23 17:50:26 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 14:43:56 +01:00
										 |  |  |         $data   = $this->generator->multiCurrencyPieChart($result); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 20:30:09 +01:00
										 |  |  |         return response()->json($data); | 
					
						
							| 
									
										
										
										
											2016-12-16 08:07:31 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-22 19:42:45 +01:00
										 |  |  | } |