2016-05-20 01:57:45 -05:00
|
|
|
<?php
|
2016-05-20 05:27:31 -05:00
|
|
|
/**
|
|
|
|
* ReportController.php
|
|
|
|
* Copyright (C) 2016 thegrumpydictator@gmail.com
|
|
|
|
*
|
2016-10-04 23:52:15 -05:00
|
|
|
* This software may be modified and distributed under the terms of the
|
|
|
|
* Creative Commons Attribution-ShareAlike 4.0 International License.
|
|
|
|
*
|
|
|
|
* See the LICENSE file for details.
|
2016-05-20 05:27:31 -05:00
|
|
|
*/
|
|
|
|
|
2016-05-20 01:57:45 -05:00
|
|
|
declare(strict_types = 1);
|
|
|
|
|
|
|
|
namespace FireflyIII\Http\Controllers;
|
2015-02-23 13:25:48 -06:00
|
|
|
|
|
|
|
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\BudgetReportHelperInterface;
|
2015-02-23 13:25:48 -06:00
|
|
|
use FireflyIII\Helpers\Report\ReportHelperInterface;
|
|
|
|
use FireflyIII\Models\Account;
|
2016-05-20 04:02:07 -05:00
|
|
|
use FireflyIII\Models\AccountType;
|
2016-10-09 14:36:03 -05:00
|
|
|
use FireflyIII\Models\Transaction;
|
2016-10-10 00:49:39 -05:00
|
|
|
use FireflyIII\Repositories\Account\AccountRepositoryInterface;
|
2016-10-09 14:36:03 -05:00
|
|
|
use FireflyIII\Repositories\Account\AccountTaskerInterface;
|
2016-05-02 13:49:19 -05:00
|
|
|
use FireflyIII\Repositories\Budget\BudgetRepositoryInterface;
|
|
|
|
use FireflyIII\Repositories\Category\CategoryRepositoryInterface;
|
2015-12-06 06:11:43 -06:00
|
|
|
use Illuminate\Support\Collection;
|
2016-01-29 00:47:18 -06:00
|
|
|
use Preferences;
|
2015-03-04 13:47:00 -06:00
|
|
|
use Session;
|
2016-04-08 07:50:18 -05:00
|
|
|
use Steam;
|
2015-02-23 13:25:48 -06:00
|
|
|
use View;
|
2015-03-04 13:47:00 -06:00
|
|
|
|
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 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-02-23 13:25:48 -06:00
|
|
|
*/
|
2016-09-16 02:36:08 -05:00
|
|
|
public function __construct()
|
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
|
|
|
|
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
|
|
|
/**
|
2016-10-10 00:49:39 -05:00
|
|
|
* @param AccountRepositoryInterface $repository
|
2016-02-04 00:28:39 -06:00
|
|
|
*
|
|
|
|
* @return View
|
|
|
|
*/
|
2016-10-10 00:49:39 -05:00
|
|
|
public function index(AccountRepositoryInterface $repository)
|
2016-02-04 00:28:39 -06:00
|
|
|
{
|
2016-09-16 02:36:08 -05:00
|
|
|
$this->createRepositories();
|
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-10-10 00:49:39 -05:00
|
|
|
$accounts = $repository->getAccountsByType([AccountType::DEFAULT, AccountType::ASSET]);
|
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
|
|
|
{
|
2016-09-16 02:36:08 -05:00
|
|
|
$this->createRepositories();
|
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');
|
|
|
|
}
|
|
|
|
|
2016-04-08 08:10:07 -05:00
|
|
|
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-10-09 14:36:03 -05:00
|
|
|
throw new FireflyException('Unfortunately, reports of the type "' . e($reportType) . '" are not available at this time.');
|
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);
|
|
|
|
}
|
|
|
|
|
2016-04-08 08:10:07 -05:00
|
|
|
// 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':
|
2016-04-08 08:10:07 -05:00
|
|
|
// always default
|
2016-04-08 07:50:18 -05:00
|
|
|
return $this->auditReport($start, $end, $accounts);
|
|
|
|
}
|
2016-02-23 13:23:10 -06:00
|
|
|
|
2016-04-08 07:50:18 -05:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Carbon $start
|
|
|
|
* @param Carbon $end
|
|
|
|
* @param Collection $accounts
|
|
|
|
*
|
|
|
|
* @return View
|
|
|
|
*/
|
|
|
|
private function auditReport(Carbon $start, Carbon $end, Collection $accounts)
|
|
|
|
{
|
2016-10-09 14:36:03 -05:00
|
|
|
/** @var AccountTaskerInterface $tasker */
|
|
|
|
$tasker = app(AccountTaskerInterface::class);
|
2016-04-08 07:50:18 -05:00
|
|
|
$auditData = [];
|
|
|
|
$dayBefore = clone $start;
|
|
|
|
$dayBefore->subDay();
|
|
|
|
/** @var Account $account */
|
|
|
|
foreach ($accounts as $account) {
|
|
|
|
// balance the day before:
|
|
|
|
$id = $account->id;
|
|
|
|
$dayBeforeBalance = Steam::balance($account, $dayBefore);
|
2016-10-09 14:36:03 -05:00
|
|
|
$journals = $tasker->getJournalsInPeriod(new Collection([$account]), [], $start, $end);
|
|
|
|
$journals = $journals->reverse();
|
|
|
|
$startBalance = $dayBeforeBalance;
|
|
|
|
|
|
|
|
|
|
|
|
/** @var Transaction $journal */
|
|
|
|
foreach ($journals as $transaction) {
|
|
|
|
$transaction->before = $startBalance;
|
|
|
|
$transactionAmount = $transaction->transaction_amount;
|
|
|
|
$newBalance = bcadd($startBalance, $transactionAmount);
|
|
|
|
$transaction->after = $newBalance;
|
|
|
|
$startBalance = $newBalance;
|
2016-04-08 07:50:18 -05:00
|
|
|
}
|
|
|
|
|
2016-04-08 08:10:07 -05:00
|
|
|
/*
|
|
|
|
* Reverse set again.
|
|
|
|
*/
|
|
|
|
$auditData[$id]['journals'] = $journals->reverse();
|
2016-10-09 14:36:03 -05:00
|
|
|
$auditData[$id]['exists'] = $journals->count() > 0;
|
2016-05-20 01:00:35 -05:00
|
|
|
$auditData[$id]['end'] = $end->formatLocalized(strval(trans('config.month_and_day')));
|
2016-04-08 07:50:18 -05:00
|
|
|
$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')));
|
2016-04-08 07:50:18 -05:00
|
|
|
$auditData[$id]['dayBeforeBalance'] = $dayBeforeBalance;
|
2016-02-04 00:28:39 -06:00
|
|
|
}
|
|
|
|
|
2016-04-08 07:50:18 -05:00
|
|
|
$reportType = 'audit';
|
|
|
|
$accountIds = join(',', $accounts->pluck('id')->toArray());
|
|
|
|
|
2016-09-10 11:37:16 -05:00
|
|
|
$hideable = ['buttons', 'icon', 'description', 'balance_before', 'amount', 'balance_after', 'date',
|
2016-09-16 02:36:08 -05:00
|
|
|
'interest_date', 'book_date', 'process_date',
|
2016-09-10 11:37:16 -05:00
|
|
|
// three new optional fields.
|
|
|
|
'due_date', 'payment_date', 'invoice_date',
|
|
|
|
'from', 'to', 'budget', 'category', 'bill',
|
|
|
|
// more new optional fields
|
|
|
|
'internal_reference', 'notes',
|
|
|
|
|
|
|
|
'create_date', 'update_date',
|
2016-04-08 07:50:18 -05:00
|
|
|
];
|
|
|
|
$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
|
|
|
}
|
|
|
|
|
2016-09-16 02:36:08 -05:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
private function createRepositories()
|
|
|
|
{
|
2016-10-26 12:45:10 -05:00
|
|
|
$this->helper = app(ReportHelperInterface::class);
|
|
|
|
$this->budgetHelper = app(BudgetReportHelperInterface::class);
|
2016-09-16 02:36:08 -05:00
|
|
|
}
|
|
|
|
|
2015-12-06 06:11:43 -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
|
2015-12-06 06:11:43 -06:00
|
|
|
*/
|
2016-02-05 02:25:15 -06:00
|
|
|
private function defaultMonth(string $reportType, Carbon $start, Carbon $end, Collection $accounts)
|
2015-12-06 06:11:43 -06:00
|
|
|
{
|
|
|
|
// get report stuff!
|
2016-10-26 12:45:10 -05:00
|
|
|
$budgets = $this->budgetHelper->getBudgetReport($start, $end, $accounts);
|
|
|
|
$bills = $this->helper->getBillReport($start, $end, $accounts);
|
|
|
|
$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());
|
2015-12-06 06:11:43 -06:00
|
|
|
|
|
|
|
// continue!
|
|
|
|
return view(
|
2015-12-13 02:01:17 -06:00
|
|
|
'reports.default.month',
|
2015-12-06 06:11:43 -06:00
|
|
|
compact(
|
2016-10-26 12:45:10 -05:00
|
|
|
'start', 'end',
|
2016-10-08 09:24:07 -05:00
|
|
|
'tags',
|
2016-10-26 12:45:10 -05:00
|
|
|
'budgets',
|
2015-12-12 10:51:07 -06:00
|
|
|
'bills',
|
2015-12-28 13:04:54 -06:00
|
|
|
'accountIds', 'reportType'
|
2015-12-06 06:11:43 -06:00
|
|
|
)
|
|
|
|
);
|
2015-12-12 12:04:30 -06:00
|
|
|
}
|
|
|
|
|
2015-12-28 00:55:09 -06:00
|
|
|
/**
|
2015-12-28 13:04:54 -06:00
|
|
|
* @param $reportType
|
2015-12-28 00:55:09 -06:00
|
|
|
* @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-13 10:31:25 -06:00
|
|
|
{
|
2015-12-14 13:45:12 -06:00
|
|
|
|
2016-10-08 09:24:07 -05:00
|
|
|
$budgets = app(BudgetRepositoryInterface::class)->getActiveBudgets();
|
|
|
|
$categories = app(CategoryRepositoryInterface::class)->getCategories();
|
|
|
|
$tags = $this->helper->tagReport($start, $end, $accounts);
|
2015-12-13 10:31:25 -06:00
|
|
|
|
|
|
|
// 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-13 10:31:25 -06:00
|
|
|
|
2015-12-14 13:34:08 -06:00
|
|
|
return view(
|
2016-01-05 14:23:58 -06:00
|
|
|
'reports.default.multi-year',
|
|
|
|
compact(
|
2016-10-25 11:53:54 -05:00
|
|
|
'budgets', 'accounts', 'categories', 'start', 'end', 'accountIds', 'reportType', 'tags'
|
2016-01-05 14:23:58 -06:00
|
|
|
)
|
2015-12-14 13:34:08 -06:00
|
|
|
);
|
2015-12-13 10:31:25 -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
|
|
|
{
|
2016-10-26 12:45:10 -05:00
|
|
|
$tags = $this->helper->tagReport($start, $end, $accounts);
|
|
|
|
$budgets = $this->budgetHelper->budgetYearOverview($start, $end, $accounts);
|
2016-04-24 13:00:20 -05:00
|
|
|
|
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(
|
2016-10-25 11:53:54 -05:00
|
|
|
'start', 'reportType', 'accountIds', 'end', 'tags', 'budgets'
|
2016-01-24 08:58:16 -06:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
2015-02-23 13:25:48 -06:00
|
|
|
}
|