middleware( function ($request, $next) { View::share('title', trans('firefly.reports')); View::share('mainTitleIcon', 'fa-line-chart'); $this->helper = app(ReportHelperInterface::class); return $next($request); } ); } /** * @param AccountRepositoryInterface $repository * * @return View */ public function index(AccountRepositoryInterface $repository) { /** @var Carbon $start */ $start = clone session('first'); $months = $this->helper->listOfMonths($start); $customFiscalYear = Preferences::get('customFiscalYear', 0)->data; // does the user have shared accounts? $accounts = $repository->getAccountsByType([AccountType::DEFAULT, AccountType::ASSET]); // 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')); } /** * @param string $reportType * @param Carbon $start * @param Carbon $end * @param Collection $accounts * * @return View * @throws FireflyException */ public function report(string $reportType, Carbon $start, Carbon $end, Collection $accounts) { // throw an error if necessary. if ($end < $start) { throw new FireflyException('End date cannot be before start date, silly!'); } // 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'); switch ($reportType) { default: throw new FireflyException('Unfortunately, reports of the type "' . e($reportType) . '" are not available at this time.'); 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 return $this->defaultMonth($reportType, $start, $end, $accounts); case 'audit': // always default return $this->auditReport($start, $end, $accounts); } } /** * @param Carbon $start * @param Carbon $end * @param Collection $accounts * * @return View */ private function auditReport(Carbon $start, Carbon $end, Collection $accounts) { $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); $collector = new JournalCollector(auth()->user()); $collector->setAccounts(new Collection([$account]))->setRange($start, $end); $journals = $collector->getJournals(); $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; } /* * Reverse set again. */ $auditData[$id]['journals'] = $journals->reverse(); $auditData[$id]['exists'] = $journals->count() > 0; $auditData[$id]['end'] = $end->formatLocalized(strval(trans('config.month_and_day'))); $auditData[$id]['endBalance'] = Steam::balance($account, $end); $auditData[$id]['dayBefore'] = $dayBefore->formatLocalized(strval(trans('config.month_and_day'))); $auditData[$id]['dayBeforeBalance'] = $dayBeforeBalance; } $reportType = 'audit'; $accountIds = join(',', $accounts->pluck('id')->toArray()); $hideable = ['buttons', 'icon', 'description', 'balance_before', 'amount', 'balance_after', 'date', 'interest_date', 'book_date', 'process_date', // 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', ]; $defaultShow = ['icon', 'description', 'balance_before', 'amount', 'balance_after', 'date', 'to']; return view('reports.audit.report', compact('start', 'end', 'reportType', 'accountIds', 'accounts', 'auditData', 'hideable', 'defaultShow')); } /** * @param $reportType * @param Carbon $start * @param Carbon $end * @param Collection $accounts * * @return View */ private function defaultMonth(string $reportType, Carbon $start, Carbon $end, Collection $accounts) { $bills = $this->helper->getBillReport($start, $end, $accounts); $tags = $this->helper->tagReport($start, $end, $accounts); // and some id's, joined: $accountIds = join(',', $accounts->pluck('id')->toArray()); // continue! return view( 'reports.default.month', compact( 'start', 'end', 'tags', 'bills', 'accountIds', 'reportType' ) ); } /** * @param $reportType * @param $start * @param $end * @param $accounts * * @return View */ private function defaultMultiYear(string $reportType, Carbon $start, Carbon $end, Collection $accounts) { // need all budgets // need all years. // and some id's, joined: $accountIds = []; /** @var Account $account */ foreach ($accounts as $account) { $accountIds[] = $account->id; } $accountIds = join(',', $accountIds); return view( 'reports.default.multi-year', compact( 'accounts', 'start', 'end', 'accountIds', 'reportType' ) ); } /** * @param $reportType * @param Carbon $start * @param Carbon $end * @param Collection $accounts * * @return View */ private function defaultYear(string $reportType, Carbon $start, Carbon $end, Collection $accounts) { 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', 'reportType', 'accountIds', 'end' ) ); } }