mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-09-24 22:48:18 +00:00
Various updates to display native/foreign amounts.
This commit is contained in:
@@ -40,6 +40,7 @@ use FireflyIII\Repositories\Budget\OperationsRepositoryInterface;
|
|||||||
use FireflyIII\Repositories\UserGroups\Currency\CurrencyRepositoryInterface;
|
use FireflyIII\Repositories\UserGroups\Currency\CurrencyRepositoryInterface;
|
||||||
use FireflyIII\User;
|
use FireflyIII\User;
|
||||||
use Illuminate\Http\JsonResponse;
|
use Illuminate\Http\JsonResponse;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class BasicController
|
* Class BasicController
|
||||||
@@ -101,10 +102,10 @@ class BasicController extends Controller
|
|||||||
$billData = $this->getBillInformation($start, $end);
|
$billData = $this->getBillInformation($start, $end);
|
||||||
$spentData = $this->getLeftToSpendInfo($start, $end);
|
$spentData = $this->getLeftToSpendInfo($start, $end);
|
||||||
$netWorthData = $this->getNetWorthInfo($start, $end);
|
$netWorthData = $this->getNetWorthInfo($start, $end);
|
||||||
// $balanceData = [];
|
$balanceData = [];
|
||||||
// $billData = [];
|
$billData = [];
|
||||||
// $spentData = [];
|
// $spentData = [];
|
||||||
// $netWorthData = [];
|
$netWorthData = [];
|
||||||
$total = array_merge($balanceData, $billData, $spentData, $netWorthData);
|
$total = array_merge($balanceData, $billData, $spentData, $netWorthData);
|
||||||
|
|
||||||
// give new keys
|
// give new keys
|
||||||
@@ -274,19 +275,22 @@ class BasicController extends Controller
|
|||||||
$available = $this->abRepository->getAvailableBudgetWithCurrency($start, $end);
|
$available = $this->abRepository->getAvailableBudgetWithCurrency($start, $end);
|
||||||
$budgets = $this->budgetRepository->getActiveBudgets();
|
$budgets = $this->budgetRepository->getActiveBudgets();
|
||||||
$spent = $this->opsRepository->sumExpenses($start, $end, null, $budgets);
|
$spent = $this->opsRepository->sumExpenses($start, $end, null, $budgets);
|
||||||
|
$days = (int) $today->diffInDays($end, true) + 1;
|
||||||
|
Log::debug(sprintf('Now in getLeftToSpendInfo("%s", "%s")', $start->format('Y-m-d H:i:s'), $end->format('Y-m-d H:i:s')));
|
||||||
|
|
||||||
foreach ($spent as $row) {
|
foreach ($spent as $row) {
|
||||||
// either an amount was budgeted or 0 is available.
|
// either an amount was budgeted or 0 is available.
|
||||||
$amount = (string) ($available[$row['currency_id']] ?? '0');
|
$currencyId = $row['currency_id'];
|
||||||
|
$amount = (string) ($available[$currencyId] ?? '0');
|
||||||
$spentInCurrency = $row['sum'];
|
$spentInCurrency = $row['sum'];
|
||||||
$leftToSpend = bcadd($amount, $spentInCurrency);
|
$leftToSpend = bcadd($amount, $spentInCurrency);
|
||||||
|
|
||||||
$days = (int) $today->diffInDays($end, true) + 1;
|
|
||||||
$perDay = '0';
|
$perDay = '0';
|
||||||
if (0 !== $days && bccomp($leftToSpend, '0') > -1) {
|
if (0 !== $days && bccomp($leftToSpend, '0') > -1) {
|
||||||
$perDay = bcdiv($leftToSpend, (string) $days);
|
$perDay = bcdiv($leftToSpend, (string) $days);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Log::debug(sprintf('Spent %s %s', $row['currency_code'], $row['sum']));
|
||||||
|
|
||||||
$return[] = [
|
$return[] = [
|
||||||
'key' => sprintf('left-to-spend-in-%s', $row['currency_code']),
|
'key' => sprintf('left-to-spend-in-%s', $row['currency_code']),
|
||||||
'title' => trans('firefly.box_left_to_spend_in_currency', ['currency' => $row['currency_symbol']]),
|
'title' => trans('firefly.box_left_to_spend_in_currency', ['currency' => $row['currency_symbol']]),
|
||||||
|
@@ -210,11 +210,19 @@ class RecalculateNativeAmounts extends Command
|
|||||||
$set = DB::table('transactions')
|
$set = DB::table('transactions')
|
||||||
->join('transaction_journals', 'transaction_journals.id', '=', 'transactions.transaction_journal_id')
|
->join('transaction_journals', 'transaction_journals.id', '=', 'transactions.transaction_journal_id')
|
||||||
->where('transaction_journals.user_group_id', $userGroup->id)
|
->where('transaction_journals.user_group_id', $userGroup->id)
|
||||||
->where(static function (DatabaseBuilder $q) use ($currency): void {
|
|
||||||
$q->whereNot('transactions.transaction_currency_id', $currency->id)
|
->where(function(DatabaseBuilder $q1) use ($currency) {
|
||||||
->orWhereNot('transactions.foreign_currency_id', $currency->id)
|
$q1->where(function(DatabaseBuilder $q2) use ($currency) {
|
||||||
;
|
$q2->whereNot('transactions.transaction_currency_id', $currency->id)->whereNull('transactions.foreign_currency_id');
|
||||||
|
})->orWhere(function(DatabaseBuilder $q3) use ($currency) {
|
||||||
|
$q3->whereNot('transactions.transaction_currency_id', $currency->id)->whereNot('transactions.foreign_currency_id', $currency->id);
|
||||||
|
});
|
||||||
})
|
})
|
||||||
|
// ->where(static function (DatabaseBuilder $q) use ($currency): void {
|
||||||
|
// $q->whereNot('transactions.transaction_currency_id', $currency->id)
|
||||||
|
// ->whereNot('transactions.foreign_currency_id', $currency->id)
|
||||||
|
// ;
|
||||||
|
// })
|
||||||
->get(['transactions.id'])
|
->get(['transactions.id'])
|
||||||
;
|
;
|
||||||
TransactionObserver::$recalculate = false;
|
TransactionObserver::$recalculate = false;
|
||||||
|
@@ -71,7 +71,7 @@ class TransactionObserver
|
|||||||
$transaction->native_amount = null;
|
$transaction->native_amount = null;
|
||||||
$transaction->native_foreign_amount = null;
|
$transaction->native_foreign_amount = null;
|
||||||
// first normal amount
|
// first normal amount
|
||||||
if ($transaction->transactionCurrency->id !== $userCurrency->id) {
|
if ($transaction->transactionCurrency->id !== $userCurrency->id && (null === $transaction->foreign_currency_id || (null !== $transaction->foreign_currency_id && $transaction->foreign_currency_id !== $userCurrency->id))) {
|
||||||
$converter = new ExchangeRateConverter();
|
$converter = new ExchangeRateConverter();
|
||||||
$converter->setIgnoreSettings(true);
|
$converter->setIgnoreSettings(true);
|
||||||
$transaction->native_amount = $converter->convert($transaction->transactionCurrency, $userCurrency, $transaction->transactionJournal->date, $transaction->amount);
|
$transaction->native_amount = $converter->convert($transaction->transactionCurrency, $userCurrency, $transaction->transactionJournal->date, $transaction->amount);
|
||||||
|
@@ -47,108 +47,12 @@ class BoxController extends Controller
|
|||||||
use DateCalculation;
|
use DateCalculation;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This box has three types of info to display:
|
* Deprecated method, no longer in use.
|
||||||
* 0) If the user has available amount this period and has overspent: overspent box.
|
* @deprecated
|
||||||
* 1) If the user has available amount this period and has NOT overspent: left to spend box.
|
|
||||||
* 2) if the user has no available amount set this period: spent per day
|
|
||||||
*
|
|
||||||
* @SuppressWarnings(PHPMD.ExcessiveMethodLength)
|
|
||||||
*/
|
*/
|
||||||
public function available(): JsonResponse
|
public function available(): JsonResponse
|
||||||
{
|
{
|
||||||
app('log')->debug('Now in available()');
|
return response()->json([]);
|
||||||
|
|
||||||
/** @var OperationsRepositoryInterface $opsRepository */
|
|
||||||
$opsRepository = app(OperationsRepositoryInterface::class);
|
|
||||||
|
|
||||||
/** @var AvailableBudgetRepositoryInterface $abRepository */
|
|
||||||
$abRepository = app(AvailableBudgetRepositoryInterface::class);
|
|
||||||
$abRepository->cleanup();
|
|
||||||
|
|
||||||
/** @var Carbon $start */
|
|
||||||
$start = session('start', today(config('app.timezone'))->startOfMonth());
|
|
||||||
|
|
||||||
/** @var Carbon $end */
|
|
||||||
$end = session('end', today(config('app.timezone'))->endOfMonth());
|
|
||||||
$today = today(config('app.timezone'));
|
|
||||||
$display = 2; // see method docs.
|
|
||||||
$boxTitle = (string) trans('firefly.spent');
|
|
||||||
|
|
||||||
$cache = new CacheProperties();
|
|
||||||
$cache->addProperty($start);
|
|
||||||
$cache->addProperty($end);
|
|
||||||
$cache->addProperty($today);
|
|
||||||
$cache->addProperty('box-available');
|
|
||||||
if ($cache->has()) {
|
|
||||||
return response()->json($cache->get());
|
|
||||||
}
|
|
||||||
$leftPerDayAmount = '0';
|
|
||||||
$leftToSpendAmount = '0';
|
|
||||||
|
|
||||||
$currency = app('amount')->getDefaultCurrency();
|
|
||||||
app('log')->debug(sprintf('Default currency is %s', $currency->code));
|
|
||||||
$availableBudgets = $abRepository->getAvailableBudgetsByExactDate($start, $end);
|
|
||||||
app('log')->debug(sprintf('Found %d available budget(s)', $availableBudgets->count()));
|
|
||||||
$availableBudgets = $availableBudgets->filter(
|
|
||||||
static function (AvailableBudget $availableBudget) use ($currency) { // @phpstan-ignore-line
|
|
||||||
if ($availableBudget->transaction_currency_id === $currency->id) {
|
|
||||||
app('log')->debug(sprintf(
|
|
||||||
'Will include AB #%d: from %s-%s amount %s',
|
|
||||||
$availableBudget->id,
|
|
||||||
$availableBudget->start_date->format('Y-m-d'),
|
|
||||||
$availableBudget->end_date->format('Y-m-d'),
|
|
||||||
$availableBudget->amount
|
|
||||||
));
|
|
||||||
|
|
||||||
return $availableBudget;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
app('log')->debug(sprintf('Filtered back to %d available budgets', $availableBudgets->count()));
|
|
||||||
// spent in this period, in budgets, for default currency.
|
|
||||||
// also calculate spent per day.
|
|
||||||
$spent = $opsRepository->sumExpenses($start, $end, null, null, $currency);
|
|
||||||
$spentAmount = $spent[$currency->id]['sum'] ?? '0';
|
|
||||||
app('log')->debug(sprintf('Spent for default currency for all budgets in this period: %s', $spentAmount));
|
|
||||||
|
|
||||||
$days = (int) ($today->between($start, $end) ? $today->diffInDays($start, true) + 1 : $end->diffInDays($start, true) + 1);
|
|
||||||
app('log')->debug(sprintf('Number of days left: %d', $days));
|
|
||||||
$spentPerDay = bcdiv($spentAmount, (string) $days);
|
|
||||||
app('log')->debug(sprintf('Available to spend per day: %s', $spentPerDay));
|
|
||||||
if ($availableBudgets->count() > 0) {
|
|
||||||
$display = 0; // assume user overspent
|
|
||||||
$boxTitle = (string) trans('firefly.overspent');
|
|
||||||
$totalAvailableSum = (string) $availableBudgets->sum('amount');
|
|
||||||
app('log')->debug(sprintf('Total available sum is %s', $totalAvailableSum));
|
|
||||||
// calculate with available budget.
|
|
||||||
$leftToSpendAmount = bcadd($totalAvailableSum, $spentAmount);
|
|
||||||
app('log')->debug(sprintf('So left to spend is %s', $leftToSpendAmount));
|
|
||||||
if (bccomp($leftToSpendAmount, '0') >= 0) {
|
|
||||||
app('log')->debug('Left to spend is positive or zero!');
|
|
||||||
$boxTitle = (string) trans('firefly.left_to_spend');
|
|
||||||
$activeDaysLeft = $this->activeDaysLeft($start, $end); // see method description.
|
|
||||||
$display = 1; // not overspent
|
|
||||||
$leftPerDayAmount = 0 === $activeDaysLeft ? $leftToSpendAmount : bcdiv($leftToSpendAmount, (string) $activeDaysLeft);
|
|
||||||
app('log')->debug(sprintf('Left to spend per day is %s', $leftPerDayAmount));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$return = [
|
|
||||||
'display' => $display,
|
|
||||||
'spent_total' => app('amount')->formatAnything($currency, $spentAmount, false),
|
|
||||||
'spent_per_day' => app('amount')->formatAnything($currency, $spentPerDay, false),
|
|
||||||
'left_to_spend' => app('amount')->formatAnything($currency, $leftToSpendAmount, false),
|
|
||||||
'left_per_day' => app('amount')->formatAnything($currency, $leftPerDayAmount, false),
|
|
||||||
'title' => $boxTitle,
|
|
||||||
];
|
|
||||||
app('log')->debug('Final output', $return);
|
|
||||||
|
|
||||||
$cache->store($return);
|
|
||||||
app('log')->debug('Now done with available()');
|
|
||||||
|
|
||||||
return response()->json($return);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -162,9 +66,11 @@ class BoxController extends Controller
|
|||||||
|
|
||||||
/** @var Carbon $end */
|
/** @var Carbon $end */
|
||||||
$end = session('end', today(config('app.timezone'))->endOfMonth());
|
$end = session('end', today(config('app.timezone'))->endOfMonth());
|
||||||
|
$convertToNative = app('preferences')->get('convert_to_native', false)->data;
|
||||||
$cache = new CacheProperties();
|
$cache = new CacheProperties();
|
||||||
$cache->addProperty($start);
|
$cache->addProperty($start);
|
||||||
$cache->addProperty($end);
|
$cache->addProperty($end);
|
||||||
|
$cache->addProperty($convertToNative);
|
||||||
$cache->addProperty('box-balance');
|
$cache->addProperty('box-balance');
|
||||||
if ($cache->has()) {
|
if ($cache->has()) {
|
||||||
return response()->json($cache->get());
|
return response()->json($cache->get());
|
||||||
@@ -175,18 +81,19 @@ class BoxController extends Controller
|
|||||||
$sums = [];
|
$sums = [];
|
||||||
$currency = app('amount')->getDefaultCurrency();
|
$currency = app('amount')->getDefaultCurrency();
|
||||||
|
|
||||||
|
|
||||||
// collect income of user:
|
// collect income of user:
|
||||||
/** @var GroupCollectorInterface $collector */
|
/** @var GroupCollectorInterface $collector */
|
||||||
$collector = app(GroupCollectorInterface::class);
|
$collector = app(GroupCollectorInterface::class);
|
||||||
$collector->setRange($start, $end)
|
$collector->setRange($start, $end)
|
||||||
->setTypes([TransactionType::DEPOSIT])
|
->setTypes([TransactionType::DEPOSIT]);
|
||||||
;
|
|
||||||
$set = $collector->getExtractedJournals();
|
$set = $collector->getExtractedJournals();
|
||||||
|
|
||||||
/** @var array $journal */
|
/** @var array $journal */
|
||||||
foreach ($set as $journal) {
|
foreach ($set as $journal) {
|
||||||
$currencyId = (int) $journal['currency_id'];
|
$field = $convertToNative && $currency->id !== $journal['currency_id'] ? 'native_amount' : 'amount';
|
||||||
$amount = $journal['amount'] ?? '0';
|
$currencyId = $convertToNative ? $currency->id : (int) $journal['currency_id'];
|
||||||
|
$amount = $journal[$field] ?? '0';
|
||||||
$incomes[$currencyId] ??= '0';
|
$incomes[$currencyId] ??= '0';
|
||||||
$incomes[$currencyId] = bcadd($incomes[$currencyId], app('steam')->positive($amount));
|
$incomes[$currencyId] = bcadd($incomes[$currencyId], app('steam')->positive($amount));
|
||||||
$sums[$currencyId] ??= '0';
|
$sums[$currencyId] ??= '0';
|
||||||
@@ -197,17 +104,18 @@ class BoxController extends Controller
|
|||||||
/** @var GroupCollectorInterface $collector */
|
/** @var GroupCollectorInterface $collector */
|
||||||
$collector = app(GroupCollectorInterface::class);
|
$collector = app(GroupCollectorInterface::class);
|
||||||
$collector->setRange($start, $end)
|
$collector->setRange($start, $end)
|
||||||
->setTypes([TransactionType::WITHDRAWAL])
|
->setTypes([TransactionType::WITHDRAWAL]);
|
||||||
;
|
|
||||||
$set = $collector->getExtractedJournals();
|
$set = $collector->getExtractedJournals();
|
||||||
|
|
||||||
/** @var array $journal */
|
/** @var array $journal */
|
||||||
foreach ($set as $journal) {
|
foreach ($set as $journal) {
|
||||||
$currencyId = (int) $journal['currency_id'];
|
$field = $convertToNative && $currency->id !== $journal['currency_id'] ? 'native_amount' : 'amount';
|
||||||
|
$currencyId = $convertToNative ? $currency->id : (int) $journal['currency_id'];
|
||||||
|
$amount = $journal[$field] ?? '0';
|
||||||
$expenses[$currencyId] ??= '0';
|
$expenses[$currencyId] ??= '0';
|
||||||
$expenses[$currencyId] = bcadd($expenses[$currencyId], $journal['amount'] ?? '0');
|
$expenses[$currencyId] = bcadd($expenses[$currencyId], $amount);
|
||||||
$sums[$currencyId] ??= '0';
|
$sums[$currencyId] ??= '0';
|
||||||
$sums[$currencyId] = bcadd($sums[$currencyId], $journal['amount']);
|
$sums[$currencyId] = bcadd($sums[$currencyId], $amount);
|
||||||
}
|
}
|
||||||
|
|
||||||
// format amounts:
|
// format amounts:
|
||||||
|
@@ -67,6 +67,8 @@ class Transaction extends Model
|
|||||||
'transaction_journal_id',
|
'transaction_journal_id',
|
||||||
'description',
|
'description',
|
||||||
'amount',
|
'amount',
|
||||||
|
'native_amount',
|
||||||
|
'native_foreign_amount',
|
||||||
'identifier',
|
'identifier',
|
||||||
'transaction_currency_id',
|
'transaction_currency_id',
|
||||||
'foreign_currency_id',
|
'foreign_currency_id',
|
||||||
|
@@ -60,8 +60,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
$search->whereLike('name', sprintf('%%%s', $query));
|
$search->whereLike('name', sprintf('%%%s', $query));
|
||||||
}
|
}
|
||||||
$search->orderBy('name', 'ASC')
|
$search->orderBy('name', 'ASC')
|
||||||
->where('active', true)
|
->where('active', true);
|
||||||
;
|
|
||||||
|
|
||||||
return $search->take($limit)->get();
|
return $search->take($limit)->get();
|
||||||
}
|
}
|
||||||
@@ -73,8 +72,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
$search->whereLike('name', sprintf('%s%%', $query));
|
$search->whereLike('name', sprintf('%s%%', $query));
|
||||||
}
|
}
|
||||||
$search->orderBy('name', 'ASC')
|
$search->orderBy('name', 'ASC')
|
||||||
->where('active', true)
|
->where('active', true);
|
||||||
;
|
|
||||||
|
|
||||||
return $search->take($limit)->get();
|
return $search->take($limit)->get();
|
||||||
}
|
}
|
||||||
@@ -178,8 +176,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
return $this->user->bills()
|
return $this->user->bills()
|
||||||
->orderBy('order', 'ASC')
|
->orderBy('order', 'ASC')
|
||||||
->orderBy('active', 'DESC')
|
->orderBy('active', 'DESC')
|
||||||
->orderBy('name', 'ASC')->get()
|
->orderBy('name', 'ASC')->get();
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getBillsForAccounts(Collection $accounts): Collection
|
public function getBillsForAccounts(Collection $accounts): Collection
|
||||||
@@ -220,8 +217,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
->orderBy('bills.active', 'DESC')
|
->orderBy('bills.active', 'DESC')
|
||||||
->orderBy('bills.name', 'ASC')
|
->orderBy('bills.name', 'ASC')
|
||||||
->groupBy($fields)
|
->groupBy($fields)
|
||||||
->get($fields)
|
->get($fields);
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -295,8 +291,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
{
|
{
|
||||||
return $this->user->bills()
|
return $this->user->bills()
|
||||||
->orderBy('active', 'DESC')
|
->orderBy('active', 'DESC')
|
||||||
->orderBy('name', 'ASC')->paginate($size)
|
->orderBy('name', 'ASC')->paginate($size);
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -315,8 +310,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
'transaction_journals.date',
|
'transaction_journals.date',
|
||||||
'transaction_journals.transaction_group_id',
|
'transaction_journals.transaction_group_id',
|
||||||
]
|
]
|
||||||
)
|
);
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -328,8 +322,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
->leftJoin('rule_actions', 'rule_actions.rule_id', '=', 'rules.id')
|
->leftJoin('rule_actions', 'rule_actions.rule_id', '=', 'rules.id')
|
||||||
->where('rule_actions.action_type', 'link_to_bill')
|
->where('rule_actions.action_type', 'link_to_bill')
|
||||||
->where('rule_actions.action_value', $bill->name)
|
->where('rule_actions.action_value', $bill->name)
|
||||||
->get(['rules.*'])
|
->get(['rules.*']);
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -343,8 +336,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
$rules = $this->user->rules()
|
$rules = $this->user->rules()
|
||||||
->leftJoin('rule_actions', 'rule_actions.rule_id', '=', 'rules.id')
|
->leftJoin('rule_actions', 'rule_actions.rule_id', '=', 'rules.id')
|
||||||
->where('rule_actions.action_type', 'link_to_bill')
|
->where('rule_actions.action_type', 'link_to_bill')
|
||||||
->get(['rules.id', 'rules.title', 'rule_actions.action_value', 'rules.active'])
|
->get(['rules.id', 'rules.title', 'rule_actions.action_value', 'rules.active']);
|
||||||
;
|
|
||||||
$array = [];
|
$array = [];
|
||||||
|
|
||||||
/** @var Rule $rule */
|
/** @var Rule $rule */
|
||||||
@@ -372,8 +364,7 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
$journals = $bill->transactionJournals()
|
$journals = $bill->transactionJournals()
|
||||||
->where('date', '>=', $date->year . '-01-01 00:00:00')
|
->where('date', '>=', $date->year . '-01-01 00:00:00')
|
||||||
->where('date', '<=', $date->year . '-12-31 23:59:59')
|
->where('date', '<=', $date->year . '-12-31 23:59:59')
|
||||||
->get()
|
->get();
|
||||||
;
|
|
||||||
|
|
||||||
/** @var TransactionJournal $journal */
|
/** @var TransactionJournal $journal */
|
||||||
foreach ($journals as $journal) {
|
foreach ($journals as $journal) {
|
||||||
@@ -512,13 +503,16 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
{
|
{
|
||||||
$bills = $this->getActiveBills();
|
$bills = $this->getActiveBills();
|
||||||
$return = [];
|
$return = [];
|
||||||
|
$convertToNative = app('preferences')->getForUser($this->user, 'convert_to_native', false)->data;
|
||||||
|
$default = app('amount')->getDefaultCurrency();
|
||||||
|
|
||||||
/** @var Bill $bill */
|
/** @var Bill $bill */
|
||||||
foreach ($bills as $bill) {
|
foreach ($bills as $bill) {
|
||||||
/** @var Collection $set */
|
/** @var Collection $set */
|
||||||
$set = $bill->transactionJournals()->after($start)->before($end)->get(['transaction_journals.*']);
|
$set = $bill->transactionJournals()->after($start)->before($end)->get(['transaction_journals.*']);
|
||||||
$currency = $bill->transactionCurrency;
|
$currency = $convertToNative && $bill->transactionCurrency->id !== $default->id ? $default : $bill->transactionCurrency;
|
||||||
|
$field = $convertToNative && $bill->transactionCurrency->id !== $default->id ? 'native_amount' : 'amount';
|
||||||
|
$foreignField = $convertToNative && $bill->transactionCurrency->id !== $default->id ? 'native_foreign_amount' : 'foreign_amount';
|
||||||
$return[$currency->id] ??= [
|
$return[$currency->id] ??= [
|
||||||
'id' => (string) $currency->id,
|
'id' => (string) $currency->id,
|
||||||
'name' => $currency->name,
|
'name' => $currency->name,
|
||||||
@@ -533,10 +527,10 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
/** @var null|Transaction $sourceTransaction */
|
/** @var null|Transaction $sourceTransaction */
|
||||||
$sourceTransaction = $transactionJournal->transactions()->where('amount', '<', 0)->first();
|
$sourceTransaction = $transactionJournal->transactions()->where('amount', '<', 0)->first();
|
||||||
if (null !== $sourceTransaction) {
|
if (null !== $sourceTransaction) {
|
||||||
$amount = $sourceTransaction->amount;
|
$amount = $sourceTransaction->$field;
|
||||||
if ((int) $sourceTransaction->foreign_currency_id === $currency->id) {
|
if ((int) $sourceTransaction->foreign_currency_id === $currency->id) {
|
||||||
// use foreign amount instead!
|
// use foreign amount instead!
|
||||||
$amount = (string) $sourceTransaction->foreign_amount;
|
$amount = (string) $sourceTransaction->$foreignField;
|
||||||
}
|
}
|
||||||
$return[$currency->id]['sum'] = bcadd($return[$currency->id]['sum'], $amount);
|
$return[$currency->id]['sum'] = bcadd($return[$currency->id]['sum'], $amount);
|
||||||
}
|
}
|
||||||
@@ -560,6 +554,8 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
app('log')->debug(sprintf('Now in sumUnpaidInRange("%s", "%s")', $start->format('Y-m-d'), $end->format('Y-m-d')));
|
app('log')->debug(sprintf('Now in sumUnpaidInRange("%s", "%s")', $start->format('Y-m-d'), $end->format('Y-m-d')));
|
||||||
$bills = $this->getActiveBills();
|
$bills = $this->getActiveBills();
|
||||||
$return = [];
|
$return = [];
|
||||||
|
$convertToNative = app('preferences')->getForUser($this->user, 'convert_to_native', false)->data;
|
||||||
|
$default = app('amount')->getDefaultCurrency();
|
||||||
|
|
||||||
/** @var Bill $bill */
|
/** @var Bill $bill */
|
||||||
foreach ($bills as $bill) {
|
foreach ($bills as $bill) {
|
||||||
@@ -570,9 +566,12 @@ class BillRepository implements BillRepositoryInterface
|
|||||||
// app('log')->debug(sprintf('Pay dates: %d, count: %d, left: %d', $dates->count(), $count, $total));
|
// app('log')->debug(sprintf('Pay dates: %d, count: %d, left: %d', $dates->count(), $count, $total));
|
||||||
// app('log')->debug('dates', $dates->toArray());
|
// app('log')->debug('dates', $dates->toArray());
|
||||||
|
|
||||||
|
$minField = $convertToNative && $bill->transactionCurrency->id !== $default->id ? 'native_amount_min' : 'amount_min';
|
||||||
|
$maxField = $convertToNative && $bill->transactionCurrency->id !== $default->id ? 'native_amount_max' : 'amount_max';
|
||||||
|
|
||||||
if ($total > 0) {
|
if ($total > 0) {
|
||||||
$currency = $bill->transactionCurrency;
|
$currency = $convertToNative && $bill->transactionCurrency->id !== $default->id ? $default : $bill->transactionCurrency;
|
||||||
$average = bcdiv(bcadd($bill->amount_max, $bill->amount_min), '2');
|
$average = bcdiv(bcadd($bill->$maxField, $bill->$minField), '2');
|
||||||
$return[$currency->id] ??= [
|
$return[$currency->id] ??= [
|
||||||
'id' => (string) $currency->id,
|
'id' => (string) $currency->id,
|
||||||
'name' => $currency->name,
|
'name' => $currency->name,
|
||||||
|
@@ -104,8 +104,7 @@ class AvailableBudgetRepository implements AvailableBudgetRepositoryInterface
|
|||||||
->where('transaction_currency_id', $currency->id)
|
->where('transaction_currency_id', $currency->id)
|
||||||
->where('start_date', $start->format('Y-m-d'))
|
->where('start_date', $start->format('Y-m-d'))
|
||||||
->where('end_date', $end->format('Y-m-d'))
|
->where('end_date', $end->format('Y-m-d'))
|
||||||
->first()
|
->first();
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getAvailableBudget(TransactionCurrency $currency, Carbon $start, Carbon $end): string
|
public function getAvailableBudget(TransactionCurrency $currency, Carbon $start, Carbon $end): string
|
||||||
@@ -116,8 +115,7 @@ class AvailableBudgetRepository implements AvailableBudgetRepositoryInterface
|
|||||||
$availableBudget = $this->user->availableBudgets()
|
$availableBudget = $this->user->availableBudgets()
|
||||||
->where('transaction_currency_id', $currency->id)
|
->where('transaction_currency_id', $currency->id)
|
||||||
->where('start_date', $start->format('Y-m-d'))
|
->where('start_date', $start->format('Y-m-d'))
|
||||||
->where('end_date', $end->format('Y-m-d'))->first()
|
->where('end_date', $end->format('Y-m-d'))->first();
|
||||||
;
|
|
||||||
if (null !== $availableBudget) {
|
if (null !== $availableBudget) {
|
||||||
$amount = $availableBudget->amount;
|
$amount = $availableBudget->amount;
|
||||||
}
|
}
|
||||||
@@ -130,14 +128,19 @@ class AvailableBudgetRepository implements AvailableBudgetRepositoryInterface
|
|||||||
$return = [];
|
$return = [];
|
||||||
$availableBudgets = $this->user->availableBudgets()
|
$availableBudgets = $this->user->availableBudgets()
|
||||||
->where('start_date', $start->format('Y-m-d'))
|
->where('start_date', $start->format('Y-m-d'))
|
||||||
->where('end_date', $end->format('Y-m-d'))->get()
|
->where('end_date', $end->format('Y-m-d'))->get();
|
||||||
;
|
|
||||||
|
// use native amount if necessary?
|
||||||
|
$convertToNative = app('preferences')->getForUser($this->user, 'convert_to_native', false)->data;
|
||||||
|
$default = app('amount')->getDefaultCurrency();
|
||||||
|
|
||||||
/** @var AvailableBudget $availableBudget */
|
/** @var AvailableBudget $availableBudget */
|
||||||
foreach ($availableBudgets as $availableBudget) {
|
foreach ($availableBudgets as $availableBudget) {
|
||||||
$return[$availableBudget->transaction_currency_id] = $availableBudget->amount;
|
$currencyId = $convertToNative && $availableBudget->transaction_currency_id !== $default->id ? $default->id : $availableBudget->transaction_currency_id;
|
||||||
|
$field = $convertToNative && $availableBudget->transaction_currency_id !== $default->id ? 'native_amount' : 'amount';
|
||||||
|
$return[$currencyId] = $return[$currencyId] ?? '0';
|
||||||
|
$return[$currencyId] = bcadd($return[$currencyId], $availableBudget->$field);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $return;
|
return $return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -174,8 +177,7 @@ class AvailableBudgetRepository implements AvailableBudgetRepositoryInterface
|
|||||||
return $this->user->availableBudgets()
|
return $this->user->availableBudgets()
|
||||||
->where('start_date', '=', $start->format('Y-m-d'))
|
->where('start_date', '=', $start->format('Y-m-d'))
|
||||||
->where('end_date', '=', $end->format('Y-m-d'))
|
->where('end_date', '=', $end->format('Y-m-d'))
|
||||||
->get()
|
->get();
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getByCurrencyDate(Carbon $start, Carbon $end, TransactionCurrency $currency): ?AvailableBudget
|
public function getByCurrencyDate(Carbon $start, Carbon $end, TransactionCurrency $currency): ?AvailableBudget
|
||||||
@@ -184,8 +186,7 @@ class AvailableBudgetRepository implements AvailableBudgetRepositoryInterface
|
|||||||
->availableBudgets()
|
->availableBudgets()
|
||||||
->where('transaction_currency_id', $currency->id)
|
->where('transaction_currency_id', $currency->id)
|
||||||
->where('start_date', $start->format('Y-m-d'))
|
->where('start_date', $start->format('Y-m-d'))
|
||||||
->where('end_date', $end->format('Y-m-d'))->first()
|
->where('end_date', $end->format('Y-m-d'))->first();
|
||||||
;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -196,8 +197,7 @@ class AvailableBudgetRepository implements AvailableBudgetRepositoryInterface
|
|||||||
$availableBudget = $this->user->availableBudgets()
|
$availableBudget = $this->user->availableBudgets()
|
||||||
->where('transaction_currency_id', $currency->id)
|
->where('transaction_currency_id', $currency->id)
|
||||||
->where('start_date', $start->format('Y-m-d'))
|
->where('start_date', $start->format('Y-m-d'))
|
||||||
->where('end_date', $end->format('Y-m-d'))->first()
|
->where('end_date', $end->format('Y-m-d'))->first();
|
||||||
;
|
|
||||||
if (null === $availableBudget) {
|
if (null === $availableBudget) {
|
||||||
$availableBudget = new AvailableBudget();
|
$availableBudget = new AvailableBudget();
|
||||||
$availableBudget->user()->associate($this->user);
|
$availableBudget->user()->associate($this->user);
|
||||||
|
@@ -207,11 +207,8 @@ class OperationsRepository implements OperationsRepositoryInterface
|
|||||||
?Collection $accounts = null,
|
?Collection $accounts = null,
|
||||||
?Collection $budgets = null,
|
?Collection $budgets = null,
|
||||||
?TransactionCurrency $currency = null
|
?TransactionCurrency $currency = null
|
||||||
): array {
|
): array
|
||||||
// app('log')->debug(sprintf('Now in %s', __METHOD__));
|
{
|
||||||
$start->startOfDay();
|
|
||||||
$end->endOfDay();
|
|
||||||
|
|
||||||
// this collector excludes all transfers TO
|
// this collector excludes all transfers TO
|
||||||
// liabilities (which are also withdrawals)
|
// liabilities (which are also withdrawals)
|
||||||
// because those expenses only become expenses
|
// because those expenses only become expenses
|
||||||
@@ -222,6 +219,10 @@ class OperationsRepository implements OperationsRepositoryInterface
|
|||||||
$subset = $repository->getAccountsByType(config('firefly.valid_liabilities'));
|
$subset = $repository->getAccountsByType(config('firefly.valid_liabilities'));
|
||||||
$selection = new Collection();
|
$selection = new Collection();
|
||||||
|
|
||||||
|
// default currency information for native stuff.
|
||||||
|
$convertToNative = app('preferences')->get('convert_to_native', false)->data;
|
||||||
|
$default = app('amount')->getDefaultCurrency();
|
||||||
|
|
||||||
/** @var Account $account */
|
/** @var Account $account */
|
||||||
foreach ($subset as $account) {
|
foreach ($subset as $account) {
|
||||||
if ('credit' === $repository->getMetaValue($account, 'liability_direction')) {
|
if ('credit' === $repository->getMetaValue($account, 'liability_direction')) {
|
||||||
@@ -234,8 +235,7 @@ class OperationsRepository implements OperationsRepositoryInterface
|
|||||||
$collector->setUser($this->user)
|
$collector->setUser($this->user)
|
||||||
->setRange($start, $end)
|
->setRange($start, $end)
|
||||||
->excludeDestinationAccounts($selection)
|
->excludeDestinationAccounts($selection)
|
||||||
->setTypes([TransactionType::WITHDRAWAL])
|
->setTypes([TransactionType::WITHDRAWAL]);
|
||||||
;
|
|
||||||
|
|
||||||
if (null !== $accounts) {
|
if (null !== $accounts) {
|
||||||
$collector->setAccounts($accounts);
|
$collector->setAccounts($accounts);
|
||||||
@@ -255,8 +255,7 @@ class OperationsRepository implements OperationsRepositoryInterface
|
|||||||
/** @var GroupCollectorInterface $collector */
|
/** @var GroupCollectorInterface $collector */
|
||||||
$collector = app(GroupCollectorInterface::class);
|
$collector = app(GroupCollectorInterface::class);
|
||||||
$collector->setUser($this->user)->setRange($start, $end)->setTypes([TransactionType::WITHDRAWAL])
|
$collector->setUser($this->user)->setRange($start, $end)->setTypes([TransactionType::WITHDRAWAL])
|
||||||
->setForeignCurrency($currency)->setBudgets($budgets)
|
->setForeignCurrency($currency)->setBudgets($budgets);
|
||||||
;
|
|
||||||
|
|
||||||
if (null !== $accounts) {
|
if (null !== $accounts) {
|
||||||
$collector->setAccounts($accounts);
|
$collector->setAccounts($accounts);
|
||||||
@@ -270,19 +269,36 @@ class OperationsRepository implements OperationsRepositoryInterface
|
|||||||
|
|
||||||
foreach ($journals as $journal) {
|
foreach ($journals as $journal) {
|
||||||
$currencyId = (int) $journal['currency_id'];
|
$currencyId = (int) $journal['currency_id'];
|
||||||
|
$currencyName = $journal['currency_name'];
|
||||||
|
$currencySymbol = $journal['currency_symbol'];
|
||||||
|
$currencyCode = $journal['currency_code'];
|
||||||
|
$currencyDecimalPlaces = $journal['currency_decimal_places'];
|
||||||
|
$field = 'amount';
|
||||||
|
$foreignField = 'foreign_amount';
|
||||||
|
// if the user wants everything in native currency, use it.
|
||||||
|
if ($convertToNative && $default->id !== (int) $journal['currency_id']) {
|
||||||
|
// use default currency info
|
||||||
|
$currencyId = $default->id;
|
||||||
|
$currencyName = $default->name;
|
||||||
|
$currencySymbol = $default->symbol;
|
||||||
|
$currencyCode = $default->code;
|
||||||
|
$currencyDecimalPlaces = $default->decimal_places;
|
||||||
|
$field = 'native_amount';
|
||||||
|
$foreignField = 'native_foreign_amount';
|
||||||
|
}
|
||||||
$array[$currencyId] ??= [
|
$array[$currencyId] ??= [
|
||||||
'sum' => '0',
|
'sum' => '0',
|
||||||
'currency_id' => $currencyId,
|
'currency_id' => $currencyId,
|
||||||
'currency_name' => $journal['currency_name'],
|
'currency_name' => $currencyName,
|
||||||
'currency_symbol' => $journal['currency_symbol'],
|
'currency_symbol' => $currencySymbol,
|
||||||
'currency_code' => $journal['currency_code'],
|
'currency_code' => $currencyCode,
|
||||||
'currency_decimal_places' => $journal['currency_decimal_places'],
|
'currency_decimal_places' => $currencyDecimalPlaces,
|
||||||
];
|
];
|
||||||
$array[$currencyId]['sum'] = bcadd($array[$currencyId]['sum'], app('steam')->negative($journal['amount']));
|
$array[$currencyId]['sum'] = bcadd($array[$currencyId]['sum'], app('steam')->negative($journal[$field]));
|
||||||
|
|
||||||
// also do foreign amount:
|
// also do foreign amount:
|
||||||
$foreignId = (int) $journal['foreign_currency_id'];
|
$foreignId = (int) $journal['foreign_currency_id'];
|
||||||
if (0 !== $foreignId) {
|
if (0 !== $foreignId && $foreignId !== $currencyId) {
|
||||||
$array[$foreignId] ??= [
|
$array[$foreignId] ??= [
|
||||||
'sum' => '0',
|
'sum' => '0',
|
||||||
'currency_id' => $foreignId,
|
'currency_id' => $foreignId,
|
||||||
|
Reference in New Issue
Block a user