| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * BudgetLimitController.php | 
					
						
							|  |  |  |  * Copyright (c) 2019 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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  | use FireflyIII\Models\AvailableBudget; | 
					
						
							|  |  |  | use FireflyIII\Models\Budget; | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | use FireflyIII\Models\BudgetLimit; | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  | use FireflyIII\Models\TransactionCurrency; | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Budget\OperationsRepositoryInterface; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  | use FireflyIII\Support\Http\Controllers\DateCalculation; | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | use Illuminate\Http\JsonResponse; | 
					
						
							|  |  |  | use Illuminate\Http\Request; | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							|  |  |  | use Log; | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Class BudgetLimitController | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class BudgetLimitController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  |     use DateCalculation; | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** @var BudgetLimitRepositoryInterface */ | 
					
						
							|  |  |  |     private $blRepository; | 
					
						
							|  |  |  |     /** @var CurrencyRepositoryInterface */ | 
					
						
							|  |  |  |     private $currencyRepos; | 
					
						
							|  |  |  |     /** @var OperationsRepositoryInterface */ | 
					
						
							|  |  |  |     private $opsRepository; | 
					
						
							|  |  |  |     /** @var BudgetRepositoryInterface The budget repository */ | 
					
						
							|  |  |  |     private $repository; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * AmountController constructor. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							|  |  |  |             function ($request, $next) { | 
					
						
							|  |  |  |                 app('view')->share('title', (string)trans('firefly.budgets')); | 
					
						
							|  |  |  |                 app('view')->share('mainTitleIcon', 'fa-tasks'); | 
					
						
							|  |  |  |                 $this->repository    = app(BudgetRepositoryInterface::class); | 
					
						
							|  |  |  |                 $this->opsRepository = app(OperationsRepositoryInterface::class); | 
					
						
							|  |  |  |                 $this->blRepository  = app(BudgetLimitRepositoryInterface::class); | 
					
						
							|  |  |  |                 $this->currencyRepos = app(CurrencyRepositoryInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Budget $budget | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function create(Budget $budget, Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $collection   = $this->currencyRepos->getEnabled(); | 
					
						
							|  |  |  |         $budgetLimits = $this->blRepository->getBudgetLimits($budget, $start, $end); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // remove already budgeted currencies:
 | 
					
						
							|  |  |  |         $currencies = $collection->filter( | 
					
						
							|  |  |  |             static function (TransactionCurrency $currency) use ($budgetLimits) { | 
					
						
							|  |  |  |                 /** @var AvailableBudget $budget */ | 
					
						
							|  |  |  |                 foreach ($budgetLimits as $budget) { | 
					
						
							|  |  |  |                     if ($budget->transaction_currency_id === $currency->id) { | 
					
						
							|  |  |  |                         return false; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return view('budgets.budget-limits.create', compact('start', 'end', 'currencies', 'budget')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Request     $request | 
					
						
							|  |  |  |      * @param BudgetLimit $budgetLimit | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function delete(Request $request, BudgetLimit $budgetLimit) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->blRepository->destroyBudgetLimit($budgetLimit); | 
					
						
							|  |  |  |         session()->flash('success', trans('firefly.deleted_bl')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return redirect(route('budgets.index')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Request $request | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  |      * @return JsonResponse|\Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector | 
					
						
							|  |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  |     public function store(Request $request) | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  |         // first search for existing one and update it if necessary.
 | 
					
						
							|  |  |  |         $currency = $this->currencyRepos->find((int)$request->get('transaction_currency_id')); | 
					
						
							|  |  |  |         $budget   = $this->repository->findNull((int)$request->get('budget_id')); | 
					
						
							|  |  |  |         if (null === $currency || null === $budget) { | 
					
						
							|  |  |  |             throw new FireflyException('No valid currency or budget.'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $start = Carbon::createFromFormat('Y-m-d', $request->get('start')); | 
					
						
							|  |  |  |         $end   = Carbon::createFromFormat('Y-m-d', $request->get('end')); | 
					
						
							|  |  |  |         $start->startOfDay(); | 
					
						
							|  |  |  |         $end->endOfDay(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Log::debug(sprintf('Start: %s, end: %s', $start->format('Y-m-d H:i:s'), $end->format('Y-m-d H:i:s'))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $limit = $this->blRepository->find($budget, $currency, $start, $end); | 
					
						
							|  |  |  |         if (null !== $limit) { | 
					
						
							|  |  |  |             $limit->amount = $request->get('amount'); | 
					
						
							|  |  |  |             $limit->save(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (null === $limit) { | 
					
						
							|  |  |  |             $limit = $this->blRepository->store( | 
					
						
							|  |  |  |                 [ | 
					
						
							|  |  |  |                     'budget_id'               => $request->get('budget_id'), | 
					
						
							|  |  |  |                     'transaction_currency_id' => $request->get('transaction_currency_id'), | 
					
						
							|  |  |  |                     'start_date'              => $request->get('start'), | 
					
						
							|  |  |  |                     'end_date'                => $request->get('end'), | 
					
						
							|  |  |  |                     'amount'                  => $request->get('amount'), | 
					
						
							|  |  |  |                 ] | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($request->expectsJson()) { | 
					
						
							|  |  |  |             $array = $limit->toArray(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // add some extra meta data:
 | 
					
						
							|  |  |  |             $spentArr                  = $this->opsRepository->sumExpenses($limit->start_date, $limit->end_date, null, new Collection([$budget]), $currency); | 
					
						
							|  |  |  |             $array['spent']            = $spentArr[$currency->id]['sum'] ?? '0'; | 
					
						
							|  |  |  |             $array['left_formatted']   = app('amount')->formatAnything($limit->transactionCurrency, bcadd($array['spent'], $array['amount'])); | 
					
						
							|  |  |  |             $array['amount_formatted'] = app('amount')->formatAnything($limit->transactionCurrency, $limit['amount']); | 
					
						
							|  |  |  |             $array['days_left']        = (string)$this->activeDaysLeft($start, $end); | 
					
						
							|  |  |  |             // left per day:
 | 
					
						
							|  |  |  |             $array['left_per_day'] = bcdiv(bcadd($array['spent'], $array['amount']), $array['days_left']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // left per day formatted.
 | 
					
						
							|  |  |  |             $array['left_per_day_formatted'] = app('amount')->formatAnything($limit->transactionCurrency, $array['left_per_day']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return response()->json($array); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-01 10:48:18 +02:00
										 |  |  |         return redirect(route('budgets.index')); | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Request     $request | 
					
						
							|  |  |  |      * @param BudgetLimit $budgetLimit | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return JsonResponse | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function update(Request $request, BudgetLimit $budgetLimit): JsonResponse | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $amount = $request->get('amount'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-01 11:13:03 +02:00
										 |  |  |         $limit = $this->blRepository->update($budgetLimit, ['amount' => $amount]); | 
					
						
							|  |  |  |         $array = $limit->toArray(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $spentArr                  = $this->opsRepository->sumExpenses( | 
					
						
							|  |  |  |             $limit->start_date, $limit->end_date, null, new Collection([$budgetLimit->budget]), $budgetLimit->transactionCurrency | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         $array['spent']            = $spentArr[$budgetLimit->transactionCurrency->id]['sum'] ?? '0'; | 
					
						
							|  |  |  |         $array['left_formatted']   = app('amount')->formatAnything($limit->transactionCurrency, bcadd($array['spent'], $array['amount'])); | 
					
						
							|  |  |  |         $array['amount_formatted'] = app('amount')->formatAnything($limit->transactionCurrency, $limit['amount']); | 
					
						
							|  |  |  |         $array['days_left']        = (string)$this->activeDaysLeft($limit->start_date, $limit->end_date); | 
					
						
							|  |  |  |         // left per day:
 | 
					
						
							|  |  |  |         $array['left_per_day'] = bcdiv(bcadd($array['spent'], $array['amount']), $array['days_left']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // left per day formatted.
 | 
					
						
							|  |  |  |         $array['left_per_day_formatted'] = app('amount')->formatAnything($limit->transactionCurrency, $array['left_per_day']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return response()->json($array); | 
					
						
							| 
									
										
										
										
											2019-08-31 21:47:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |