mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-02-03 04:00:40 -06:00
408 lines
14 KiB
PHP
408 lines
14 KiB
PHP
<?php
|
|
/**
|
|
* BillController.php
|
|
* Copyright (c) 2017 thegrumpydictator@gmail.com
|
|
*
|
|
* This file is part of Firefly III.
|
|
*
|
|
* Firefly III is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Firefly III 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Firefly III. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
declare(strict_types=1);
|
|
|
|
namespace FireflyIII\Http\Controllers;
|
|
|
|
use Carbon\Carbon;
|
|
use FireflyIII\Helpers\Attachments\AttachmentHelperInterface;
|
|
use FireflyIII\Helpers\Collector\JournalCollectorInterface;
|
|
use FireflyIII\Http\Requests\BillFormRequest;
|
|
use FireflyIII\Models\Bill;
|
|
use FireflyIII\Repositories\Bill\BillRepositoryInterface;
|
|
use FireflyIII\Repositories\RuleGroup\RuleGroupRepositoryInterface;
|
|
use FireflyIII\TransactionRules\TransactionMatcher;
|
|
use FireflyIII\Transformers\BillTransformer;
|
|
use Illuminate\Http\RedirectResponse;
|
|
use Illuminate\Http\Request;
|
|
use Illuminate\Support\Collection;
|
|
use League\Fractal\Manager;
|
|
use League\Fractal\Resource\Item;
|
|
use League\Fractal\Serializer\DataArraySerializer;
|
|
use Symfony\Component\HttpFoundation\ParameterBag;
|
|
use URL;
|
|
|
|
/**
|
|
* Class BillController.
|
|
*
|
|
* @SuppressWarnings(PHPMD.CouplingBetweenObjects)
|
|
*/
|
|
class BillController extends Controller
|
|
{
|
|
/** @var AttachmentHelperInterface Helper for attachments. */
|
|
private $attachments;
|
|
/** @var BillRepositoryInterface Bill repository */
|
|
private $billRepository;
|
|
/** @var RuleGroupRepositoryInterface Rule group repository */
|
|
private $ruleGroupRepos;
|
|
|
|
/**
|
|
* BillController constructor.
|
|
*/
|
|
public function __construct()
|
|
{
|
|
parent::__construct();
|
|
|
|
$maxFileSize = app('steam')->phpBytes(ini_get('upload_max_filesize'));
|
|
$maxPostSize = app('steam')->phpBytes(ini_get('post_max_size'));
|
|
$uploadSize = min($maxFileSize, $maxPostSize);
|
|
app('view')->share('uploadSize', $uploadSize);
|
|
|
|
$this->middleware(
|
|
function ($request, $next) {
|
|
app('view')->share('title', (string)trans('firefly.bills'));
|
|
app('view')->share('mainTitleIcon', 'fa-calendar-o');
|
|
$this->attachments = app(AttachmentHelperInterface::class);
|
|
$this->billRepository = app(BillRepositoryInterface::class);
|
|
$this->ruleGroupRepos = app(RuleGroupRepositoryInterface::class);
|
|
|
|
return $next($request);
|
|
}
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Create a new bill.
|
|
*
|
|
* @param Request $request
|
|
*
|
|
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
|
|
*/
|
|
public function create(Request $request)
|
|
{
|
|
$periods = [];
|
|
/** @var array $billPeriods */
|
|
$billPeriods = config('firefly.bill_periods');
|
|
foreach ($billPeriods as $current) {
|
|
$periods[$current] = strtolower((string)trans('firefly.repeat_freq_' . $current));
|
|
}
|
|
$subTitle = (string)trans('firefly.create_new_bill');
|
|
$defaultCurrency = app('amount')->getDefaultCurrency();
|
|
|
|
// put previous url in session if not redirect from store (not "create another").
|
|
if (true !== session('bills.create.fromStore')) {
|
|
$this->rememberPreviousUri('bills.create.uri');
|
|
}
|
|
$request->session()->forget('bills.create.fromStore');
|
|
|
|
return view('bills.create', compact('periods', 'subTitle', 'defaultCurrency'));
|
|
}
|
|
|
|
/**
|
|
* Delete a bill.
|
|
*
|
|
* @param Bill $bill
|
|
*
|
|
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
|
|
*/
|
|
public function delete(Bill $bill)
|
|
{
|
|
// put previous url in session
|
|
$this->rememberPreviousUri('bills.delete.uri');
|
|
$subTitle = (string)trans('firefly.delete_bill', ['name' => $bill->name]);
|
|
|
|
return view('bills.delete', compact('bill', 'subTitle'));
|
|
}
|
|
|
|
/**
|
|
* Destroy a bill.
|
|
*
|
|
* @param Request $request
|
|
* @param Bill $bill
|
|
*
|
|
* @return RedirectResponse|\Illuminate\Routing\Redirector
|
|
*/
|
|
public function destroy(Request $request, Bill $bill)
|
|
{
|
|
$name = $bill->name;
|
|
$this->billRepository->destroy($bill);
|
|
|
|
$request->session()->flash('success', (string)trans('firefly.deleted_bill', ['name' => $name]));
|
|
app('preferences')->mark();
|
|
|
|
return redirect($this->getPreviousUri('bills.delete.uri'));
|
|
}
|
|
|
|
/**
|
|
* Edit a bill.
|
|
*
|
|
* @param Request $request
|
|
* @param Bill $bill
|
|
*
|
|
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
|
|
*/
|
|
public function edit(Request $request, Bill $bill)
|
|
{
|
|
$periods = [];
|
|
/** @var array $billPeriods */
|
|
$billPeriods = config('firefly.bill_periods');
|
|
|
|
foreach ($billPeriods as $current) {
|
|
$periods[$current] = (string)trans('firefly.' . $current);
|
|
}
|
|
|
|
$subTitle = (string)trans('firefly.edit_bill', ['name' => $bill->name]);
|
|
|
|
// put previous url in session if not redirect from store (not "return_to_edit").
|
|
if (true !== session('bills.edit.fromUpdate')) {
|
|
$this->rememberPreviousUri('bills.edit.uri');
|
|
}
|
|
|
|
$currency = app('amount')->getDefaultCurrency();
|
|
$bill->amount_min = round((float)$bill->amount_min, $currency->decimal_places);
|
|
$bill->amount_max = round((float)$bill->amount_max, $currency->decimal_places);
|
|
$defaultCurrency = app('amount')->getDefaultCurrency();
|
|
|
|
// code to handle active-checkboxes
|
|
$hasOldInput = null !== $request->old('_token');
|
|
|
|
$preFilled = [
|
|
'notes' => $this->billRepository->getNoteText($bill),
|
|
'transaction_currency_id' => $bill->transaction_currency_id,
|
|
'active' => $hasOldInput ? (bool)$request->old('active') : $bill->active,
|
|
];
|
|
|
|
$request->session()->flash('preFilled', $preFilled);
|
|
$request->session()->forget('bills.edit.fromUpdate');
|
|
|
|
return view('bills.edit', compact('subTitle', 'periods', 'bill', 'defaultCurrency', 'preFilled'));
|
|
}
|
|
|
|
/**
|
|
* Show all bills.
|
|
*
|
|
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
|
|
*/
|
|
public function index()
|
|
{
|
|
$start = session('start');
|
|
$end = session('end');
|
|
$pageSize = (int)app('preferences')->get('listPageSize', 50)->data;
|
|
$paginator = $this->billRepository->getPaginator($pageSize);
|
|
$parameters = new ParameterBag();
|
|
$parameters->set('start', $start);
|
|
$parameters->set('end', $end);
|
|
$transformer = new BillTransformer($parameters);
|
|
/** @var Collection $bills */
|
|
$bills = $paginator->getCollection()->map(
|
|
function (Bill $bill) use ($transformer) {
|
|
return $transformer->transform($bill);
|
|
}
|
|
);
|
|
$bills = $bills->sortBy(
|
|
function (array $bill) {
|
|
return (int)!$bill['active'] . strtolower($bill['name']);
|
|
}
|
|
);
|
|
|
|
// add info about rules:
|
|
$rules = $this->billRepository->getRulesForBills($paginator->getCollection());
|
|
$bills = $bills->map(
|
|
function (array $bill) use ($rules) {
|
|
$bill['rules'] = $rules[$bill['id']] ?? [];
|
|
|
|
return $bill;
|
|
}
|
|
);
|
|
|
|
$paginator->setPath(route('bills.index'));
|
|
|
|
return view('bills.index', compact('bills', 'paginator'));
|
|
}
|
|
|
|
/**
|
|
* Rescan bills for transactions.
|
|
*
|
|
* @param Request $request
|
|
* @param Bill $bill
|
|
*
|
|
* @return RedirectResponse|\Illuminate\Routing\Redirector
|
|
* @throws \FireflyIII\Exceptions\FireflyException
|
|
*/
|
|
public function rescan(Request $request, Bill $bill)
|
|
{
|
|
if (false === $bill->active) {
|
|
$request->session()->flash('warning', (string)trans('firefly.cannot_scan_inactive_bill'));
|
|
}
|
|
if (true === $bill->active) {
|
|
$set = $this->billRepository->getRulesForBill($bill);
|
|
$total = 0;
|
|
foreach ($set as $rule) {
|
|
// simply fire off all rules?
|
|
/** @var TransactionMatcher $matcher */
|
|
$matcher = app(TransactionMatcher::class);
|
|
$matcher->setLimit(100000); // large upper limit
|
|
$matcher->setRange(100000); // large upper limit
|
|
$matcher->setRule($rule);
|
|
$matchingTransactions = $matcher->findTransactionsByRule();
|
|
$total += $matchingTransactions->count();
|
|
$this->billRepository->linkCollectionToBill($bill, $matchingTransactions);
|
|
}
|
|
|
|
|
|
$request->session()->flash('success', (string)trans('firefly.rescanned_bill', ['total' => $total]));
|
|
app('preferences')->mark();
|
|
}
|
|
|
|
return redirect(URL::previous());
|
|
}
|
|
|
|
/**
|
|
* Show a bill.
|
|
*
|
|
* @param Request $request
|
|
* @param Bill $bill
|
|
*
|
|
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
|
|
*/
|
|
public function show(Request $request, Bill $bill)
|
|
{
|
|
// add info about rules:
|
|
$rules = $this->billRepository->getRulesForBill($bill);
|
|
$subTitle = $bill->name;
|
|
/** @var Carbon $start */
|
|
$start = session('start');
|
|
/** @var Carbon $end */
|
|
$end = session('end');
|
|
$year = $start->year;
|
|
$page = (int)$request->get('page');
|
|
$pageSize = (int)app('preferences')->get('listPageSize', 50)->data;
|
|
$yearAverage = $this->billRepository->getYearAverage($bill, $start);
|
|
$overallAverage = $this->billRepository->getOverallAverage($bill);
|
|
$manager = new Manager();
|
|
$manager->setSerializer(new DataArraySerializer());
|
|
$manager->parseIncludes(['attachments', 'notes']);
|
|
|
|
// Make a resource out of the data and
|
|
$parameters = new ParameterBag();
|
|
$parameters->set('start', $start);
|
|
$parameters->set('end', $end);
|
|
$resource = new Item($bill, new BillTransformer($parameters), 'bill');
|
|
$object = $manager->createData($resource)->toArray();
|
|
|
|
// use collector:
|
|
/** @var JournalCollectorInterface $collector */
|
|
$collector = app(JournalCollectorInterface::class);
|
|
$collector->setAllAssetAccounts()->setBills(new Collection([$bill]))->setLimit($pageSize)->setPage($page)->withBudgetInformation()
|
|
->withCategoryInformation();
|
|
$transactions = $collector->getPaginatedJournals();
|
|
$transactions->setPath(route('bills.show', [$bill->id]));
|
|
|
|
|
|
return view('bills.show', compact('transactions', 'rules', 'yearAverage', 'overallAverage', 'year', 'object', 'bill', 'subTitle'));
|
|
}
|
|
|
|
|
|
/**
|
|
* Store a new bill.
|
|
*
|
|
* @param BillFormRequest $request
|
|
*
|
|
* @return RedirectResponse
|
|
*
|
|
* @SuppressWarnings(PHPMD.CyclomaticComplexity)
|
|
* @SuppressWarnings(PHPMD.ExcessiveMethodLength)
|
|
*/
|
|
public function store(BillFormRequest $request): RedirectResponse
|
|
{
|
|
$billData = $request->getBillData();
|
|
$billData['active'] = true;
|
|
$bill = $this->billRepository->store($billData);
|
|
if (null === $bill) {
|
|
$request->session()->flash('error', (string)trans('firefly.bill_store_error'));
|
|
|
|
return redirect(route('bills.create'))->withInput();
|
|
}
|
|
$request->session()->flash('success', (string)trans('firefly.stored_new_bill', ['name' => $bill->name]));
|
|
app('preferences')->mark();
|
|
|
|
/** @var array $files */
|
|
$files = $request->hasFile('attachments') ? $request->file('attachments') : null;
|
|
$this->attachments->saveAttachmentsForModel($bill, $files);
|
|
|
|
if (\count($this->attachments->getMessages()->get('attachments')) > 0) {
|
|
$request->session()->flash('info', $this->attachments->getMessages()->get('attachments')); // @codeCoverageIgnore
|
|
}
|
|
|
|
// do return to original bill form?
|
|
$return = 'false';
|
|
if (1 === (int)$request->get('create_another')) {
|
|
$return = 'true';
|
|
}
|
|
|
|
$group = null;
|
|
// find first rule group, or create one:
|
|
$count = $this->ruleGroupRepos->count();
|
|
if (0 === $count) {
|
|
$data = [
|
|
'title' => (string)trans('firefly.rulegroup_for_bills_title'),
|
|
'description' => (string)trans('firefly.rulegroup_for_bills_description'),
|
|
];
|
|
$group = $this->ruleGroupRepos->store($data);
|
|
}
|
|
if ($count > 0) {
|
|
$group = $this->ruleGroupRepos->getActiveGroups($bill->user)->first();
|
|
}
|
|
|
|
// redirect to page that will create a new rule.
|
|
$params = http_build_query(['fromBill' => $bill->id, 'return' => $return]);
|
|
|
|
return redirect(route('rules.create', [$group->id]) . '?' . $params);
|
|
}
|
|
|
|
/**
|
|
* Update a bill.
|
|
*
|
|
* @param BillFormRequest $request
|
|
* @param Bill $bill
|
|
*
|
|
* @return RedirectResponse
|
|
*/
|
|
public function update(BillFormRequest $request, Bill $bill): RedirectResponse
|
|
{
|
|
$billData = $request->getBillData();
|
|
$bill = $this->billRepository->update($bill, $billData);
|
|
|
|
$request->session()->flash('success', (string)trans('firefly.updated_bill', ['name' => $bill->name]));
|
|
app('preferences')->mark();
|
|
|
|
/** @var array $files */
|
|
$files = $request->hasFile('attachments') ? $request->file('attachments') : null;
|
|
$this->attachments->saveAttachmentsForModel($bill, $files);
|
|
|
|
// flash messages
|
|
if (\count($this->attachments->getMessages()->get('attachments')) > 0) {
|
|
$request->session()->flash('info', $this->attachments->getMessages()->get('attachments')); // @codeCoverageIgnore
|
|
}
|
|
$redirect = redirect($this->getPreviousUri('bills.edit.uri'));
|
|
|
|
if (1 === (int)$request->get('return_to_edit')) {
|
|
// @codeCoverageIgnoreStart
|
|
$request->session()->put('bills.edit.fromUpdate', true);
|
|
|
|
$redirect = redirect(route('bills.edit', [$bill->id]))->withInput(['return_to_edit' => 1]);
|
|
// @codeCoverageIgnoreEnd
|
|
}
|
|
|
|
return $redirect;
|
|
}
|
|
}
|