| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2016-05-20 12:41:23 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * BudgetController.php | 
					
						
							|  |  |  |  * Copyright (C) 2016 thegrumpydictator@gmail.com | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-10-05 06:52:15 +02:00
										 |  |  |  * This software may be modified and distributed under the terms of the | 
					
						
							|  |  |  |  * Creative Commons Attribution-ShareAlike 4.0 International License. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * See the LICENSE file for details. | 
					
						
							| 
									
										
										
										
											2016-05-20 12:41:23 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-05 12:08:25 +01:00
										 |  |  | declare(strict_types = 1); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Controllers\Chart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  | use FireflyIII\Generator\Chart\Basic\GeneratorInterface; | 
					
						
							| 
									
										
										
										
											2016-12-28 11:34:00 +01:00
										 |  |  | use FireflyIII\Helpers\Collector\JournalCollectorInterface; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							|  |  |  | use FireflyIII\Models\Budget; | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  | use FireflyIII\Models\BudgetLimit; | 
					
						
							| 
									
										
										
										
											2016-11-05 18:08:44 +01:00
										 |  |  | use FireflyIII\Models\Transaction; | 
					
						
							|  |  |  | use FireflyIII\Models\TransactionType; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  | use FireflyIII\Support\CacheProperties; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  | use Navigation; | 
					
						
							|  |  |  | use Preferences; | 
					
						
							|  |  |  | use Response; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class BudgetController | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |  * @SuppressWarnings(PHPMD.CouplingBetweenObjects) // can't realy be helped.
 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |  * @package FireflyIII\Http\Controllers\Chart | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class BudgetController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 10:44:06 +01:00
										 |  |  |     /** @var GeneratorInterface */ | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     protected $generator; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     /** @var  BudgetRepositoryInterface */ | 
					
						
							|  |  |  |     protected $repository; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2016-09-16 09:09:54 +02:00
										 |  |  |      * BudgetController constructor. | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							|  |  |  |             function ($request, $next) { | 
					
						
							|  |  |  |                 $this->generator  = app(GeneratorInterface::class); | 
					
						
							|  |  |  |                 $this->repository = app(BudgetRepositoryInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-25 09:57:39 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2016-05-05 21:25:20 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @param Budget $budget | 
					
						
							| 
									
										
										
										
											2016-04-25 09:57:39 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     public function budget(Budget $budget) | 
					
						
							| 
									
										
										
										
											2016-04-25 09:57:39 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |         $first = $this->repository->firstUseDate($budget); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $range = Preferences::get('viewRange', '1M')->data; | 
					
						
							|  |  |  |         $last  = session('end', new Carbon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($first); | 
					
						
							|  |  |  |         $cache->addProperty($last); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         $cache->addProperty('chart.budget.budget'); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $final = clone $last; | 
					
						
							|  |  |  |         $final->addYears(2); | 
					
						
							|  |  |  |         $budgetCollection = new Collection([$budget]); | 
					
						
							|  |  |  |         $last             = Navigation::endOfX($last, $range, $final); // not to overshoot.
 | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         $entries          = []; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         while ($first < $last) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // periodspecific dates:
 | 
					
						
							|  |  |  |             $currentStart = Navigation::startOfPeriod($first, $range); | 
					
						
							|  |  |  |             $currentEnd   = Navigation::endOfPeriod($first, $range); | 
					
						
							|  |  |  |             // sub another day because reasons.
 | 
					
						
							|  |  |  |             $currentEnd->subDay(); | 
					
						
							| 
									
										
										
										
											2017-01-05 09:08:35 +01:00
										 |  |  |             $spent            = $this->repository->spentInPeriod($budgetCollection, new Collection, $currentStart, $currentEnd); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |             $format           = Navigation::periodShow($first, $range); | 
					
						
							|  |  |  |             $entries[$format] = bcmul($spent, '-1'); | 
					
						
							|  |  |  |             $first            = Navigation::addPeriod($first, $range, 0); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 10:44:06 +01:00
										 |  |  |         $data = $this->generator->singleSet(strval(trans('firefly.spent')), $entries); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2016-04-25 09:57:39 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Shows the amount left in a specific budget limit. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |      * @SuppressWarnings(PHPMD.CyclomaticComplexity) // it's exactly five.
 | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @param Budget      $budget | 
					
						
							|  |  |  |      * @param BudgetLimit $budgetLimit | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     public function budgetLimit(Budget $budget, BudgetLimit $budgetLimit) | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |         if ($budgetLimit->budget->id != $budget->id) { | 
					
						
							|  |  |  |             throw new FireflyException('This budget limit is not part of this budget.'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $start = clone $budgetLimit->start_date; | 
					
						
							|  |  |  |         $end   = clone $budgetLimit->end_date; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         $cache->addProperty('chart.budget.budget.limit'); | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |         $cache->addProperty($budgetLimit->id); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         $entries          = []; | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |         $amount           = $budgetLimit->amount; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $budgetCollection = new Collection([$budget]); | 
					
						
							|  |  |  |         while ($start <= $end) { | 
					
						
							| 
									
										
										
										
											2017-01-05 09:08:35 +01:00
										 |  |  |             $spent            = $this->repository->spentInPeriod($budgetCollection, new Collection, $start, $start); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |             $amount           = bcadd($amount, $spent); | 
					
						
							|  |  |  |             $format           = $start->formatLocalized(strval(trans('config.month_and_day'))); | 
					
						
							|  |  |  |             $entries[$format] = $amount; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             $start->addDay(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-12-15 10:44:06 +01:00
										 |  |  |         $data = $this->generator->singleSet(strval(trans('firefly.left')), $entries); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Shows a budget list with spent/left/overspent. | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |      * @SuppressWarnings(PHPMD.CyclomaticComplexity) // it's exactly five.
 | 
					
						
							|  |  |  |      * @SuppressWarnings(PHPMD.ExcessiveMethodLength) // 46 lines, I'm fine with this.
 | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     public function frontpage() | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $start = session('start', Carbon::now()->startOfMonth()); | 
					
						
							|  |  |  |         $end   = session('end', Carbon::now()->endOfMonth()); | 
					
						
							|  |  |  |         // chart properties for cache:
 | 
					
						
							|  |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         $cache->addProperty('chart.budget.frontpage'); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |         $budgets   = $this->repository->getActiveBudgets(); | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |         $chartData = [ | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |             ['label' => strval(trans('firefly.spent_in_budget')), 'entries' => [], 'type' => 'bar',], | 
					
						
							|  |  |  |             ['label' => strval(trans('firefly.left_to_spend')), 'entries' => [], 'type' => 'bar',], | 
					
						
							|  |  |  |             ['label' => strval(trans('firefly.overspent')), 'entries' => [], 'type' => 'bar',], | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /** @var Budget $budget */ | 
					
						
							|  |  |  |         foreach ($budgets as $budget) { | 
					
						
							|  |  |  |             // get relevant repetitions:
 | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |             $limits   = $this->repository->getBudgetLimits($budget, $start, $end); | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |             $expenses = $this->getExpensesForBudget($limits, $budget, $start, $end); | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |             foreach ($expenses as $name => $row) { | 
					
						
							|  |  |  |                 $chartData[0]['entries'][$name] = $row['spent']; | 
					
						
							|  |  |  |                 $chartData[1]['entries'][$name] = $row['left']; | 
					
						
							|  |  |  |                 $chartData[2]['entries'][$name] = $row['overspent']; | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         // for no budget:
 | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |         $spent = $this->spentInPeriodWithout($start, $end); | 
					
						
							|  |  |  |         $name  = strval(trans('firefly.no_budget')); | 
					
						
							|  |  |  |         if (bccomp($spent, '0') !== 0) { | 
					
						
							|  |  |  |             $chartData[0]['entries'][$name] = bcmul($spent, '-1'); | 
					
						
							|  |  |  |             $chartData[1]['entries'][$name] = '0'; | 
					
						
							|  |  |  |             $chartData[2]['entries'][$name] = '0'; | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 10:44:06 +01:00
										 |  |  |         $data = $this->generator->multiSet($chartData); | 
					
						
							| 
									
										
										
										
											2016-05-06 10:32:26 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 08:41:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-24 20:23:17 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @SuppressWarnings(PHPMD.CyclomaticComplexity) // it's exactly five.
 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Budget     $budget | 
					
						
							|  |  |  |      * @param Carbon     $start | 
					
						
							|  |  |  |      * @param Carbon     $end | 
					
						
							|  |  |  |      * @param Collection $accounts | 
					
						
							| 
									
										
										
										
											2016-04-26 09:21:57 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Http\JsonResponse | 
					
						
							| 
									
										
										
										
											2016-04-24 20:23:17 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     public function period(Budget $budget, Collection $accounts, Carbon $start, Carbon $end) | 
					
						
							| 
									
										
										
										
											2016-04-24 20:23:17 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-05-06 22:53:08 +02:00
										 |  |  |         // chart properties for cache:
 | 
					
						
							|  |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty($accounts); | 
					
						
							|  |  |  |         $cache->addProperty($budget->id); | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         $cache->addProperty('chart.budget.period'); | 
					
						
							| 
									
										
										
										
											2016-05-06 22:53:08 +02:00
										 |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2016-05-06 22:53:08 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         $periods  = Navigation::listOfPeriods($start, $end); | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |         $entries  = $this->repository->getBudgetPeriodReport(new Collection([$budget]), $accounts, $start, $end); // get the expenses
 | 
					
						
							|  |  |  |         $budgeted = $this->getBudgetedInPeriod($budget, $start, $end); | 
					
						
							| 
									
										
										
										
											2016-05-06 22:53:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         // join them into one set of data:
 | 
					
						
							|  |  |  |         $chartData = [ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |             ['label' => strval(trans('firefly.spent')), 'type' => 'bar', 'entries' => [],], | 
					
						
							|  |  |  |             ['label' => strval(trans('firefly.budgeted')), 'type' => 'bar', 'entries' => [],], | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         ]; | 
					
						
							| 
									
										
										
										
											2016-11-19 13:37:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         foreach (array_keys($periods) as $period) { | 
					
						
							|  |  |  |             $label                           = $periods[$period]; | 
					
						
							|  |  |  |             $spent                           = isset($entries[$budget->id]['entries'][$period]) ? $entries[$budget->id]['entries'][$period] : '0'; | 
					
						
							| 
									
										
										
										
											2016-12-23 15:52:05 +01:00
										 |  |  |             $limit                           = isset($budgeted[$period]) ? $budgeted[$period] : 0; | 
					
						
							| 
									
										
										
										
											2016-12-30 13:45:02 +01:00
										 |  |  |             $chartData[0]['entries'][$label] = round(bcmul($spent, '-1'), 12); | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |             $chartData[1]['entries'][$label] = $limit; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-12-15 10:44:06 +01:00
										 |  |  |         $data = $this->generator->multiSet($chartData); | 
					
						
							| 
									
										
										
										
											2016-05-06 22:53:08 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @param Collection $accounts | 
					
						
							|  |  |  |      * @param Carbon     $start | 
					
						
							|  |  |  |      * @param Carbon     $end | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Http\JsonResponse | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     public function periodNoBudget(Collection $accounts, Carbon $start, Carbon $end) | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // chart properties for cache:
 | 
					
						
							|  |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty($accounts); | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         $cache->addProperty('chart.budget.no-budget'); | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // the expenses:
 | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         $periods   = Navigation::listOfPeriods($start, $end); | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |         $entries   = $this->repository->getNoBudgetPeriodReport($accounts, $start, $end); | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |         $chartData = []; | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // join them:
 | 
					
						
							|  |  |  |         foreach (array_keys($periods) as $period) { | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:10 +01:00
										 |  |  |             $label             = $periods[$period]; | 
					
						
							|  |  |  |             $spent             = isset($entries['entries'][$period]) ? $entries['entries'][$period] : '0'; | 
					
						
							| 
									
										
										
										
											2016-12-15 10:41:56 +01:00
										 |  |  |             $chartData[$label] = bcmul($spent, '-1'); | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-12-15 10:44:06 +01:00
										 |  |  |         $data = $this->generator->singleSet(strval(trans('firefly.spent')), $chartData); | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |         $cache->store($data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Budget $budget | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function getBudgetedInPeriod(Budget $budget, Carbon $start, Carbon $end): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $key      = Navigation::preferredCarbonFormat($start, $end); | 
					
						
							|  |  |  |         $range    = Navigation::preferredRangeFormat($start, $end); | 
					
						
							|  |  |  |         $current  = clone $start; | 
					
						
							|  |  |  |         $budgeted = []; | 
					
						
							|  |  |  |         while ($current < $end) { | 
					
						
							|  |  |  |             $currentStart     = Navigation::startOfPeriod($current, $range); | 
					
						
							|  |  |  |             $currentEnd       = Navigation::endOfPeriod($current, $range); | 
					
						
							|  |  |  |             $budgetLimits     = $this->repository->getBudgetLimits($budget, $currentStart, $currentEnd); | 
					
						
							|  |  |  |             $index            = $currentStart->format($key); | 
					
						
							|  |  |  |             $budgeted[$index] = $budgetLimits->sum('amount'); | 
					
						
							|  |  |  |             $currentEnd->addDay(); | 
					
						
							|  |  |  |             $current = clone $currentEnd; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $budgeted; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @SuppressWarnings(PHPMD.CyclomaticComplexity) // it's 6 but ok.
 | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |      * @param Collection $limits | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |      * @param Budget     $budget | 
					
						
							|  |  |  |      * @param Carbon     $start | 
					
						
							|  |  |  |      * @param Carbon     $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |     private function getExpensesForBudget(Collection $limits, Budget $budget, Carbon $start, Carbon $end): array | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |         $return = []; | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |         if ($limits->count() === 0) { | 
					
						
							| 
									
										
										
										
											2017-01-05 09:08:35 +01:00
										 |  |  |             $spent = $this->repository->spentInPeriod(new Collection([$budget]), new Collection, $start, $end); | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |             if (bccomp($spent, '0') !== 0) { | 
					
						
							| 
									
										
										
										
											2016-12-29 17:56:12 +01:00
										 |  |  |                 $return[$budget->name]['spent']     = bcmul($spent, '-1'); | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |                 $return[$budget->name]['left']      = 0; | 
					
						
							|  |  |  |                 $return[$budget->name]['overspent'] = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return $return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |         $rows = $this->spentInPeriodMulti($budget, $limits); | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |         foreach ($rows as $name => $row) { | 
					
						
							|  |  |  |             if (bccomp($row['spent'], '0') !== 0 || bccomp($row['left'], '0') !== 0) { | 
					
						
							|  |  |  |                 $return[$name]['spent']     = bcmul($row['spent'], '-1'); | 
					
						
							|  |  |  |                 $return[$name]['left']      = $row['left']; | 
					
						
							|  |  |  |                 $return[$name]['overspent'] = bcmul($row['overspent'], '-1'); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         unset($rows, $row); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @SuppressWarnings(PHPMD.CyclomaticComplexity) // it's exactly five.
 | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |      * Returns an array with the following values: | 
					
						
							|  |  |  |      * 0 => | 
					
						
							|  |  |  |      *   'name' => name of budget + repetition | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |      *   'left' => left in budget repetition (always zero) | 
					
						
							|  |  |  |      *   'overspent' => spent more than budget repetition? (always zero) | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |      *   'spent' => actually spent in period for budget | 
					
						
							|  |  |  |      * 1 => (etc) | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |      * @param Budget     $budget | 
					
						
							|  |  |  |      * @param Collection $limits | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |      * @return array | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-01-02 20:42:29 +01:00
										 |  |  |     private function spentInPeriodMulti(Budget $budget, Collection $limits): array | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         $return = []; | 
					
						
							|  |  |  |         $format = strval(trans('config.month_and_day')); | 
					
						
							|  |  |  |         $name   = $budget->name; | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |         /** @var BudgetLimit $budgetLimit */ | 
					
						
							|  |  |  |         foreach ($limits as $budgetLimit) { | 
					
						
							| 
									
										
										
										
											2017-01-05 09:08:35 +01:00
										 |  |  |             $expenses = $this->repository->spentInPeriod(new Collection([$budget]), new Collection, $budgetLimit->start_date, $budgetLimit->end_date); | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |             if ($limits->count() > 1) { | 
					
						
							| 
									
										
										
										
											2016-09-16 09:09:54 +02:00
										 |  |  |                 $name = $budget->name . ' ' . trans( | 
					
						
							|  |  |  |                         'firefly.between_dates', | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |                         [ | 
					
						
							|  |  |  |                             'start' => $budgetLimit->start_date->formatLocalized($format), | 
					
						
							|  |  |  |                             'end'   => $budgetLimit->end_date->formatLocalized($format), | 
					
						
							|  |  |  |                         ] | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |                     ); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-02-23 17:43:29 +01:00
										 |  |  |             /* | 
					
						
							|  |  |  |              * amount: amount of budget limit | 
					
						
							|  |  |  |              * left: amount of budget limit min spent, or 0 when < 0. | 
					
						
							|  |  |  |              * spent: spent, or amount of budget limit when > amount | 
					
						
							|  |  |  |              */ | 
					
						
							| 
									
										
										
										
											2016-12-30 08:41:48 +01:00
										 |  |  |             $amount        = $budgetLimit->amount; | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |             $left          = bccomp(bcadd($amount, $expenses), '0') < 1 ? '0' : bcadd($amount, $expenses); | 
					
						
							| 
									
										
										
										
											2017-02-23 17:43:29 +01:00
										 |  |  |             $spent         = bccomp($expenses, $amount) === 1 ? $expenses : bcmul($amount, '-1'); | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |             $overspent     = bccomp(bcadd($amount, $expenses), '0') < 1 ? bcadd($amount, $expenses) : '0'; | 
					
						
							|  |  |  |             $return[$name] = [ | 
					
						
							|  |  |  |                 'left'      => $left, | 
					
						
							|  |  |  |                 'overspent' => $overspent, | 
					
						
							|  |  |  |                 'spent'     => $spent, | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |             ]; | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |         return $return; | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  |      * Returns an array with the following values: | 
					
						
							|  |  |  |      * 'name' => "no budget" in local language | 
					
						
							|  |  |  |      * 'repetition_left' => left in budget repetition (always zero) | 
					
						
							|  |  |  |      * 'repetition_overspent' => spent more than budget repetition? (always zero) | 
					
						
							|  |  |  |      * 'spent' => actually spent in period for budget | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-11-05 18:08:44 +01:00
										 |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |      * @return string | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |     private function spentInPeriodWithout(Carbon $start, Carbon $end): string | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-11-05 18:08:44 +01:00
										 |  |  |         // collector
 | 
					
						
							| 
									
										
										
										
											2016-12-28 11:34:00 +01:00
										 |  |  |         /** @var JournalCollectorInterface $collector */ | 
					
						
							| 
									
										
										
										
											2017-02-05 16:16:15 +01:00
										 |  |  |         $collector = app(JournalCollectorInterface::class); | 
					
						
							| 
									
										
										
										
											2016-11-19 13:37:44 +01:00
										 |  |  |         $types     = [TransactionType::WITHDRAWAL]; | 
					
						
							| 
									
										
										
										
											2016-11-05 18:08:44 +01:00
										 |  |  |         $collector->setAllAssetAccounts()->setTypes($types)->setRange($start, $end)->withoutBudget(); | 
					
						
							|  |  |  |         $journals = $collector->getJournals(); | 
					
						
							|  |  |  |         $sum      = '0'; | 
					
						
							|  |  |  |         /** @var Transaction $entry */ | 
					
						
							|  |  |  |         foreach ($journals as $entry) { | 
					
						
							|  |  |  |             $sum = bcadd($entry->transaction_amount, $sum); | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-12-15 09:49:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-29 17:42:46 +01:00
										 |  |  |         return $sum; | 
					
						
							| 
									
										
										
										
											2016-07-30 16:29:04 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-20 19:56:14 +02:00
										 |  |  | } |