| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * CategoryController.php | 
					
						
							|  |  |  |  * Copyright (c) 2019 thegrumpydictator@gmail.com | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This file is part of Firefly III (https://github.com/firefly-iii). | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as | 
					
						
							|  |  |  |  * published by the Free Software Foundation, either version 3 of the | 
					
						
							|  |  |  |  * License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * GNU Affero General Public License for more details. | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  | declare(strict_types=1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Api\V1\Controllers\Chart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							|  |  |  | use FireflyIII\Api\V1\Controllers\Controller; | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  | use FireflyIII\Api\V1\Requests\DateRequest; | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  | use FireflyIII\Repositories\Category\CategoryRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2019-08-27 10:52:07 +02:00
										 |  |  | use FireflyIII\Repositories\Category\NoCategoryRepositoryInterface; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Category\OperationsRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  | use FireflyIII\User; | 
					
						
							|  |  |  | use Illuminate\Http\JsonResponse; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class CategoryController | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class CategoryController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /** @var CategoryRepositoryInterface */ | 
					
						
							|  |  |  |     private $categoryRepository; | 
					
						
							| 
									
										
										
										
											2019-08-27 10:52:07 +02:00
										 |  |  |     /** @var NoCategoryRepositoryInterface */ | 
					
						
							|  |  |  |     private $noCatRepository; | 
					
						
							| 
									
										
										
										
											2019-08-27 12:11:46 +02:00
										 |  |  |     /** @var OperationsRepositoryInterface */ | 
					
						
							|  |  |  |     private $opsRepository; | 
					
						
							| 
									
										
										
										
											2019-08-27 10:52:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * AccountController constructor. | 
					
						
							| 
									
										
										
										
											2019-08-27 10:52:07 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  |      * @codeCoverageIgnore | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							|  |  |  |             function ($request, $next) { | 
					
						
							|  |  |  |                 /** @var User $user */ | 
					
						
							|  |  |  |                 $user                     = auth()->user(); | 
					
						
							|  |  |  |                 $this->categoryRepository = app(CategoryRepositoryInterface::class); | 
					
						
							| 
									
										
										
										
											2019-08-27 10:52:07 +02:00
										 |  |  |                 $this->opsRepository      = app(OperationsRepositoryInterface::class); | 
					
						
							| 
									
										
										
										
											2019-08-27 12:11:46 +02:00
										 |  |  |                 $this->noCatRepository    = app(NoCategoryRepositoryInterface::class); | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |                 $this->categoryRepository->setUser($user); | 
					
						
							| 
									
										
										
										
											2019-08-27 10:52:07 +02:00
										 |  |  |                 $this->opsRepository->setUser($user); | 
					
						
							|  |  |  |                 $this->noCatRepository->setUser($user); | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  |      * @param DateRequest $request | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return JsonResponse | 
					
						
							| 
									
										
										
										
											2019-06-13 18:07:49 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-08-12 17:10:58 +02:00
										 |  |  |      * TODO after 4.8,0, simplify | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  |     public function overview(DateRequest $request): JsonResponse | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // parameters for chart:
 | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  |         $dates = $request->getAll(); | 
					
						
							| 
									
										
										
										
											2019-05-30 12:39:06 +02:00
										 |  |  |         /** @var Carbon $start */ | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  |         $start = $dates['start']; | 
					
						
							| 
									
										
										
										
											2019-05-30 12:39:06 +02:00
										 |  |  |         /** @var Carbon $end */ | 
					
						
							| 
									
										
										
										
											2019-06-09 15:28:54 +02:00
										 |  |  |         $end = $dates['end']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-28 12:28:23 +02:00
										 |  |  |         $tempData      = []; | 
					
						
							|  |  |  |         $spentWith     = $this->opsRepository->listExpenses($start, $end); | 
					
						
							|  |  |  |         $earnedWith    = $this->opsRepository->listIncome($start, $end); | 
					
						
							|  |  |  |         $spentWithout  = $this->noCatRepository->listExpenses($start, $end); | 
					
						
							|  |  |  |         $earnedWithout = $this->noCatRepository->listIncome($start, $end); | 
					
						
							|  |  |  |         $categories    = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ([$spentWith, $earnedWith] as $set) { | 
					
						
							|  |  |  |             foreach ($set as $currency) { | 
					
						
							|  |  |  |                 foreach ($currency['categories'] as $category) { | 
					
						
							|  |  |  |                     $categories[] = $category['name']; | 
					
						
							|  |  |  |                     $inKey        = sprintf('%d-i', $currency['currency_id']); | 
					
						
							|  |  |  |                     $outKey       = sprintf('%d-e', $currency['currency_id']); | 
					
						
							|  |  |  |                     // make data arrays if not yet present.
 | 
					
						
							|  |  |  |                     $tempData[$inKey]  = $tempData[$inKey] ?? [ | 
					
						
							|  |  |  |                             'currency_id'             => $currency['currency_id'], | 
					
						
							|  |  |  |                             'label'                   => (string)trans('firefly.box_earned_in_currency', ['currency' => $currency['currency_name']]), | 
					
						
							|  |  |  |                             'currency_code'           => $currency['currency_code'], | 
					
						
							|  |  |  |                             'currency_symbol'         => $currency['currency_symbol'], | 
					
						
							|  |  |  |                             'currency_decimal_places' => $currency['currency_decimal_places'], | 
					
						
							|  |  |  |                             'type'                    => 'bar', // line, area or bar
 | 
					
						
							|  |  |  |                             'yAxisID'                 => 0, // 0, 1, 2
 | 
					
						
							|  |  |  |                             'entries'                 => [ | 
					
						
							|  |  |  |                                 // per category:
 | 
					
						
							|  |  |  |                                 // "category" => 5,
 | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                         ]; | 
					
						
							|  |  |  |                     $tempData[$outKey] = $tempData[$outKey] ?? [ | 
					
						
							|  |  |  |                             'currency_id'             => $currency['currency_id'], | 
					
						
							|  |  |  |                             'label'                   => (string)trans('firefly.box_spent_in_currency', ['currency' => $currency['currency_name']]), | 
					
						
							|  |  |  |                             'currency_code'           => $currency['currency_code'], | 
					
						
							|  |  |  |                             'currency_symbol'         => $currency['currency_symbol'], | 
					
						
							|  |  |  |                             'currency_decimal_places' => $currency['currency_decimal_places'], | 
					
						
							|  |  |  |                             'type'                    => 'bar', // line, area or bar
 | 
					
						
							|  |  |  |                             'yAxisID'                 => 0, // 0, 1, 2
 | 
					
						
							|  |  |  |                             'entries'                 => [ | 
					
						
							|  |  |  |                                 // per category:
 | 
					
						
							|  |  |  |                                 // "category" => 5,
 | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     foreach ($category['transaction_journals'] as $journal) { | 
					
						
							|  |  |  |                         // is it expense or income?
 | 
					
						
							|  |  |  |                         $letter                                  = -1 === bccomp($journal['amount'], '0') ? 'e' : 'i'; | 
					
						
							|  |  |  |                         $currentKey                              = sprintf('%d-%s', $currency['currency_id'], $letter); | 
					
						
							|  |  |  |                         $name                                    = $category['name']; | 
					
						
							|  |  |  |                         $tempData[$currentKey]['entries'][$name] = $tempData[$currentKey]['entries'][$name] ?? '0'; | 
					
						
							|  |  |  |                         $tempData[$currentKey]['entries'][$name] = bcadd($tempData[$currentKey]['entries'][$name], $journal['amount']); | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-28 12:28:23 +02:00
										 |  |  |         foreach ([$spentWithout, $earnedWithout] as $set) { | 
					
						
							|  |  |  |             foreach ($set as $currency) { | 
					
						
							|  |  |  |                 $inKey        = sprintf('%d-i', $currency['currency_id']); | 
					
						
							|  |  |  |                 $outKey       = sprintf('%d-e', $currency['currency_id']); | 
					
						
							|  |  |  |                 $categories[] = (string)trans('firefly.no_category'); | 
					
						
							|  |  |  |                 // make data arrays if not yet present.
 | 
					
						
							|  |  |  |                 $tempData[$inKey]  = $tempData[$inKey] ?? [ | 
					
						
							|  |  |  |                         'currency_id'             => $currency['currency_id'], | 
					
						
							|  |  |  |                         'label'                   => (string)trans('firefly.box_earned_in_currency', ['currency' => $currency['currency_name']]), | 
					
						
							|  |  |  |                         'currency_code'           => $currency['currency_code'], | 
					
						
							|  |  |  |                         'currency_symbol'         => $currency['currency_symbol'], | 
					
						
							|  |  |  |                         'currency_decimal_places' => $currency['currency_decimal_places'], | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |                         'type'                    => 'bar', // line, area or bar
 | 
					
						
							|  |  |  |                         'yAxisID'                 => 0, // 0, 1, 2
 | 
					
						
							| 
									
										
										
										
											2019-08-28 12:28:23 +02:00
										 |  |  |                         'entries'                 => [ | 
					
						
							|  |  |  |                             // per category:
 | 
					
						
							|  |  |  |                             // "category" => 5,
 | 
					
						
							|  |  |  |                         ], | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |                     ]; | 
					
						
							| 
									
										
										
										
											2019-08-28 12:28:23 +02:00
										 |  |  |                 $tempData[$outKey] = $tempData[$outKey] ?? [ | 
					
						
							|  |  |  |                         'currency_id'             => $currency['currency_id'], | 
					
						
							|  |  |  |                         'label'                   => (string)trans('firefly.box_spent_in_currency', ['currency' => $currency['currency_name']]), | 
					
						
							|  |  |  |                         'currency_code'           => $currency['currency_code'], | 
					
						
							|  |  |  |                         'currency_symbol'         => $currency['currency_symbol'], | 
					
						
							|  |  |  |                         'currency_decimal_places' => $currency['currency_decimal_places'], | 
					
						
							|  |  |  |                         'type'                    => 'bar', // line, area or bar
 | 
					
						
							|  |  |  |                         'yAxisID'                 => 0, // 0, 1, 2
 | 
					
						
							|  |  |  |                         'entries'                 => [ | 
					
						
							|  |  |  |                             // per category:
 | 
					
						
							|  |  |  |                             // "category" => 5,
 | 
					
						
							|  |  |  |                         ], | 
					
						
							|  |  |  |                     ]; | 
					
						
							|  |  |  |                 foreach ($currency['transaction_journals'] as $journal) { | 
					
						
							|  |  |  |                     // is it expense or income?
 | 
					
						
							|  |  |  |                     $letter                                  = -1 === bccomp($journal['amount'], '0') ? 'e' : 'i'; | 
					
						
							|  |  |  |                     $currentKey                              = sprintf('%d-%s', $currency['currency_id'], $letter); | 
					
						
							|  |  |  |                     $name                                    = (string)trans('firefly.no_category'); | 
					
						
							|  |  |  |                     $tempData[$currentKey]['entries'][$name] = $tempData[$currentKey]['entries'][$name] ?? '0'; | 
					
						
							|  |  |  |                     $tempData[$currentKey]['entries'][$name] = bcadd($tempData[$currentKey]['entries'][$name], $journal['amount']); | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-01-27 10:51:14 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |         // re-sort every spent array and add 0 for missing entries.
 | 
					
						
							|  |  |  |         foreach ($tempData as $index => $set) { | 
					
						
							|  |  |  |             $oldSet = $set['entries']; | 
					
						
							|  |  |  |             $newSet = []; | 
					
						
							| 
									
										
										
										
											2019-08-28 12:28:23 +02:00
										 |  |  |             foreach ($categories as $category) { | 
					
						
							| 
									
										
										
										
											2019-09-04 17:39:39 +02:00
										 |  |  |                 $value             = $oldSet[$category] ?? '0'; | 
					
						
							|  |  |  |                 $value             = -1 === bccomp($value, '0') ? bcmul($value, '-1') : $value; | 
					
						
							| 
									
										
										
										
											2019-08-28 12:28:23 +02:00
										 |  |  |                 $newSet[$category] = $value; | 
					
						
							| 
									
										
										
										
											2019-01-26 20:18:42 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |             $tempData[$index]['entries'] = $newSet; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $chartData = array_values($tempData); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return response()->json($chartData); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-02-09 10:36:59 +01:00
										 |  |  | } |