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

444 lines
17 KiB
PHP
Raw Normal View History

2016-05-20 01:57:45 -05:00
<?php
declare(strict_types = 1);
namespace FireflyIII\Http\Controllers;
use Auth;
2015-02-09 00:23:39 -06:00
use Carbon\Carbon;
2015-07-10 13:48:45 -05:00
use ExpandedForm;
use FireflyIII\Http\Requests\AccountFormRequest;
use FireflyIII\Models\Account;
2016-05-19 23:58:13 -05:00
use FireflyIII\Models\AccountType;
use FireflyIII\Repositories\Account\AccountRepositoryInterface as ARI;
2016-05-13 08:53:39 -05:00
use FireflyIII\Support\CacheProperties;
use Illuminate\Pagination\LengthAwarePaginator;
use Illuminate\Support\Collection;
2015-02-22 01:38:46 -06:00
use Input;
2016-05-13 08:53:39 -05:00
use Navigation;
2015-06-03 14:25:11 -05:00
use Preferences;
2015-02-09 00:23:39 -06:00
use Session;
2015-03-02 08:27:36 -06:00
use Steam;
2015-04-28 01:12:12 -05:00
use URL;
2015-03-02 08:44:06 -06:00
use View;
2015-06-03 14:25:11 -05:00
/**
* Class AccountController
*
* @package FireflyIII\Http\Controllers
*/
class AccountController extends Controller
{
2015-02-11 00:35:10 -06:00
/**
2016-02-04 00:27:03 -06:00
*
2015-02-11 00:35:10 -06:00
*/
public function __construct()
{
2015-04-28 08:26:30 -05:00
parent::__construct();
View::share('mainTitleIcon', 'fa-credit-card');
2015-05-14 06:17:53 -05:00
View::share('title', trans('firefly.accounts'));
}
/**
* @param string $what
*
* @return \Illuminate\View\View
*/
2016-02-05 02:25:15 -06:00
public function create(string $what = 'asset')
{
2016-04-26 14:40:15 -05:00
$subTitleIcon = config('firefly.subIconsByIdentifier.' . $what);
$subTitle = trans('firefly.make_new_' . $what . '_account');
2016-03-30 10:47:13 -05:00
Session::flash('preFilled', []);
2015-04-28 01:12:12 -05:00
// put previous url in session if not redirect from store (not "create another").
2016-02-04 00:27:03 -06:00
if (session('accounts.create.fromStore') !== true) {
2015-04-28 01:12:12 -05:00
Session::put('accounts.create.url', URL::previous());
}
Session::forget('accounts.create.fromStore');
2015-05-25 01:01:06 -05:00
Session::flash('gaEventCategory', 'accounts');
Session::flash('gaEventAction', 'create-' . $what);
2015-04-28 01:12:12 -05:00
return view('accounts.create', compact('subTitleIcon', 'what', 'subTitle'));
}
/**
2015-12-31 10:20:54 -06:00
* @param ARI $repository
* @param Account $account
*
* @return \Illuminate\View\View
*/
public function delete(ARI $repository, Account $account)
{
2016-04-26 14:40:15 -05:00
$typeName = config('firefly.shortNamesByFullName.' . $account->accountType->type);
2015-07-10 13:48:45 -05:00
$subTitle = trans('firefly.delete_' . $typeName . '_account', ['name' => $account->name]);
2016-05-13 08:53:39 -05:00
$accountList = ExpandedForm::makeSelectListWithEmpty($repository->getAccountsByType([$account->accountType->type]));
2015-07-10 13:48:45 -05:00
unset($accountList[$account->id]);
2015-04-28 01:12:12 -05:00
// put previous url in session
Session::put('accounts.delete.url', URL::previous());
2015-05-25 01:01:06 -05:00
Session::flash('gaEventCategory', 'accounts');
Session::flash('gaEventAction', 'delete-' . $typeName);
2015-04-28 01:12:12 -05:00
2015-07-10 13:48:45 -05:00
return view('accounts.delete', compact('account', 'subTitle', 'accountList'));
}
/**
2015-12-31 10:20:54 -06:00
* @param ARI $repository
* @param Account $account
*
* @return \Illuminate\Http\RedirectResponse
*/
public function destroy(ARI $repository, Account $account)
{
$type = $account->accountType->type;
2016-04-26 14:40:15 -05:00
$typeName = config('firefly.shortNamesByFullName.' . $type);
$name = $account->name;
2016-03-12 04:05:26 -06:00
$moveTo = $repository->find(intval(Input::get('move_account_before_delete')));
2015-07-10 13:48:45 -05:00
$repository->destroy($account, $moveTo);
2016-03-20 05:38:01 -05:00
Session::flash('success', strval(trans('firefly.' . $typeName . '_deleted', ['name' => $name])));
Preferences::mark();
2016-02-04 00:27:03 -06:00
return redirect(session('accounts.delete.url'));
}
2015-03-01 03:44:10 -06:00
/**
2015-12-31 10:20:54 -06:00
* @param ARI $repository
* @param Account $account
2015-03-01 03:44:10 -06:00
*
* @return \Illuminate\View\View
2015-03-01 03:44:10 -06:00
*/
public function edit(ARI $repository, Account $account)
{
2015-04-03 15:54:21 -05:00
2016-04-26 14:40:15 -05:00
$what = config('firefly.shortNamesByFullName')[$account->accountType->type];
2015-05-21 00:30:38 -05:00
$subTitle = trans('firefly.edit_' . $what . '_account', ['name' => $account->name]);
2016-04-26 14:40:15 -05:00
$subTitleIcon = config('firefly.subIconsByIdentifier.' . $what);
2015-04-03 15:54:21 -05:00
$openingBalance = $repository->openingBalanceTransaction($account);
2015-04-28 01:12:12 -05:00
// put previous url in session if not redirect from store (not "return_to_edit").
2016-02-04 00:27:03 -06:00
if (session('accounts.edit.fromUpdate') !== true) {
2015-04-28 01:12:12 -05:00
Session::put('accounts.edit.url', URL::previous());
}
Session::forget('accounts.edit.fromUpdate');
// pre fill some useful values.
// the opening balance is tricky:
$openingBalanceAmount = null;
2016-02-06 09:22:12 -06:00
if ($openingBalance->id) {
$transaction = $repository->getFirstTransaction($openingBalance, $account);
$openingBalanceAmount = $transaction->amount;
}
$preFilled = [
'accountNumber' => $account->getMeta('accountNumber'),
2015-04-03 15:54:21 -05:00
'accountRole' => $account->getMeta('accountRole'),
'ccType' => $account->getMeta('ccType'),
'ccMonthlyPaymentDate' => $account->getMeta('ccMonthlyPaymentDate'),
'openingBalanceDate' => $openingBalance->id ? $openingBalance->date->format('Y-m-d') : null,
2015-04-03 15:54:21 -05:00
'openingBalance' => $openingBalanceAmount,
2016-01-15 16:12:52 -06:00
'virtualBalance' => round($account->virtual_balance, 2),
];
Session::flash('preFilled', $preFilled);
2015-05-25 01:01:06 -05:00
Session::flash('gaEventCategory', 'accounts');
Session::flash('gaEventAction', 'edit-' . $what);
return view('accounts.edit', compact('account', 'subTitle', 'subTitleIcon', 'openingBalance', 'what'));
}
2015-02-11 00:35:10 -06:00
/**
2015-12-31 10:20:54 -06:00
* @param ARI $repository
2015-05-03 05:54:39 -05:00
* @param $what
2015-02-11 00:35:10 -06:00
*
* @return \Illuminate\View\View
2015-02-11 00:35:10 -06:00
*/
2016-02-05 02:25:15 -06:00
public function index(ARI $repository, string $what)
{
2016-03-19 10:51:52 -05:00
$what = $what ?? 'asset';
2016-03-18 04:53:59 -05:00
2015-05-16 02:41:14 -05:00
$subTitle = trans('firefly.' . $what . '_accounts');
2016-04-26 14:40:15 -05:00
$subTitleIcon = config('firefly.subIconsByIdentifier.' . $what);
$types = config('firefly.accountTypesByIdentifier.' . $what);
2016-05-13 08:53:39 -05:00
$accounts = $repository->getAccountsByType($types);
2016-05-15 08:08:59 -05:00
$start = clone session('start', Carbon::now()->startOfMonth());
$end = clone session('end', Carbon::now()->endOfMonth());
2015-03-04 12:09:57 -06:00
$start->subDay();
2015-07-10 00:39:59 -05:00
2016-01-02 09:57:31 -06:00
$ids = $accounts->pluck('id')->toArray();
2015-07-10 00:39:59 -05:00
$startBalances = Steam::balancesById($ids, $start);
$endBalances = Steam::balancesById($ids, $end);
2015-07-26 12:07:02 -05:00
$activities = Steam::getLastActivities($ids);
2015-07-10 00:39:59 -05:00
2015-04-11 08:01:42 -05:00
$accounts->each(
function (Account $account) use ($activities, $startBalances, $endBalances) {
2016-01-02 09:57:31 -06:00
$account->lastActivityDate = $this->isInArray($activities, $account->id);
$account->startBalance = $this->isInArray($startBalances, $account->id);
$account->endBalance = $this->isInArray($endBalances, $account->id);
2015-03-02 08:27:36 -06:00
}
);
return view('accounts.index', compact('what', 'subTitleIcon', 'subTitle', 'accounts'));
}
2015-02-21 05:16:41 -06:00
/**
2015-12-31 10:20:54 -06:00
* @param ARI $repository
* @param Account $account
2015-02-21 05:16:41 -06:00
*
* @return \Illuminate\View\View
2015-02-21 05:16:41 -06:00
*/
public function show(ARI $repository, Account $account)
2015-02-21 05:16:41 -06:00
{
2016-05-13 08:53:39 -05:00
// show journals from current period only:
2016-05-17 09:00:27 -05:00
$subTitleIcon = config('firefly.subIconsByIdentifier.' . $account->accountType->type);
$subTitle = $account->name;
$range = Preferences::get('viewRange', '1M')->data;
$start = session('start', Navigation::startOfPeriod(new Carbon, $range));
$end = session('end', Navigation::endOfPeriod(new Carbon, $range));
$page = intval(Input::get('page'));
$pageSize = Preferences::get('transactionPageSize', 50)->data;
$offset = ($page - 1) * $pageSize;
$set = $repository->journalsInPeriod(new Collection([$account]), [], $start, $end);
$count = $set->count();
$subSet = $set->splice($offset, $pageSize);
$journals = new LengthAwarePaginator($subSet, $count, $pageSize, $page);
2015-03-29 14:27:51 -05:00
$journals->setPath('accounts/show/' . $account->id);
2016-05-13 08:53:39 -05:00
// grouped other months thing:
// oldest transaction in account:
$start = $repository->firstUseDate($account);
if ($start->year == 1900) {
$start = new Carbon;
}
$range = Preferences::get('viewRange', '1M')->data;
$start = Navigation::startOfPeriod($start, $range);
$end = Navigation::endOfX(new Carbon, $range);
$entries = new Collection;
// chart properties for cache:
$cache = new CacheProperties;
$cache->addProperty($start);
$cache->addProperty($end);
$cache->addProperty('account-show');
$cache->addProperty($account->id);
if ($cache->has()) {
2016-05-20 01:00:35 -05:00
$entries = $cache->get();
return view('accounts.show', compact('account', 'what', 'entries', 'subTitleIcon', 'journals', 'subTitle'));
2016-05-13 08:53:39 -05:00
}
while ($end >= $start) {
$end = Navigation::startOfPeriod($end, $range);
$currentEnd = Navigation::endOfPeriod($end, $range);
2016-05-19 23:58:13 -05:00
$spent = $this->spentInPeriod($account, $end, $currentEnd);
$earned = $this->earnedInPeriod($account, $end, $currentEnd);
2016-05-13 08:53:39 -05:00
$dateStr = $end->format('Y-m-d');
$dateName = Navigation::periodShow($end, $range);
$entries->push([$dateStr, $dateName, $spent, $earned]);
$end = Navigation::subtractPeriod($end, $range, 1);
}
2016-05-13 08:53:39 -05:00
return view('accounts.show', compact('account', 'what', 'entries', 'subTitleIcon', 'journals', 'subTitle'));
2015-02-21 05:16:41 -06:00
}
2016-05-15 08:08:59 -05:00
/**
* @param ARI $repository
* @param Account $account
* @param string $date
*
* @return View
*/
public function showWithDate(ARI $repository, Account $account, string $date)
{
$carbon = new Carbon($date);
$range = Preferences::get('viewRange', '1M')->data;
$start = Navigation::startOfPeriod($carbon, $range);
$end = Navigation::endOfPeriod($carbon, $range);
$subTitle = $account->name;
$page = intval(Input::get('page'));
$pageSize = Preferences::get('transactionPageSize', 50)->data;
$offset = ($page - 1) * $pageSize;
$set = $repository->journalsInPeriod(new Collection([$account]), [], $start, $end);
$count = $set->count();
$subSet = $set->splice($offset, $pageSize);
$journals = new LengthAwarePaginator($subSet, $count, $pageSize, $page);
$journals->setPath('categories/show/' . $account->id . '/' . $date);
return view('accounts.show_with_date', compact('category', 'journals', 'subTitle', 'carbon'));
}
2015-02-11 00:35:10 -06:00
/**
2015-12-31 10:20:54 -06:00
* @param AccountFormRequest $request
* @param ARI $repository
2015-02-11 00:35:10 -06:00
*
* @return \Illuminate\Http\RedirectResponse
*/
public function store(AccountFormRequest $request, ARI $repository)
{
2015-02-09 00:23:39 -06:00
$accountData = [
2015-02-11 00:35:10 -06:00
'name' => $request->input('name'),
2015-02-09 00:56:24 -06:00
'accountType' => $request->input('what'),
2015-07-26 12:07:02 -05:00
'virtualBalance' => round($request->input('virtualBalance'), 2),
'virtualBalanceCurrency' => intval($request->input('amount_currency_id_virtualBalance')),
2015-02-09 00:23:39 -06:00
'active' => true,
'user' => Auth::user()->id,
2015-07-03 05:51:14 -05:00
'iban' => $request->input('iban'),
'accountNumber' => $request->input('accountNumber'),
2015-02-09 00:23:39 -06:00
'accountRole' => $request->input('accountRole'),
2015-07-26 12:07:02 -05:00
'openingBalance' => round($request->input('openingBalance'), 2),
2015-06-06 16:09:12 -05:00
'openingBalanceDate' => new Carbon((string)$request->input('openingBalanceDate')),
'openingBalanceCurrency' => intval($request->input('amount_currency_id_openingBalance')),
2015-02-09 00:23:39 -06:00
];
2015-07-06 11:04:13 -05:00
$account = $repository->store($accountData);
2015-02-09 00:23:39 -06:00
2016-03-20 05:38:01 -05:00
Session::flash('success', strval(trans('firefly.stored_new_account', ['name' => $account->name])));
Preferences::mark();
2015-02-09 00:23:39 -06:00
2016-03-12 00:04:47 -06:00
// update preferences if necessary:
$frontPage = Preferences::get('frontPageAccounts', [])->data;
if (count($frontPage) > 0) {
$frontPage[] = $account->id;
Preferences::set('frontPageAccounts', $frontPage);
}
2015-03-02 13:05:28 -06:00
if (intval(Input::get('create_another')) === 1) {
2015-04-28 01:12:12 -05:00
// set value so create routine will not overwrite URL:
Session::put('accounts.create.fromStore', true);
2015-07-06 09:12:22 -05:00
return redirect(route('accounts.create', [$request->input('what')]))->withInput();
2015-03-02 13:05:28 -06:00
}
2015-04-28 01:12:12 -05:00
// redirect to previous URL.
2016-02-04 00:27:03 -06:00
return redirect(session('accounts.create.url'));
}
/**
2015-12-31 10:20:54 -06:00
* @param AccountFormRequest $request
* @param ARI $repository
* @param Account $account
*
* @return \Illuminate\Http\RedirectResponse
*/
public function update(AccountFormRequest $request, ARI $repository, Account $account)
{
2015-04-28 01:12:12 -05:00
$accountData = [
'name' => $request->input('name'),
'active' => $request->input('active'),
'user' => Auth::user()->id,
2015-07-03 05:51:14 -05:00
'iban' => $request->input('iban'),
'accountNumber' => $request->input('accountNumber'),
'accountRole' => $request->input('accountRole'),
2015-07-26 12:07:02 -05:00
'virtualBalance' => round($request->input('virtualBalance'), 2),
'openingBalance' => round($request->input('openingBalance'), 2),
2015-06-06 16:09:12 -05:00
'openingBalanceDate' => new Carbon((string)$request->input('openingBalanceDate')),
'openingBalanceCurrency' => intval($request->input('amount_currency_id_openingBalance')),
2015-04-03 15:54:21 -05:00
'ccType' => $request->input('ccType'),
'ccMonthlyPaymentDate' => $request->input('ccMonthlyPaymentDate'),
];
$repository->update($account, $accountData);
2016-03-20 05:38:01 -05:00
Session::flash('success', strval(trans('firefly.updated_account', ['name' => $account->name])));
Preferences::mark();
2015-03-02 13:05:28 -06:00
if (intval(Input::get('return_to_edit')) === 1) {
2015-04-28 01:12:12 -05:00
// set value so edit routine will not overwrite URL:
Session::put('accounts.edit.fromUpdate', true);
2015-07-06 09:12:22 -05:00
return redirect(route('accounts.edit', [$account->id]))->withInput(['return_to_edit' => 1]);
2015-03-02 13:05:28 -06:00
}
2015-04-28 01:12:12 -05:00
// redirect to previous URL.
2016-02-04 00:27:03 -06:00
return redirect(session('accounts.edit.url'));
}
2016-01-02 09:57:31 -06:00
/**
* @param array $array
2016-02-05 02:25:15 -06:00
* @param int $entryId
2016-01-02 09:57:31 -06:00
*
* @return null|mixed
*/
2016-02-05 02:25:15 -06:00
protected function isInArray(array $array, int $entryId)
2016-01-02 09:57:31 -06:00
{
if (isset($array[$entryId])) {
return $array[$entryId];
}
return '';
2016-01-02 09:57:31 -06:00
}
2016-05-19 23:58:13 -05:00
/**
* Asset accounts actually earn money by being the destination of a deposit or the destination
* of a transfer. The money moves to them.
*
* A revenue account doesn't really earn money itself. Money is earned "from" the revenue account.
* So, the call to find out how many money has been earned by/from a revenue account is slightly different.
*
*
*
* @param Account $account
* @param Carbon $start
* @param Carbon $end
*
* @return string
*/
private function earnedInPeriod(Account $account, Carbon $start, Carbon $end)
{
/** @var ARI $repository */
$repository = app(ARI::class);
$collection = new Collection([$account]);
$type = $account->accountType->type;
switch ($type) {
case AccountType::DEFAULT:
case AccountType::ASSET:
return $repository->earnedInPeriod($collection, $start, $end);
case AccountType::REVENUE:
return $repository->earnedFromInPeriod($collection, $start, $end);
default:
return '0';
}
}
/**
* Asset accounts actually spend money by being the source of a withdrawal or the source
* of a transfer. The money moves away from them.
*
* An expense account doesn't really spend money itself. Money is spent "at" the expense account.
* So, the call to find out how many money has been spent on/at an expense account is slightly different.
*
*
*
* @param Account $account
* @param Carbon $start
* @param Carbon $end
*
* @return string
*/
private function spentInPeriod(Account $account, Carbon $start, Carbon $end): string
{
/** @var ARI $repository */
$repository = app(ARI::class);
$collection = new Collection([$account]);
$type = $account->accountType->type;
switch ($type) {
case AccountType::DEFAULT:
case AccountType::ASSET:
return $repository->spentInPeriod($collection, $start, $end);
case AccountType::EXPENSE:
return $repository->spentAtInPeriod($collection, $start, $end);
default:
return '0';
}
}
}