firefly-iii/app/Http/Controllers/ReportController.php

336 lines
12 KiB
PHP
Raw Normal View History

2015-02-23 13:25:48 -06:00
<?php namespace FireflyIII\Http\Controllers;
use Carbon\Carbon;
2016-02-05 22:01:34 -06:00
use FireflyIII\Exceptions\FireflyException;
2016-03-02 06:13:33 -06:00
use FireflyIII\Helpers\Report\AccountReportHelperInterface;
use FireflyIII\Helpers\Report\BalanceReportHelperInterface;
use FireflyIII\Helpers\Report\BudgetReportHelperInterface;
2015-02-23 13:25:48 -06:00
use FireflyIII\Helpers\Report\ReportHelperInterface;
use FireflyIII\Models\Account;
use FireflyIII\Models\TransactionJournal;
use FireflyIII\Repositories\Account\AccountRepositoryInterface as ARI;
2016-05-02 13:49:19 -05:00
use FireflyIII\Repositories\Budget\BudgetRepositoryInterface;
use FireflyIII\Repositories\Category\CategoryRepositoryInterface;
use Illuminate\Support\Collection;
2016-01-29 00:47:18 -06:00
use Preferences;
use Session;
use Steam;
2015-02-23 13:25:48 -06:00
use View;
2015-02-23 13:25:48 -06:00
/**
* Class ReportController
*
* @package FireflyIII\Http\Controllers
*/
class ReportController extends Controller
{
2016-03-02 06:13:33 -06:00
/** @var AccountReportHelperInterface */
2016-01-27 13:45:49 -06:00
protected $accountHelper;
2016-03-02 06:13:33 -06:00
/** @var BalanceReportHelperInterface */
2016-01-27 14:35:59 -06:00
protected $balanceHelper;
2016-03-02 06:13:33 -06:00
/** @var BudgetReportHelperInterface */
2016-01-27 14:18:51 -06:00
protected $budgetHelper;
2015-03-29 05:25:46 -05:00
/** @var ReportHelperInterface */
protected $helper;
2015-02-23 13:25:48 -06:00
/**
2016-02-04 00:28:39 -06:00
*
2015-05-24 04:41:52 -05:00
*
2015-03-29 05:25:46 -05:00
* @param ReportHelperInterface $helper
2015-02-23 13:25:48 -06:00
*/
2015-05-16 09:04:51 -05:00
public function __construct(ReportHelperInterface $helper)
2015-02-23 13:25:48 -06:00
{
2015-05-15 15:00:00 -05:00
parent::__construct();
2016-01-27 13:45:49 -06:00
$this->helper = $helper;
2016-05-02 13:49:19 -05:00
$this->accountHelper = app(AccountReportHelperInterface::class);
$this->budgetHelper = app(BudgetReportHelperInterface::class);
$this->balanceHelper = app(BalanceReportHelperInterface::class);
2015-03-29 05:25:46 -05:00
2015-05-15 14:01:24 -05:00
View::share('title', trans('firefly.reports'));
2015-02-23 13:25:48 -06:00
View::share('mainTitleIcon', 'fa-line-chart');
}
2016-02-04 00:28:39 -06:00
/**
* @param ARI $repository
*
* @return View
* @internal param ReportHelperInterface $helper
*/
public function index(ARI $repository)
{
2016-02-05 08:41:40 -06:00
/** @var Carbon $start */
2016-02-07 02:11:46 -06:00
$start = clone session('first');
2016-02-04 00:28:39 -06:00
$months = $this->helper->listOfMonths($start);
$customFiscalYear = Preferences::get('customFiscalYear', 0)->data;
// does the user have shared accounts?
2016-05-13 08:53:39 -05:00
$accounts = $repository->getAccountsByType(['Default account', 'Asset account']);
2016-02-04 00:28:39 -06:00
// get id's for quick links:
$accountIds = [];
/** @var Account $account */
foreach ($accounts as $account) {
$accountIds [] = $account->id;
}
$accountList = join(',', $accountIds);
return view('reports.index', compact('months', 'accounts', 'start', 'accountList', 'customFiscalYear'));
}
/**
2016-02-07 00:56:58 -06:00
* @param string $reportType
2016-02-04 00:28:39 -06:00
* @param Carbon $start
* @param Carbon $end
* @param Collection $accounts
*
* @return View
2016-02-07 00:56:58 -06:00
* @throws FireflyException
2016-02-04 00:28:39 -06:00
*/
2016-02-05 02:25:15 -06:00
public function report(string $reportType, Carbon $start, Carbon $end, Collection $accounts)
2016-02-04 00:28:39 -06:00
{
// throw an error if necessary.
if ($end < $start) {
2016-02-05 22:04:41 -06:00
throw new FireflyException('End date cannot be before start date, silly!');
2016-02-04 00:28:39 -06:00
}
// lower threshold
if ($start < session('first')) {
$start = session('first');
}
View::share(
'subTitle', trans(
'firefly.report_' . $reportType,
[
'start' => $start->formatLocalized($this->monthFormat),
'end' => $end->formatLocalized($this->monthFormat),
]
)
);
View::share('subTitleIcon', 'fa-calendar');
2016-02-04 00:28:39 -06:00
switch ($reportType) {
default:
2016-02-23 13:23:10 -06:00
throw new FireflyException('Unfortunately, reports of the type "' . e($reportType) . '" are not yet available. ');
2016-02-04 00:28:39 -06:00
case 'default':
// more than one year date difference means year report.
if ($start->diffInMonths($end) > 12) {
return $this->defaultMultiYear($reportType, $start, $end, $accounts);
}
// more than two months date difference means year report.
if ($start->diffInMonths($end) > 1) {
return $this->defaultYear($reportType, $start, $end, $accounts);
}
// otherwise default
2016-02-04 00:28:39 -06:00
return $this->defaultMonth($reportType, $start, $end, $accounts);
2016-02-23 13:23:10 -06:00
case 'audit':
// always default
return $this->auditReport($start, $end, $accounts);
}
2016-02-23 13:23:10 -06:00
}
/**
* @param Carbon $start
* @param Carbon $end
* @param Collection $accounts
*
* @return View
*/
private function auditReport(Carbon $start, Carbon $end, Collection $accounts)
{
/** @var ARI $repos */
2016-05-01 08:05:29 -05:00
$repos = app(ARI::class);
$auditData = [];
$dayBefore = clone $start;
$dayBefore->subDay();
/** @var Account $account */
foreach ($accounts as $account) {
// balance the day before:
$id = $account->id;
$first = $repos->oldestJournalDate($account);
$last = $repos->newestJournalDate($account);
$exists = false;
$journals = new Collection;
$dayBeforeBalance = Steam::balance($account, $dayBefore);
/*
* Is there even activity on this account between the requested dates?
*/
if ($start->between($first, $last) || $end->between($first, $last)) {
2016-05-15 11:36:40 -05:00
$exists = true;
2016-05-17 09:47:43 -05:00
$journals = $repos->journalsInPeriod(new Collection([$account]), [], $start, $end);
}
/*
* Reverse set, get balances.
*/
$journals = $journals->reverse();
$startBalance = $dayBeforeBalance;
/** @var TransactionJournal $journal */
foreach ($journals as $journal) {
$journal->before = $startBalance;
2016-05-17 09:11:19 -05:00
$transactionAmount = $journal->source_amount;
// get currently relevant transaction:
if (intval($journal->destination_account_id) === $account->id) {
2016-05-17 09:11:19 -05:00
$transactionAmount = $journal->destination_amount;
}
$newBalance = bcadd($startBalance, $transactionAmount);
$journal->after = $newBalance;
$startBalance = $newBalance;
}
/*
* Reverse set again.
*/
$auditData[$id]['journals'] = $journals->reverse();
$auditData[$id]['exists'] = $exists;
2016-05-20 01:00:35 -05:00
$auditData[$id]['end'] = $end->formatLocalized(strval(trans('config.month_and_day')));
$auditData[$id]['endBalance'] = Steam::balance($account, $end);
2016-05-20 01:00:35 -05:00
$auditData[$id]['dayBefore'] = $dayBefore->formatLocalized(strval(trans('config.month_and_day')));
$auditData[$id]['dayBeforeBalance'] = $dayBeforeBalance;
2016-02-04 00:28:39 -06:00
}
$reportType = 'audit';
$accountIds = join(',', $accounts->pluck('id')->toArray());
$hideable = ['buttons', 'icon', 'description', 'balance_before', 'amount', 'balance_after', 'date', 'book_date', 'process_date', 'interest_date',
'from', 'to', 'budget', 'category', 'bill', 'create_date', 'update_date',
];
$defaultShow = ['icon', 'description', 'balance_before', 'amount', 'balance_after', 'date', 'to'];
return view('reports.audit.report', compact('start', 'end', 'reportType', 'accountIds', 'accounts', 'auditData', 'hideable', 'defaultShow'));
2016-02-04 00:28:39 -06:00
}
/**
2015-12-28 13:04:54 -06:00
* @param $reportType
2015-12-12 10:51:07 -06:00
* @param Carbon $start
* @param Carbon $end
* @param Collection $accounts
*
* @return View
*/
2016-02-05 02:25:15 -06:00
private function defaultMonth(string $reportType, Carbon $start, Carbon $end, Collection $accounts)
{
$incomeTopLength = 8;
$expenseTopLength = 8;
// get report stuff!
2016-01-27 14:52:21 -06:00
$accountReport = $this->accountHelper->getAccountReport($start, $end, $accounts);
$incomes = $this->helper->getIncomeReport($start, $end, $accounts);
$expenses = $this->helper->getExpenseReport($start, $end, $accounts);
$budgets = $this->budgetHelper->getBudgetReport($start, $end, $accounts);
$categories = $this->helper->getCategoryReport($start, $end, $accounts);
$balance = $this->balanceHelper->getBalanceReport($start, $end, $accounts);
2016-01-01 12:52:55 -06:00
$bills = $this->helper->getBillReport($start, $end, $accounts);
2016-03-01 14:31:25 -06:00
$tags = $this->helper->tagReport($start, $end, $accounts);
2015-12-12 10:51:07 -06:00
// and some id's, joined:
2016-01-01 12:46:12 -06:00
$accountIds = join(',', $accounts->pluck('id')->toArray());
// continue!
return view(
2015-12-13 02:01:17 -06:00
'reports.default.month',
compact(
2015-12-28 13:04:54 -06:00
'start', 'end', 'reportType',
2016-03-01 14:31:25 -06:00
'accountReport', 'tags',
'incomes', 'incomeTopLength',
'expenses', 'expenseTopLength',
2015-12-12 10:51:07 -06:00
'budgets', 'balance',
'categories',
2015-12-12 10:51:07 -06:00
'bills',
2015-12-28 13:04:54 -06:00
'accountIds', 'reportType'
)
);
}
/**
2015-12-28 13:04:54 -06:00
* @param $reportType
* @param $start
* @param $end
* @param $accounts
*
* @return View
*/
2016-02-05 02:25:15 -06:00
private function defaultMultiYear(string $reportType, Carbon $start, Carbon $end, Collection $accounts)
{
2015-12-14 13:45:12 -06:00
2016-01-05 14:23:58 -06:00
$incomeTopLength = 8;
$expenseTopLength = 8;
// list of users stuff:
2016-05-02 13:49:19 -05:00
$budgets = app(BudgetRepositoryInterface::class)->getActiveBudgets();
$categories = app(CategoryRepositoryInterface::class)->getCategories();
2016-01-27 14:52:21 -06:00
$accountReport = $this->accountHelper->getAccountReport($start, $end, $accounts);
$incomes = $this->helper->getIncomeReport($start, $end, $accounts);
$expenses = $this->helper->getExpenseReport($start, $end, $accounts);
$tags = $this->helper->tagReport($start, $end, $accounts);
// and some id's, joined:
$accountIds = [];
/** @var Account $account */
foreach ($accounts as $account) {
$accountIds[] = $account->id;
}
2015-12-15 05:38:18 -06:00
$accountIds = join(',', $accountIds);
2015-12-14 13:34:08 -06:00
return view(
2016-01-05 14:23:58 -06:00
'reports.default.multi-year',
compact(
'budgets', 'accounts', 'categories', 'start', 'end', 'accountIds', 'reportType', 'accountReport', 'incomes', 'expenses',
'incomeTopLength', 'expenseTopLength', 'tags'
2016-01-05 14:23:58 -06:00
)
2015-12-14 13:34:08 -06:00
);
}
2016-01-24 08:58:16 -06:00
/**
* @param $reportType
* @param Carbon $start
* @param Carbon $end
* @param Collection $accounts
*
* @return View
*/
2016-02-05 02:25:15 -06:00
private function defaultYear(string $reportType, Carbon $start, Carbon $end, Collection $accounts)
2016-01-24 08:58:16 -06:00
{
$incomeTopLength = 8;
$expenseTopLength = 8;
2016-01-27 13:46:38 -06:00
$accountReport = $this->accountHelper->getAccountReport($start, $end, $accounts);
2016-01-24 08:58:16 -06:00
$incomes = $this->helper->getIncomeReport($start, $end, $accounts);
$expenses = $this->helper->getExpenseReport($start, $end, $accounts);
2016-03-01 14:31:25 -06:00
$tags = $this->helper->tagReport($start, $end, $accounts);
2016-01-24 08:58:16 -06:00
2016-04-24 13:00:20 -05:00
// find the budgets we've spent money on this period with these accounts:
$budgets = $this->budgetHelper->getBudgetsWithExpenses($start, $end, $accounts);
2016-01-24 08:58:16 -06:00
Session::flash('gaEventCategory', 'report');
Session::flash('gaEventAction', 'year');
Session::flash('gaEventLabel', $start->format('Y'));
// and some id's, joined:
$accountIds = [];
/** @var Account $account */
foreach ($accounts as $account) {
$accountIds[] = $account->id;
}
$accountIds = join(',', $accountIds);
return view(
'reports.default.year',
compact(
'start', 'accountReport', 'incomes', 'reportType', 'accountIds', 'end',
'expenses', 'incomeTopLength', 'expenseTopLength', 'tags', 'budgets'
2016-01-24 08:58:16 -06:00
)
);
}
2015-02-23 13:25:48 -06:00
}