| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Controllers\Chart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							|  |  |  | use FireflyIII\Models\Budget; | 
					
						
							|  |  |  | use FireflyIII\Models\LimitRepetition; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2015-06-03 18:22:47 +02:00
										 |  |  | use FireflyIII\Support\CacheProperties; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  | use Navigation; | 
					
						
							|  |  |  | use Preferences; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | use Response; | 
					
						
							|  |  |  | use Session; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class BudgetController | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package FireflyIII\Http\Controllers\Chart | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class BudgetController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** @var  \FireflyIII\Generator\Chart\Budget\BudgetChartGenerator */ | 
					
						
							|  |  |  |     protected $generator; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2015-06-28 10:03:34 +02:00
										 |  |  |      * @codeCoverageIgnore | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  |         // create chart generator:
 | 
					
						
							| 
									
										
										
										
											2015-07-07 19:09:45 +02:00
										 |  |  |         $this->generator = app('FireflyIII\Generator\Chart\Budget\BudgetChartGenerator'); | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param BudgetRepositoryInterface $repository | 
					
						
							|  |  |  |      * @param Budget                    $budget | 
					
						
							| 
									
										
										
										
											2015-05-26 08:17:58 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     public function budget(BudgetRepositoryInterface $repository, Budget $budget) | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |     { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         // dates and times
 | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |         $first = $repository->getFirstBudgetLimitDate($budget); | 
					
						
							| 
									
										
										
										
											2015-05-24 15:03:45 +02:00
										 |  |  |         $range = Preferences::get('viewRange', '1M')->data; | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |         $last  = Session::get('end', new Carbon); | 
					
						
							|  |  |  |         $final = clone $last; | 
					
						
							|  |  |  |         $final->addYears(2); | 
					
						
							|  |  |  |         $last = Navigation::endOfX($last, $range, $final); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         // chart properties for cache:
 | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($first); | 
					
						
							|  |  |  |         $cache->addProperty($last); | 
					
						
							|  |  |  |         $cache->addProperty('budget'); | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2015-06-27 22:22:27 +02:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $entries = new Collection; | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |         while ($first < $last) { | 
					
						
							|  |  |  |             $end = Navigation::addPeriod($first, $range, 0); | 
					
						
							| 
									
										
										
										
											2015-06-05 16:49:16 +02:00
										 |  |  |             $end->subDay(); | 
					
						
							|  |  |  |             $chartDate = clone $end; | 
					
						
							|  |  |  |             $chartDate->startOfMonth(); | 
					
						
							| 
									
										
										
										
											2015-08-02 07:04:43 +02:00
										 |  |  |             $spent = $repository->balanceInPeriod($budget, $first, $end); | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |             $entries->push([$chartDate, $spent]); | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |             $first = Navigation::addPeriod($first, $range, 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $data = $this->generator->budget($entries); | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2015-05-20 06:49:22 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Shows the amount left in a specific budget limit. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param BudgetRepositoryInterface $repository | 
					
						
							|  |  |  |      * @param Budget                    $budget | 
					
						
							|  |  |  |      * @param LimitRepetition           $repetition | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     public function budgetLimit(BudgetRepositoryInterface $repository, Budget $budget, LimitRepetition $repetition) | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $start = clone $repetition->startdate; | 
					
						
							|  |  |  |         $end   = $repetition->enddate; | 
					
						
							| 
									
										
										
										
											2015-07-06 18:04:13 +02:00
										 |  |  |         bcscale(2); | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         // chart properties for cache:
 | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty('budget'); | 
					
						
							|  |  |  |         $cache->addProperty('limit'); | 
					
						
							|  |  |  |         $cache->addProperty($budget->id); | 
					
						
							|  |  |  |         $cache->addProperty($repetition->id); | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2015-06-27 22:22:27 +02:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $entries = new Collection; | 
					
						
							|  |  |  |         $amount  = $repetition->amount; | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while ($start <= $end) { | 
					
						
							|  |  |  |             /* | 
					
						
							|  |  |  |              * Sum of expenses on this day: | 
					
						
							|  |  |  |              */ | 
					
						
							| 
									
										
										
										
											2015-07-06 18:04:13 +02:00
										 |  |  |             $sum    = $repository->expensesOnDayCorrected($budget, $start); | 
					
						
							|  |  |  |             $amount = bcadd($amount, $sum); | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |             $entries->push([clone $start, $amount]); | 
					
						
							| 
									
										
										
										
											2015-05-16 13:06:38 +02:00
										 |  |  |             $start->addDay(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $data = $this->generator->budgetLimit($entries); | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +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. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param BudgetRepositoryInterface $repository | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     public function frontpage(BudgetRepositoryInterface $repository) | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $budgets    = $repository->getBudgets(); | 
					
						
							|  |  |  |         $start      = Session::get('start', Carbon::now()->startOfMonth()); | 
					
						
							|  |  |  |         $end        = Session::get('end', Carbon::now()->endOfMonth()); | 
					
						
							|  |  |  |         $allEntries = new Collection; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         // chart properties for cache:
 | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty('budget'); | 
					
						
							|  |  |  |         $cache->addProperty('all'); | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2015-06-27 22:22:27 +02:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         bcscale(2); | 
					
						
							| 
									
										
										
										
											2015-06-02 17:14:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /** @var Budget $budget */ | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |         foreach ($budgets as $budget) { | 
					
						
							|  |  |  |             $repetitions = $repository->getBudgetLimitRepetitions($budget, $start, $end); | 
					
						
							|  |  |  |             if ($repetitions->count() == 0) { | 
					
						
							| 
									
										
										
										
											2015-08-02 07:04:43 +02:00
										 |  |  |                 $expenses = $repository->balanceInPeriod($budget, $start, $end, true); | 
					
						
							| 
									
										
										
										
											2015-06-27 17:05:39 +02:00
										 |  |  |                 $allEntries->push([$budget->name, 0, 0, $expenses, 0, 0]); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             /** @var LimitRepetition $repetition */ | 
					
						
							|  |  |  |             foreach ($repetitions as $repetition) { | 
					
						
							| 
									
										
										
										
											2015-08-02 07:04:43 +02:00
										 |  |  |                 $expenses = $repository->balanceInPeriod($budget, $repetition->startdate, $repetition->enddate, true); | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |                 // $left can be less than zero.
 | 
					
						
							|  |  |  |                 // $overspent can be more than zero ( = overspending)
 | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |                 $left      = max(bcsub($repetition->amount, $expenses), 0); // limited at zero.
 | 
					
						
							|  |  |  |                 $overspent = max(bcsub($expenses, $repetition->amount), 0); // limited at zero.
 | 
					
						
							| 
									
										
										
										
											2015-06-28 08:33:23 +02:00
										 |  |  |                 $name      = $budget->name; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |                 // $spent is maxed to the repetition amount:
 | 
					
						
							|  |  |  |                 $spent = $expenses > $repetition->amount ? $repetition->amount : $expenses; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 17:05:39 +02:00
										 |  |  |                 $allEntries->push([$name, $left, $spent, $overspent, $repetition->amount, $expenses]); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $noBudgetExpenses = $repository->getWithoutBudgetSum($start, $end) * -1; | 
					
						
							| 
									
										
										
										
											2015-06-27 17:05:39 +02:00
										 |  |  |         $allEntries->push([trans('firefly.noBudget'), 0, 0, $noBudgetExpenses, 0, 0]); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $data = $this->generator->frontpage($allEntries); | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Show a yearly overview for a budget. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param BudgetRepositoryInterface $repository | 
					
						
							|  |  |  |      * @param                           $year | 
					
						
							|  |  |  |      * @param bool                      $shared | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Symfony\Component\HttpFoundation\Response | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |     public function year(BudgetRepositoryInterface $repository, $year, $shared = false) | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-07-31 18:18:54 +02:00
										 |  |  |         $start      = new Carbon($year . '-01-01'); | 
					
						
							|  |  |  |         $end        = new Carbon($year . '-12-31'); | 
					
						
							|  |  |  |         $shared     = $shared == 'shared' ? true : false; | 
					
						
							|  |  |  |         $allBudgets = $repository->getBudgets(); | 
					
						
							|  |  |  |         $budgets    = new Collection; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         // chart properties for cache:
 | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache = new CacheProperties(); | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty('budget'); | 
					
						
							|  |  |  |         $cache->addProperty('year'); | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							| 
									
										
										
										
											2015-06-27 22:22:27 +02:00
										 |  |  |             return Response::json($cache->get()); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-31 18:18:54 +02:00
										 |  |  |         // filter empty budgets:
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($allBudgets as $budget) { | 
					
						
							| 
									
										
										
										
											2015-08-02 07:04:43 +02:00
										 |  |  |             $spent = $repository->balanceInPeriod($budget, $start, $end, $shared); | 
					
						
							| 
									
										
										
										
											2015-07-31 18:18:54 +02:00
										 |  |  |             if ($spent != 0) { | 
					
						
							|  |  |  |                 $budgets->push($budget); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |         $entries = new Collection; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while ($start < $end) { | 
					
						
							|  |  |  |             // month is the current end of the period:
 | 
					
						
							|  |  |  |             $month = clone $start; | 
					
						
							|  |  |  |             $month->endOfMonth(); | 
					
						
							|  |  |  |             $row = [clone $start]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // each budget, fill the row:
 | 
					
						
							|  |  |  |             foreach ($budgets as $budget) { | 
					
						
							| 
									
										
										
										
											2015-08-02 07:04:43 +02:00
										 |  |  |                 $spent = $repository->balanceInPeriod($budget, $start, $month, $shared); | 
					
						
							| 
									
										
										
										
											2015-07-31 18:18:54 +02:00
										 |  |  |                 $row[] = $spent * -1; | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-06-27 11:44:18 +02:00
										 |  |  |             $entries->push($row); | 
					
						
							| 
									
										
										
										
											2015-06-05 16:49:16 +02:00
										 |  |  |             $start->endOfMonth()->addDay(); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-31 18:18:54 +02:00
										 |  |  |         $data = $this->generator->year($allBudgets, $entries); | 
					
						
							| 
									
										
										
										
											2015-06-03 21:25:11 +02:00
										 |  |  |         $cache->store($data); | 
					
						
							| 
									
										
										
										
											2015-06-02 17:44:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return Response::json($data); | 
					
						
							| 
									
										
										
										
											2015-05-16 09:41:14 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-20 19:56:14 +02:00
										 |  |  | } |