mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-25 21:16:47 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			401 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			401 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /**
 | |
|  * ReportController.php
 | |
|  * Copyright (c) 2019 james@firefly-iii.org
 | |
|  *
 | |
|  * This file is part of Firefly III (https://github.com/firefly-iii).
 | |
|  *
 | |
|  * 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.
 | |
|  *
 | |
|  * This program 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 Affero General Public License for more details.
 | |
|  *
 | |
|  * 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/>.
 | |
|  */
 | |
| declare(strict_types=1);
 | |
| 
 | |
| namespace FireflyIII\Http\Controllers;
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use FireflyIII\Enums\AccountTypeEnum;
 | |
| use FireflyIII\Exceptions\FireflyException;
 | |
| use FireflyIII\Generator\Report\ReportGeneratorFactory;
 | |
| use FireflyIII\Helpers\Report\ReportHelperInterface;
 | |
| use FireflyIII\Http\Requests\ReportFormRequest;
 | |
| use FireflyIII\Models\Account;
 | |
| use FireflyIII\Repositories\Account\AccountRepositoryInterface;
 | |
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface;
 | |
| use FireflyIII\Support\Http\Controllers\RenderPartialViews;
 | |
| use Illuminate\Contracts\View\Factory;
 | |
| use Illuminate\Http\JsonResponse;
 | |
| use Illuminate\Http\RedirectResponse;
 | |
| use Illuminate\Routing\Redirector;
 | |
| use Illuminate\Support\Collection;
 | |
| use Illuminate\View\View;
 | |
| 
 | |
| /**
 | |
|  * Class ReportController.
 | |
|  */
 | |
| class ReportController extends Controller
 | |
| {
 | |
|     use RenderPartialViews;
 | |
| 
 | |
|     protected ReportHelperInterface   $helper;
 | |
|     private BudgetRepositoryInterface $repository;
 | |
| 
 | |
|     /**
 | |
|      * ReportController constructor.
 | |
|      */
 | |
|     public function __construct()
 | |
|     {
 | |
|         parent::__construct();
 | |
| 
 | |
|         $this->middleware(
 | |
|             function ($request, $next) {
 | |
|                 app('view')->share('title', (string) trans('firefly.reports'));
 | |
|                 app('view')->share('mainTitleIcon', 'fa-bar-chart');
 | |
|                 app('view')->share('subTitleIcon', 'fa-calendar');
 | |
|                 $this->helper     = app(ReportHelperInterface::class);
 | |
|                 $this->repository = app(BudgetRepositoryInterface::class);
 | |
| 
 | |
|                 return $next($request);
 | |
|             }
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show audit report.
 | |
|      *
 | |
|      * @return Factory|string|View
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function auditReport(Collection $accounts, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         if ($end < $start) {
 | |
|             return view('error')->with('message', (string) trans('firefly.end_after_start_date'));
 | |
|         }
 | |
|         $this->repository->cleanupBudgets();
 | |
|         $start->endOfDay(); // end of day so the final balance is at the end of that day.
 | |
|         $end->endOfDay();
 | |
| 
 | |
|         app('view')->share(
 | |
|             'subTitle',
 | |
|             trans(
 | |
|                 'firefly.report_audit',
 | |
|                 [
 | |
|                     'start' => $start->isoFormat($this->monthAndDayFormat),
 | |
|                     'end'   => $end->isoFormat($this->monthAndDayFormat),
 | |
|                 ]
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $generator = ReportGeneratorFactory::reportGenerator('Audit', $start, $end);
 | |
|         $generator->setAccounts($accounts);
 | |
| 
 | |
|         return $generator->generate();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show budget report.
 | |
|      *
 | |
|      * @return Factory|string|View
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function budgetReport(Collection $accounts, Collection $budgets, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         if ($end < $start) {
 | |
|             return view('error')->with('message', (string) trans('firefly.end_after_start_date'));
 | |
|         }
 | |
|         $this->repository->cleanupBudgets();
 | |
|         $start->endOfDay(); // end of day so the final balance is at the end of that day.
 | |
|         $end->endOfDay();
 | |
| 
 | |
|         app('view')->share(
 | |
|             'subTitle',
 | |
|             trans(
 | |
|                 'firefly.report_budget',
 | |
|                 [
 | |
|                     'start' => $start->isoFormat($this->monthAndDayFormat),
 | |
|                     'end'   => $end->isoFormat($this->monthAndDayFormat),
 | |
|                 ]
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $generator = ReportGeneratorFactory::reportGenerator('Budget', $start, $end);
 | |
|         $generator->setAccounts($accounts);
 | |
|         $generator->setBudgets($budgets);
 | |
| 
 | |
|         return $generator->generate();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show category report.
 | |
|      *
 | |
|      * @return Factory|string|View
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function categoryReport(Collection $accounts, Collection $categories, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         if ($end < $start) {
 | |
|             return view('error')->with('message', (string) trans('firefly.end_after_start_date'));
 | |
|         }
 | |
|         $this->repository->cleanupBudgets();
 | |
|         $start->endOfDay(); // end of day so the final balance is at the end of that day.
 | |
|         $end->endOfDay();
 | |
| 
 | |
|         app('view')->share(
 | |
|             'subTitle',
 | |
|             trans(
 | |
|                 'firefly.report_category',
 | |
|                 [
 | |
|                     'start' => $start->isoFormat($this->monthAndDayFormat),
 | |
|                     'end'   => $end->isoFormat($this->monthAndDayFormat),
 | |
|                 ]
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $generator = ReportGeneratorFactory::reportGenerator('Category', $start, $end);
 | |
|         $generator->setAccounts($accounts);
 | |
|         $generator->setCategories($categories);
 | |
| 
 | |
|         return $generator->generate();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show default report.
 | |
|      *
 | |
|      * @return Factory|string|View
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function defaultReport(Collection $accounts, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         if ($end < $start) {
 | |
|             return view('error')->with('message', (string) trans('firefly.end_after_start_date'));
 | |
|         }
 | |
| 
 | |
|         $this->repository->cleanupBudgets();
 | |
|         $start->endOfDay(); // end of day so the final balance is at the end of that day.
 | |
|         $end->endOfDay();
 | |
| 
 | |
|         app('view')->share(
 | |
|             'subTitle',
 | |
|             trans(
 | |
|                 'firefly.report_default',
 | |
|                 [
 | |
|                     'start' => $start->isoFormat($this->monthAndDayFormat),
 | |
|                     'end'   => $end->isoFormat($this->monthAndDayFormat),
 | |
|                 ]
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $generator = ReportGeneratorFactory::reportGenerator('Standard', $start, $end);
 | |
|         $generator->setAccounts($accounts);
 | |
| 
 | |
|         return $generator->generate();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show account report.
 | |
|      *
 | |
|      * @return string
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function doubleReport(Collection $accounts, Collection $expense, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         if ($end < $start) {
 | |
|             [$start, $end] = [$end, $start];
 | |
|         }
 | |
| 
 | |
|         $this->repository->cleanupBudgets();
 | |
|         $start->endOfDay(); // end of day so the final balance is at the end of that day.
 | |
|         $end->endOfDay();
 | |
| 
 | |
|         app('view')->share(
 | |
|             'subTitle',
 | |
|             trans(
 | |
|                 'firefly.report_double',
 | |
|                 [
 | |
|                     'start' => $start->isoFormat($this->monthAndDayFormat),
 | |
|                     'end'   => $end->isoFormat($this->monthAndDayFormat),
 | |
|                 ]
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $generator = ReportGeneratorFactory::reportGenerator('Account', $start, $end);
 | |
|         $generator->setAccounts($accounts);
 | |
|         $generator->setExpense($expense);
 | |
| 
 | |
|         return $generator->generate();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show index.
 | |
|      *
 | |
|      * @return Factory|View
 | |
|      */
 | |
|     public function index(AccountRepositoryInterface $repository)
 | |
|     {
 | |
|         /** @var Carbon $start */
 | |
|         $start            = clone session('first', today(config('app.timezone')));
 | |
|         $months           = $this->helper->listOfMonths($start);
 | |
|         $customFiscalYear = app('preferences')->get('customFiscalYear', 0)->data;
 | |
|         $accounts         = $repository->getAccountsByType(
 | |
|             [AccountTypeEnum::DEFAULT->value, AccountTypeEnum::ASSET->value, AccountTypeEnum::DEBT->value, AccountTypeEnum::LOAN->value, AccountTypeEnum::MORTGAGE->value]
 | |
|         );
 | |
| 
 | |
|         // group accounts by role:
 | |
|         $groupedAccounts  = [];
 | |
| 
 | |
|         /** @var Account $account */
 | |
|         foreach ($accounts as $account) {
 | |
|             $type                                                                        = $account->accountType->type;
 | |
|             $role                                                                        = sprintf('opt_group_%s', $repository->getMetaValue($account, 'account_role'));
 | |
| 
 | |
|             if (in_array($type, [AccountTypeEnum::MORTGAGE->value, AccountTypeEnum::DEBT->value, AccountTypeEnum::LOAN->value], true)) {
 | |
|                 $role = sprintf('opt_group_l_%s', $type);
 | |
|             }
 | |
| 
 | |
|             if ('opt_group_' === $role) {
 | |
|                 $role = 'opt_group_defaultAsset';
 | |
|             }
 | |
|             $groupedAccounts[(string) trans(sprintf('firefly.%s', $role))][$account->id] = $account;
 | |
|         }
 | |
|         ksort($groupedAccounts);
 | |
| 
 | |
|         $accountList      = implode(',', $accounts->pluck('id')->toArray());
 | |
|         $this->repository->cleanupBudgets();
 | |
| 
 | |
|         return view('reports.index', compact('months', 'accounts', 'start', 'accountList', 'groupedAccounts', 'customFiscalYear'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show options for reports.
 | |
|      *
 | |
|      * @return JsonResponse
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function options(string $reportType)
 | |
|     {
 | |
|         $result = match ($reportType) {
 | |
|             default    => $this->noReportOptions(),
 | |
|             'category' => $this->categoryReportOptions(),
 | |
|             'budget'   => $this->budgetReportOptions(),
 | |
|             'tag'      => $this->tagReportOptions(),
 | |
|             'double'   => $this->doubleReportOptions(),
 | |
|         };
 | |
| 
 | |
|         return response()->json(['html' => $result]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Process the submit of report.
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function postIndex(ReportFormRequest $request): Redirector|RedirectResponse|View
 | |
|     {
 | |
|         // report type:
 | |
|         $reportType = $request->get('report_type');
 | |
|         $start      = $request->getStartDate()->format('Ymd');
 | |
|         $end        = $request->getEndDate()->format('Ymd');
 | |
|         $accounts   = implode(',', $request->getAccountList()->pluck('id')->toArray());
 | |
|         $categories = implode(',', $request->getCategoryList()->pluck('id')->toArray());
 | |
|         $budgets    = implode(',', $request->getBudgetList()->pluck('id')->toArray());
 | |
|         $tags       = implode(',', $request->getTagList()->pluck('id')->toArray());
 | |
|         $double     = implode(',', $request->getDoubleList()->pluck('id')->toArray());
 | |
| 
 | |
|         if (0 === $request->getAccountList()->count()) {
 | |
|             app('log')->debug('Account count is zero');
 | |
|             session()->flash('error', (string) trans('firefly.select_at_least_one_account'));
 | |
| 
 | |
|             return redirect(route('reports.index'));
 | |
|         }
 | |
| 
 | |
|         if ('category' === $reportType && 0 === $request->getCategoryList()->count()) {
 | |
|             session()->flash('error', (string) trans('firefly.select_at_least_one_category'));
 | |
| 
 | |
|             return redirect(route('reports.index'));
 | |
|         }
 | |
| 
 | |
|         if ('budget' === $reportType && 0 === $request->getBudgetList()->count()) {
 | |
|             session()->flash('error', (string) trans('firefly.select_at_least_one_budget'));
 | |
| 
 | |
|             return redirect(route('reports.index'));
 | |
|         }
 | |
| 
 | |
|         if ('tag' === $reportType && 0 === $request->getTagList()->count()) {
 | |
|             session()->flash('error', (string) trans('firefly.select_at_least_one_tag'));
 | |
| 
 | |
|             return redirect(route('reports.index'));
 | |
|         }
 | |
| 
 | |
|         if ('double' === $reportType && 0 === $request->getDoubleList()->count()) {
 | |
|             session()->flash('error', (string) trans('firefly.select_at_least_one_expense'));
 | |
| 
 | |
|             return redirect(route('reports.index'));
 | |
|         }
 | |
| 
 | |
|         if ($request->getEndDate() < $request->getStartDate()) {
 | |
|             return view('error')->with('message', (string) trans('firefly.end_after_start_date'));
 | |
|         }
 | |
| 
 | |
|         $url        = match ($reportType) {
 | |
|             default    => route('reports.report.default', [$accounts, $start, $end]),
 | |
|             'category' => route('reports.report.category', [$accounts, $categories, $start, $end]),
 | |
|             'audit'    => route('reports.report.audit', [$accounts, $start, $end]),
 | |
|             'budget'   => route('reports.report.budget', [$accounts, $budgets, $start, $end]),
 | |
|             'tag'      => route('reports.report.tag', [$accounts, $tags, $start, $end]),
 | |
|             'double'   => route('reports.report.double', [$accounts, $double, $start, $end]),
 | |
|         };
 | |
| 
 | |
|         return redirect($url);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a tag report.
 | |
|      *
 | |
|      * @return Factory|string|View
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function tagReport(Collection $accounts, Collection $tags, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         if ($end < $start) {
 | |
|             return view('error')->with('message', (string) trans('firefly.end_after_start_date'));
 | |
|         }
 | |
|         $this->repository->cleanupBudgets();
 | |
|         $start->endOfDay(); // end of day so the final balance is at the end of that day.
 | |
|         $end->endOfDay();
 | |
| 
 | |
|         app('view')->share(
 | |
|             'subTitle',
 | |
|             trans(
 | |
|                 'firefly.report_tag',
 | |
|                 [
 | |
|                     'start' => $start->isoFormat($this->monthAndDayFormat),
 | |
|                     'end'   => $end->isoFormat($this->monthAndDayFormat),
 | |
|                 ]
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $generator = ReportGeneratorFactory::reportGenerator('Tag', $start, $end);
 | |
|         $generator->setAccounts($accounts);
 | |
|         $generator->setTags($tags);
 | |
| 
 | |
|         return $generator->generate();
 | |
|     }
 | |
| }
 |