| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * ShowController.php | 
					
						
							|  |  |  |  * Copyright (c) 2018 thegrumpydictator@gmail.com | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of Firefly III. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Firefly III is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Firefly III is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with Firefly III. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | declare(strict_types=1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Controllers\Budget; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							|  |  |  | use FireflyIII\Helpers\Collector\JournalCollectorInterface; | 
					
						
							|  |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							|  |  |  | use FireflyIII\Models\Budget; | 
					
						
							|  |  |  | use FireflyIII\Models\BudgetLimit; | 
					
						
							|  |  |  | use FireflyIII\Models\TransactionType; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Journal\JournalRepositoryInterface; | 
					
						
							|  |  |  | use FireflyIII\Support\CacheProperties; | 
					
						
							|  |  |  | use Illuminate\Http\Request; | 
					
						
							|  |  |  | use Illuminate\Support\Collection; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Class ShowController | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class ShowController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** @var BudgetRepositoryInterface */ | 
					
						
							|  |  |  |     private $repository; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 16:08:34 +02:00
										 |  |  |         app('view')->share('hideBudgets', true); | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							|  |  |  |             function ($request, $next) { | 
					
						
							|  |  |  |                 app('view')->share('title', trans('firefly.budgets')); | 
					
						
							|  |  |  |                 app('view')->share('mainTitleIcon', 'fa-tasks'); | 
					
						
							|  |  |  |                 $this->repository = app(BudgetRepositoryInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |      * @param Request     $request | 
					
						
							|  |  |  |      * @param Carbon|null $start | 
					
						
							|  |  |  |      * @param Carbon|null $end | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |     public function noBudget(Request $request, Carbon $start = null, Carbon $end = null) | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |         /** @var Carbon $start */ | 
					
						
							|  |  |  |         $start = $start ?? session('start'); | 
					
						
							|  |  |  |         /** @var Carbon $end */ | 
					
						
							|  |  |  |         $end      = $end ?? session('end'); | 
					
						
							|  |  |  |         $subTitle = trans( | 
					
						
							|  |  |  |             'firefly.without_budget_between', | 
					
						
							|  |  |  |             ['start' => $start->formatLocalized($this->monthAndDayFormat), 'end' => $end->formatLocalized($this->monthAndDayFormat)] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         $periods  = $this->getPeriodOverview(); | 
					
						
							|  |  |  |         $page     = (int)$request->get('page'); | 
					
						
							|  |  |  |         $pageSize = (int)app('preferences')->get('listPageSize', 50)->data; | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |         /** @var JournalCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(JournalCollectorInterface::class); | 
					
						
							|  |  |  |         $collector->setAllAssetAccounts()->setRange($start, $end)->setTypes([TransactionType::WITHDRAWAL])->setLimit($pageSize)->setPage($page) | 
					
						
							|  |  |  |                   ->withoutBudget()->withOpposingAccount(); | 
					
						
							|  |  |  |         $transactions = $collector->getPaginatedJournals(); | 
					
						
							|  |  |  |         $transactions->setPath(route('budgets.no-budget')); | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |         return view('budgets.no-budget', compact('transactions', 'subTitle', 'periods', 'start', 'end')); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Request                    $request | 
					
						
							|  |  |  |      * @param JournalRepositoryInterface $repository | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function noBudgetAll(Request $request, JournalRepositoryInterface $repository) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $subTitle = trans('firefly.all_journals_without_budget'); | 
					
						
							|  |  |  |         $first    = $repository->firstNull(); | 
					
						
							|  |  |  |         $start    = null === $first ? new Carbon : $first->date; | 
					
						
							|  |  |  |         $end      = new Carbon; | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |         $page     = (int)$request->get('page'); | 
					
						
							| 
									
										
										
										
											2018-07-14 16:08:34 +02:00
										 |  |  |         $pageSize = (int)app('preferences')->get('listPageSize', 50)->data; | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |         $moment   = 'all'; | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /** @var JournalCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(JournalCollectorInterface::class); | 
					
						
							|  |  |  |         $collector->setAllAssetAccounts()->setRange($start, $end)->setTypes([TransactionType::WITHDRAWAL])->setLimit($pageSize)->setPage($page) | 
					
						
							|  |  |  |                   ->withoutBudget()->withOpposingAccount(); | 
					
						
							|  |  |  |         $transactions = $collector->getPaginatedJournals(); | 
					
						
							|  |  |  |         $transactions->setPath(route('budgets.no-budget')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return view('budgets.no-budget', compact('transactions', 'subTitle', 'moment', 'periods', 'start', 'end')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Request $request | 
					
						
							|  |  |  |      * @param Budget  $budget | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function show(Request $request, Budget $budget) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var Carbon $start */ | 
					
						
							|  |  |  |         $start      = session('first', Carbon::create()->startOfYear()); | 
					
						
							|  |  |  |         $end        = new Carbon; | 
					
						
							|  |  |  |         $page       = (int)$request->get('page'); | 
					
						
							| 
									
										
										
										
											2018-07-14 16:08:34 +02:00
										 |  |  |         $pageSize   = (int)app('preferences')->get('listPageSize', 50)->data; | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |         $limits     = $this->getLimits($budget, $start, $end); | 
					
						
							|  |  |  |         $repetition = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // collector:
 | 
					
						
							|  |  |  |         /** @var JournalCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(JournalCollectorInterface::class); | 
					
						
							|  |  |  |         $collector->setAllAssetAccounts()->setRange($start, $end)->setBudget($budget)->setLimit($pageSize)->setPage($page)->withBudgetInformation(); | 
					
						
							|  |  |  |         $transactions = $collector->getPaginatedJournals(); | 
					
						
							|  |  |  |         $transactions->setPath(route('budgets.show', [$budget->id])); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $subTitle = trans('firefly.all_journals_for_budget', ['name' => $budget->name]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return view('budgets.show', compact('limits', 'budget', 'repetition', 'transactions', 'subTitle')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Request     $request | 
					
						
							|  |  |  |      * @param Budget      $budget | 
					
						
							|  |  |  |      * @param BudgetLimit $budgetLimit | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View | 
					
						
							|  |  |  |      * @throws FireflyException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function showByBudgetLimit(Request $request, Budget $budget, BudgetLimit $budgetLimit) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($budgetLimit->budget->id !== $budget->id) { | 
					
						
							|  |  |  |             throw new FireflyException('This budget limit is not part of this budget.'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $page     = (int)$request->get('page'); | 
					
						
							| 
									
										
										
										
											2018-07-14 16:08:34 +02:00
										 |  |  |         $pageSize = (int)app('preferences')->get('listPageSize', 50)->data; | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |         $subTitle = trans( | 
					
						
							|  |  |  |             'firefly.budget_in_period', | 
					
						
							|  |  |  |             [ | 
					
						
							|  |  |  |                 'name'  => $budget->name, | 
					
						
							|  |  |  |                 'start' => $budgetLimit->start_date->formatLocalized($this->monthAndDayFormat), | 
					
						
							|  |  |  |                 'end'   => $budgetLimit->end_date->formatLocalized($this->monthAndDayFormat), | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // collector:
 | 
					
						
							|  |  |  |         /** @var JournalCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(JournalCollectorInterface::class); | 
					
						
							|  |  |  |         $collector->setAllAssetAccounts()->setRange($budgetLimit->start_date, $budgetLimit->end_date) | 
					
						
							|  |  |  |                   ->setBudget($budget)->setLimit($pageSize)->setPage($page)->withBudgetInformation(); | 
					
						
							|  |  |  |         $transactions = $collector->getPaginatedJournals(); | 
					
						
							|  |  |  |         $transactions->setPath(route('budgets.show', [$budget->id, $budgetLimit->id])); | 
					
						
							|  |  |  |         $start  = session('first', Carbon::create()->startOfYear()); | 
					
						
							|  |  |  |         $end    = new Carbon; | 
					
						
							|  |  |  |         $limits = $this->getLimits($budget, $start, $end); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return view('budgets.show', compact('limits', 'budget', 'budgetLimit', 'transactions', 'subTitle')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Budget $budget | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function getLimits(Budget $budget, Carbon $start, Carbon $end): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // properties for cache
 | 
					
						
							|  |  |  |         $cache = new CacheProperties; | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty($budget->id); | 
					
						
							|  |  |  |         $cache->addProperty('get-limits'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							|  |  |  |             return $cache->get(); // @codeCoverageIgnore
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $set    = $this->repository->getBudgetLimits($budget, $start, $end); | 
					
						
							|  |  |  |         $limits = new Collection(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @var BudgetLimit $entry */ | 
					
						
							|  |  |  |         foreach ($set as $entry) { | 
					
						
							|  |  |  |             $entry->spent = $this->repository->spentInPeriod(new Collection([$budget]), new Collection(), $entry->start_date, $entry->end_date); | 
					
						
							|  |  |  |             $limits->push($entry); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $cache->store($limits); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function getPeriodOverview(): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var JournalRepositoryInterface $repository */ | 
					
						
							|  |  |  |         $repository = app(JournalRepositoryInterface::class); | 
					
						
							|  |  |  |         $first      = $repository->firstNull(); | 
					
						
							|  |  |  |         $start      = null === $first ? new Carbon : $first->date; | 
					
						
							| 
									
										
										
										
											2018-07-14 16:08:34 +02:00
										 |  |  |         $range      = app('preferences')->get('viewRange', '1M')->data; | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |         $start      = app('navigation')->startOfPeriod($start, $range); | 
					
						
							|  |  |  |         $end        = app('navigation')->endOfX(new Carbon, $range, null); | 
					
						
							|  |  |  |         $entries    = new Collection; | 
					
						
							|  |  |  |         $cache      = new CacheProperties; | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty('no-budget-period-entries'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							|  |  |  |             return $cache->get(); // @codeCoverageIgnore
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $dates = app('navigation')->blockPeriods($start, $end, $range); | 
					
						
							|  |  |  |         foreach ($dates as $date) { | 
					
						
							|  |  |  |             /** @var JournalCollectorInterface $collector */ | 
					
						
							|  |  |  |             $collector = app(JournalCollectorInterface::class); | 
					
						
							|  |  |  |             $collector->setAllAssetAccounts()->setRange($date['start'], $date['end'])->withoutBudget()->withOpposingAccount()->setTypes( | 
					
						
							|  |  |  |                 [TransactionType::WITHDRAWAL] | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |             $set      = $collector->getJournals(); | 
					
						
							|  |  |  |             $sum      = (string)($set->sum('transaction_amount') ?? '0'); | 
					
						
							|  |  |  |             $journals = $set->count(); | 
					
						
							|  |  |  |             /** @noinspection PhpUndefinedMethodInspection */ | 
					
						
							|  |  |  |             $dateStr  = $date['end']->format('Y-m-d'); | 
					
						
							|  |  |  |             $dateName = app('navigation')->periodShow($date['end'], $date['period']); | 
					
						
							| 
									
										
										
										
											2018-07-14 17:23:44 +02:00
										 |  |  |             $entries->push( | 
					
						
							|  |  |  |                 ['string' => $dateStr, 'name' => $dateName, 'count' => $journals, 'sum' => $sum, 'date' => clone $date['end'], | 
					
						
							|  |  |  |                  'start'  => $date['start'], | 
					
						
							|  |  |  |                  'end'    => $date['end'], | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 ] | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2018-07-14 15:22:21 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         $cache->store($entries); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $entries; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |