mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-10-13 16:00:13 +00:00
Start improving bill overview.
This commit is contained in:
@@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller;
|
|||||||
use FireflyIII\Exceptions\FireflyException;
|
use FireflyIII\Exceptions\FireflyException;
|
||||||
use FireflyIII\Models\Bill;
|
use FireflyIII\Models\Bill;
|
||||||
use FireflyIII\Repositories\Bill\BillRepositoryInterface;
|
use FireflyIII\Repositories\Bill\BillRepositoryInterface;
|
||||||
|
use FireflyIII\Support\JsonApi\Enrichments\SubscriptionEnrichment;
|
||||||
use FireflyIII\Transformers\BillTransformer;
|
use FireflyIII\Transformers\BillTransformer;
|
||||||
|
use FireflyIII\User;
|
||||||
use Illuminate\Http\JsonResponse;
|
use Illuminate\Http\JsonResponse;
|
||||||
use Illuminate\Pagination\LengthAwarePaginator;
|
use Illuminate\Pagination\LengthAwarePaginator;
|
||||||
use League\Fractal\Pagination\IlluminatePaginatorAdapter;
|
use League\Fractal\Pagination\IlluminatePaginatorAdapter;
|
||||||
@@ -76,6 +78,15 @@ class ShowController extends Controller
|
|||||||
$bills = $bills->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize);
|
$bills = $bills->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize);
|
||||||
$paginator = new LengthAwarePaginator($bills, $count, $pageSize, $this->parameters->get('page'));
|
$paginator = new LengthAwarePaginator($bills, $count, $pageSize, $this->parameters->get('page'));
|
||||||
|
|
||||||
|
// enrich
|
||||||
|
/** @var User $admin */
|
||||||
|
$admin = auth()->user();
|
||||||
|
$enrichment = new SubscriptionEnrichment();
|
||||||
|
$enrichment->setUser($admin);
|
||||||
|
$enrichment->setConvertToNative($this->convertToNative);
|
||||||
|
$enrichment->setNative($this->nativeCurrency);
|
||||||
|
$bills = $enrichment->enrich($bills);
|
||||||
|
|
||||||
/** @var BillTransformer $transformer */
|
/** @var BillTransformer $transformer */
|
||||||
$transformer = app(BillTransformer::class);
|
$transformer = app(BillTransformer::class);
|
||||||
$transformer->setParameters($this->parameters);
|
$transformer->setParameters($this->parameters);
|
||||||
|
157
app/Support/JsonApi/Enrichments/SubscriptionEnrichment.php
Normal file
157
app/Support/JsonApi/Enrichments/SubscriptionEnrichment.php
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace FireflyIII\Support\JsonApi\Enrichments;
|
||||||
|
|
||||||
|
use FireflyIII\Models\Account;
|
||||||
|
use FireflyIII\Models\Bill;
|
||||||
|
use FireflyIII\Models\Note;
|
||||||
|
use FireflyIII\Models\ObjectGroup;
|
||||||
|
use FireflyIII\Models\TransactionCurrency;
|
||||||
|
use FireflyIII\Models\UserGroup;
|
||||||
|
use FireflyIII\Support\Facades\Steam;
|
||||||
|
use FireflyIII\Support\Http\Api\ExchangeRateConverter;
|
||||||
|
use FireflyIII\User;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
|
use Illuminate\Support\Facades\DB;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
|
class SubscriptionEnrichment implements EnrichmentInterface
|
||||||
|
{
|
||||||
|
private User $user;
|
||||||
|
private UserGroup $userGroup;
|
||||||
|
private Collection $collection;
|
||||||
|
private bool $convertToNative = false;
|
||||||
|
private array $subscriptionIds = [];
|
||||||
|
private array $objectGroups = [];
|
||||||
|
private array $mappedObjects = [];
|
||||||
|
private array $notes = [];
|
||||||
|
private TransactionCurrency $nativeCurrency;
|
||||||
|
|
||||||
|
public function enrich(Collection $collection): Collection
|
||||||
|
{
|
||||||
|
$this->collection = $collection;
|
||||||
|
$this->collectSubscriptionIds();
|
||||||
|
$this->collectNotes();
|
||||||
|
$this->collectObjectGroups();
|
||||||
|
|
||||||
|
$notes = $this->notes;
|
||||||
|
$objectGroups = $this->objectGroups;
|
||||||
|
$this->collection = $this->collection->map(function (Bill $item) use ($notes, $objectGroups) {
|
||||||
|
$id = (int) $item->id;
|
||||||
|
$currency = $item->transactionCurrency;
|
||||||
|
$meta = [
|
||||||
|
'notes' => null,
|
||||||
|
'object_group_id' => null,
|
||||||
|
'object_group_title' => null,
|
||||||
|
'object_group_order' => null,
|
||||||
|
];
|
||||||
|
$amounts = [
|
||||||
|
'amount_min' => Steam::bcround($item->amount_min, $currency->decimal_places),
|
||||||
|
'amount_max' => Steam::bcround($item->amount_max, $currency->decimal_places),
|
||||||
|
'average' => Steam::bcround(bcdiv(bcadd($item->amount_min, $item->amount_max), '2'), $currency->decimal_places),
|
||||||
|
];
|
||||||
|
|
||||||
|
// add object group if available
|
||||||
|
if (array_key_exists($id, $this->mappedObjects)) {
|
||||||
|
$key = $this->mappedObjects[$id];
|
||||||
|
$meta['object_group_id'] = $objectGroups[$key]['id'];
|
||||||
|
$meta['object_group_title'] = $objectGroups[$key]['title'];
|
||||||
|
$meta['object_group_order'] = $objectGroups[$key]['order'];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add notes if available.
|
||||||
|
if (array_key_exists($item->id, $notes)) {
|
||||||
|
$meta['notes'] = $notes[$item->id];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert amounts to native currency if needed
|
||||||
|
if ($this->convertToNative && $item->currency_id !== $this->nativeCurrency->id) {
|
||||||
|
$converter = new ExchangeRateConverter();
|
||||||
|
$amounts = [
|
||||||
|
'amount_min' => Steam::bcround($converter->convert($item->transactionCurrency, $this->nativeCurrency, today(), $item->amount_min), $this->nativeCurrency->decimal_places),
|
||||||
|
'amount_max' => Steam::bcround($converter->convert($item->transactionCurrency, $this->nativeCurrency, today(), $item->amount_max), $this->nativeCurrency->decimal_places),
|
||||||
|
];
|
||||||
|
$amounts['average'] =Steam::bcround(bcdiv(bcadd($amounts['amount_min'], $amounts['amount_max']), '2'), $this->nativeCurrency->decimal_places);
|
||||||
|
}
|
||||||
|
$item->amounts = $amounts;
|
||||||
|
$item->meta = $meta;
|
||||||
|
return $item;
|
||||||
|
});
|
||||||
|
|
||||||
|
return $collection;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function enrichSingle(Model|array $model): array|Model
|
||||||
|
{
|
||||||
|
Log::debug(__METHOD__);
|
||||||
|
$collection = new Collection([$model]);
|
||||||
|
$collection = $this->enrich($collection);
|
||||||
|
|
||||||
|
return $collection->first();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectNotes(): void
|
||||||
|
{
|
||||||
|
$notes = Note::query()->whereIn('noteable_id', $this->subscriptionIds)
|
||||||
|
->whereNotNull('notes.text')
|
||||||
|
->where('notes.text', '!=', '')
|
||||||
|
->where('noteable_type', Bill::class)->get(['notes.noteable_id', 'notes.text'])->toArray()
|
||||||
|
;
|
||||||
|
foreach ($notes as $note) {
|
||||||
|
$this->notes[(int) $note['noteable_id']] = (string) $note['text'];
|
||||||
|
}
|
||||||
|
Log::debug(sprintf('Enrich with %d note(s)', count($this->notes)));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setUser(User $user): void
|
||||||
|
{
|
||||||
|
$this->user = $user;
|
||||||
|
$this->userGroup = $user->userGroup;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setUserGroup(UserGroup $userGroup): void
|
||||||
|
{
|
||||||
|
$this->userGroup = $userGroup;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setConvertToNative(bool $convertToNative): void
|
||||||
|
{
|
||||||
|
$this->convertToNative = $convertToNative;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setNative(TransactionCurrency $nativeCurrency): void
|
||||||
|
{
|
||||||
|
$this->nativeCurrency = $nativeCurrency;
|
||||||
|
}
|
||||||
|
private function collectSubscriptionIds(): void
|
||||||
|
{
|
||||||
|
/** @var Bill $bill */
|
||||||
|
foreach ($this->collection as $bill) {
|
||||||
|
$this->subscriptionIds[] = (int) $bill->id;
|
||||||
|
}
|
||||||
|
$this->subscriptionIds = array_unique($this->subscriptionIds);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectObjectGroups(): void
|
||||||
|
{
|
||||||
|
$set = DB::table('object_groupables')
|
||||||
|
->whereIn('object_groupable_id', $this->subscriptionIds)
|
||||||
|
->where('object_groupable_type', Bill::class)
|
||||||
|
->get(['object_groupable_id','object_group_id']);
|
||||||
|
|
||||||
|
$ids = array_unique($set->pluck('object_group_id')->toArray());
|
||||||
|
|
||||||
|
foreach($set as $entry) {
|
||||||
|
$this->mappedObjects[(int)$entry->object_groupable_id] = (int)$entry->object_group_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
$groups = ObjectGroup::whereIn('id', $ids)->get(['id', 'title','order'])->toArray();
|
||||||
|
foreach($groups as $group) {
|
||||||
|
$group['id'] = (int) $group['id'];
|
||||||
|
$group['order'] = (int) $group['order'];
|
||||||
|
$this->objectGroups[(int)$group['id']] = $group;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -44,7 +44,7 @@ class BillTransformer extends AbstractTransformer
|
|||||||
{
|
{
|
||||||
private readonly BillDateCalculator $calculator;
|
private readonly BillDateCalculator $calculator;
|
||||||
private readonly bool $convertToNative;
|
private readonly bool $convertToNative;
|
||||||
private readonly TransactionCurrency $default;
|
private readonly TransactionCurrency $native;
|
||||||
private readonly BillRepositoryInterface $repository;
|
private readonly BillRepositoryInterface $repository;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -54,7 +54,7 @@ class BillTransformer extends AbstractTransformer
|
|||||||
{
|
{
|
||||||
$this->repository = app(BillRepositoryInterface::class);
|
$this->repository = app(BillRepositoryInterface::class);
|
||||||
$this->calculator = app(BillDateCalculator::class);
|
$this->calculator = app(BillDateCalculator::class);
|
||||||
$this->default = Amount::getNativeCurrency();
|
$this->native = Amount::getNativeCurrency();
|
||||||
$this->convertToNative = Amount::convertToNative();
|
$this->convertToNative = Amount::convertToNative();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -66,33 +66,19 @@ class BillTransformer extends AbstractTransformer
|
|||||||
*/
|
*/
|
||||||
public function transform(Bill $bill): array
|
public function transform(Bill $bill): array
|
||||||
{
|
{
|
||||||
$default = $this->parameters->get('defaultCurrency') ?? $this->default;
|
$paidData = $this->paidData($bill);
|
||||||
|
$lastPaidDate = $this->getLastPaidDate($paidData);
|
||||||
$paidData = $this->paidData($bill);
|
$start = $this->parameters->get('start') ?? today()->subYears(10);
|
||||||
$lastPaidDate = $this->getLastPaidDate($paidData);
|
$end = $this->parameters->get('end') ?? today()->addYears(10);
|
||||||
$start = $this->parameters->get('start') ?? today()->subYears(10);
|
$payDates = $this->calculator->getPayDates($start, $end, $bill->date, $bill->repeat_freq, $bill->skip, $lastPaidDate);
|
||||||
$end = $this->parameters->get('end') ?? today()->addYears(10);
|
$currency = $bill->transactionCurrency;
|
||||||
$payDates = $this->calculator->getPayDates($start, $end, $bill->date, $bill->repeat_freq, $bill->skip, $lastPaidDate);
|
|
||||||
$currency = $bill->transactionCurrency;
|
|
||||||
$notes = $this->repository->getNoteText($bill);
|
|
||||||
$notes = '' === $notes ? null : $notes;
|
|
||||||
$objectGroupId = null;
|
|
||||||
$objectGroupOrder = null;
|
|
||||||
$objectGroupTitle = null;
|
|
||||||
$this->repository->setUser($bill->user);
|
$this->repository->setUser($bill->user);
|
||||||
|
|
||||||
/** @var null|ObjectGroup $objectGroup */
|
|
||||||
$objectGroup = $bill->objectGroups->first();
|
|
||||||
if (null !== $objectGroup) {
|
|
||||||
$objectGroupId = $objectGroup->id;
|
|
||||||
$objectGroupOrder = $objectGroup->order;
|
|
||||||
$objectGroupTitle = $objectGroup->title;
|
|
||||||
}
|
|
||||||
|
|
||||||
$paidDataFormatted = [];
|
$paidDataFormatted = [];
|
||||||
$payDatesFormatted = [];
|
$payDatesFormatted = [];
|
||||||
foreach ($paidData as $object) {
|
foreach ($paidData as $object) {
|
||||||
$date = Carbon::createFromFormat('!Y-m-d', $object['date'], config('app.timezone'));
|
$date = Carbon::createFromFormat('!Y-m-d', $object['date'], config('app.timezone'));
|
||||||
if (!$date instanceof Carbon) {
|
if (!$date instanceof Carbon) {
|
||||||
$date = today(config('app.timezone'));
|
$date = today(config('app.timezone'));
|
||||||
}
|
}
|
||||||
@@ -101,24 +87,24 @@ class BillTransformer extends AbstractTransformer
|
|||||||
}
|
}
|
||||||
|
|
||||||
foreach ($payDates as $string) {
|
foreach ($payDates as $string) {
|
||||||
$date = Carbon::createFromFormat('!Y-m-d', $string, config('app.timezone'));
|
$date = Carbon::createFromFormat('!Y-m-d', $string, config('app.timezone'));
|
||||||
if (!$date instanceof Carbon) {
|
if (!$date instanceof Carbon) {
|
||||||
$date = today(config('app.timezone'));
|
$date = today(config('app.timezone'));
|
||||||
}
|
}
|
||||||
$payDatesFormatted[] = $date->toAtomString();
|
$payDatesFormatted[] = $date->toAtomString();
|
||||||
}
|
}
|
||||||
// next expected match
|
// next expected match
|
||||||
$nem = null;
|
$nem = null;
|
||||||
$nemDate = null;
|
$nemDate = null;
|
||||||
$nemDiff = trans('firefly.not_expected_period');
|
$nemDiff = trans('firefly.not_expected_period');
|
||||||
$firstPayDate = $payDates[0] ?? null;
|
$firstPayDate = $payDates[0] ?? null;
|
||||||
|
|
||||||
if (null !== $firstPayDate) {
|
if (null !== $firstPayDate) {
|
||||||
$nemDate = Carbon::createFromFormat('!Y-m-d', $firstPayDate, config('app.timezone'));
|
$nemDate = Carbon::createFromFormat('!Y-m-d', $firstPayDate, config('app.timezone'));
|
||||||
if (!$nemDate instanceof Carbon) {
|
if (!$nemDate instanceof Carbon) {
|
||||||
$nemDate = today(config('app.timezone'));
|
$nemDate = today(config('app.timezone'));
|
||||||
}
|
}
|
||||||
$nem = $nemDate->toAtomString();
|
$nem = $nemDate->toAtomString();
|
||||||
|
|
||||||
// nullify again when it's outside the current view range.
|
// nullify again when it's outside the current view range.
|
||||||
if (
|
if (
|
||||||
@@ -139,7 +125,7 @@ class BillTransformer extends AbstractTransformer
|
|||||||
|
|
||||||
$current = $payDatesFormatted[0] ?? null;
|
$current = $payDatesFormatted[0] ?? null;
|
||||||
if (null !== $current && !$nemDate->isToday()) {
|
if (null !== $current && !$nemDate->isToday()) {
|
||||||
$temp2 = Carbon::createFromFormat('Y-m-d\TH:i:sP', $current);
|
$temp2 = Carbon::createFromFormat('Y-m-d\TH:i:sP', $current);
|
||||||
if (!$temp2 instanceof Carbon) {
|
if (!$temp2 instanceof Carbon) {
|
||||||
$temp2 = today(config('app.timezone'));
|
$temp2 = today(config('app.timezone'));
|
||||||
}
|
}
|
||||||
@@ -149,43 +135,44 @@ class BillTransformer extends AbstractTransformer
|
|||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'id' => $bill->id,
|
'id' => $bill->id,
|
||||||
'created_at' => $bill->created_at->toAtomString(),
|
'created_at' => $bill->created_at->toAtomString(),
|
||||||
'updated_at' => $bill->updated_at->toAtomString(),
|
'updated_at' => $bill->updated_at->toAtomString(),
|
||||||
'currency_id' => (string)$bill->transaction_currency_id,
|
'currency_id' => (string)$bill->transaction_currency_id,
|
||||||
'currency_code' => $currency->code,
|
'currency_code' => $currency->code,
|
||||||
'currency_symbol' => $currency->symbol,
|
'currency_symbol' => $currency->symbol,
|
||||||
'currency_decimal_places' => $currency->decimal_places,
|
'currency_decimal_places' => $currency->decimal_places,
|
||||||
'native_currency_id' => null === $default ? null : (string)$default->id,
|
|
||||||
'native_currency_code' => $default?->code,
|
'native_currency_id' => (string)$this->native->id,
|
||||||
'native_currency_symbol' => $default?->symbol,
|
'native_currency_code' => $this->native->code,
|
||||||
'native_currency_decimal_places' => $default?->decimal_places,
|
'native_currency_symbol' => $this->native->symbol,
|
||||||
'name' => $bill->name,
|
'native_currency_decimal_places' => $this->native->decimal_places,
|
||||||
'amount_min' => app('steam')->bcround($bill->amount_min, $currency->decimal_places),
|
|
||||||
'amount_max' => app('steam')->bcround($bill->amount_max, $currency->decimal_places),
|
'name' => $bill->name,
|
||||||
'native_amount_min' => $this->convertToNative ? app('steam')->bcround($bill->native_amount_min, $default->decimal_places) : null,
|
'amount_min' => $bill->amounts['amount_min'],
|
||||||
'native_amount_max' => $this->convertToNative ? app('steam')->bcround($bill->native_amount_max, $default->decimal_places) : null,
|
'amount_max' => $bill->amounts['amount_max'],
|
||||||
'date' => $bill->date->toAtomString(),
|
'amount_avg' => $bill->amounts['average'],
|
||||||
'end_date' => $bill->end_date?->toAtomString(),
|
'date' => $bill->date->toAtomString(),
|
||||||
'extension_date' => $bill->extension_date?->toAtomString(),
|
'end_date' => $bill->end_date?->toAtomString(),
|
||||||
'repeat_freq' => $bill->repeat_freq,
|
'extension_date' => $bill->extension_date?->toAtomString(),
|
||||||
'skip' => $bill->skip,
|
'repeat_freq' => $bill->repeat_freq,
|
||||||
'active' => $bill->active,
|
'skip' => $bill->skip,
|
||||||
'order' => $bill->order,
|
'active' => $bill->active,
|
||||||
'notes' => $notes,
|
'order' => $bill->order,
|
||||||
'object_group_id' => null !== $objectGroupId ? (string)$objectGroupId : null,
|
'notes' => $bill->meta['notes'],
|
||||||
'object_group_order' => $objectGroupOrder,
|
'object_group_id' => $bill->meta['object_group_id'],
|
||||||
'object_group_title' => $objectGroupTitle,
|
'object_group_order' => $bill->meta['object_group_order'],
|
||||||
|
'object_group_title' => $bill->meta['object_group_title'],
|
||||||
|
|
||||||
// these fields need work:
|
// these fields need work:
|
||||||
'next_expected_match' => $nem,
|
// 'next_expected_match' => $nem,
|
||||||
'next_expected_match_diff' => $nemDiff,
|
// 'next_expected_match_diff' => $nemDiff,
|
||||||
'pay_dates' => $payDatesFormatted,
|
// 'pay_dates' => $payDatesFormatted,
|
||||||
'paid_dates' => $paidDataFormatted,
|
// 'paid_dates' => $paidDataFormatted,
|
||||||
'links' => [
|
'links' => [
|
||||||
[
|
[
|
||||||
'rel' => 'self',
|
'rel' => 'self',
|
||||||
'uri' => '/bills/'.$bill->id,
|
'uri' => '/bills/' . $bill->id,
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
@@ -207,13 +194,13 @@ class BillTransformer extends AbstractTransformer
|
|||||||
// 2023-07-18 this particular date is used to search for the last paid date.
|
// 2023-07-18 this particular date is used to search for the last paid date.
|
||||||
// 2023-07-18 the cloned $searchDate is used to grab the correct transactions.
|
// 2023-07-18 the cloned $searchDate is used to grab the correct transactions.
|
||||||
/** @var Carbon $start */
|
/** @var Carbon $start */
|
||||||
$start = clone $this->parameters->get('start');
|
$start = clone $this->parameters->get('start');
|
||||||
$searchStart = clone $start;
|
$searchStart = clone $start;
|
||||||
$start->subDay();
|
$start->subDay();
|
||||||
|
|
||||||
/** @var Carbon $end */
|
/** @var Carbon $end */
|
||||||
$end = clone $this->parameters->get('end');
|
$end = clone $this->parameters->get('end');
|
||||||
$searchEnd = clone $end;
|
$searchEnd = clone $end;
|
||||||
|
|
||||||
// move the search dates to the start of the day.
|
// move the search dates to the start of the day.
|
||||||
$searchStart->startOfDay();
|
$searchStart->startOfDay();
|
||||||
@@ -223,7 +210,7 @@ class BillTransformer extends AbstractTransformer
|
|||||||
Log::debug(sprintf('Search parameters are: start: %s', $searchStart->format('Y-m-d')));
|
Log::debug(sprintf('Search parameters are: start: %s', $searchStart->format('Y-m-d')));
|
||||||
|
|
||||||
// Get from database when bill was paid.
|
// Get from database when bill was paid.
|
||||||
$set = $this->repository->getPaidDatesInRange($bill, $searchStart, $searchEnd);
|
$set = $this->repository->getPaidDatesInRange($bill, $searchStart, $searchEnd);
|
||||||
Log::debug(sprintf('Count %d entries in getPaidDatesInRange()', $set->count()));
|
Log::debug(sprintf('Count %d entries in getPaidDatesInRange()', $set->count()));
|
||||||
|
|
||||||
// Grab from array the most recent payment. If none exist, fall back to the start date and pretend *that* was the last paid date.
|
// Grab from array the most recent payment. If none exist, fall back to the start date and pretend *that* was the last paid date.
|
||||||
@@ -232,17 +219,17 @@ class BillTransformer extends AbstractTransformer
|
|||||||
Log::debug(sprintf('Result of lastPaidDate is %s', $lastPaidDate->format('Y-m-d')));
|
Log::debug(sprintf('Result of lastPaidDate is %s', $lastPaidDate->format('Y-m-d')));
|
||||||
|
|
||||||
// At this point the "next match" is exactly after the last time the bill was paid.
|
// At this point the "next match" is exactly after the last time the bill was paid.
|
||||||
$result = [];
|
$result = [];
|
||||||
foreach ($set as $entry) {
|
foreach ($set as $entry) {
|
||||||
$array = [
|
$array = [
|
||||||
'transaction_group_id' => (string)$entry->transaction_group_id,
|
'transaction_group_id' => (string)$entry->transaction_group_id,
|
||||||
'transaction_journal_id' => (string)$entry->id,
|
'transaction_journal_id' => (string)$entry->id,
|
||||||
'date' => $entry->date->format('Y-m-d'),
|
'date' => $entry->date->format('Y-m-d'),
|
||||||
'date_object' => $entry->date,
|
'date_object' => $entry->date,
|
||||||
'currency_id' => $entry->transaction_currency_id,
|
'currency_id' => $entry->transaction_currency_id,
|
||||||
'currency_code' => $entry->transaction_currency_code,
|
'currency_code' => $entry->transaction_currency_code,
|
||||||
'currency_decimal_places' => $entry->transaction_currency_decimal_places,
|
'currency_decimal_places' => $entry->transaction_currency_decimal_places,
|
||||||
'amount' => Steam::bcround($entry->amount, $entry->transaction_currency_decimal_places),
|
'amount' => Steam::bcround($entry->amount, $entry->transaction_currency_decimal_places),
|
||||||
];
|
];
|
||||||
if (null !== $entry->foreign_amount && null !== $entry->foreign_currency_code) {
|
if (null !== $entry->foreign_amount && null !== $entry->foreign_currency_code) {
|
||||||
$array['foreign_currency_id'] = $entry->foreign_currency_id;
|
$array['foreign_currency_id'] = $entry->foreign_currency_id;
|
||||||
|
@@ -163,14 +163,10 @@ function downloadSubscriptions(params) {
|
|||||||
currency_code: bill.currency_code,
|
currency_code: bill.currency_code,
|
||||||
paid: 0,
|
paid: 0,
|
||||||
unpaid: 0,
|
unpaid: 0,
|
||||||
// native_currency_code: bill.native_currency_code,
|
|
||||||
// native_paid: 0,
|
|
||||||
//native_unpaid: 0,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
subscriptionData[objectGroupId].payment_info[bill.currency_code].unpaid += totalAmount;
|
subscriptionData[objectGroupId].payment_info[bill.currency_code].unpaid += totalAmount;
|
||||||
//subscriptionData[objectGroupId].payment_info[bill.currency_code].native_unpaid += totalNativeAmount;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (current.attributes.paid_dates.length > 0) {
|
if (current.attributes.paid_dates.length > 0) {
|
||||||
@@ -178,8 +174,6 @@ function downloadSubscriptions(params) {
|
|||||||
if (current.attributes.paid_dates.hasOwnProperty(ii)) {
|
if (current.attributes.paid_dates.hasOwnProperty(ii)) {
|
||||||
// bill is paid!
|
// bill is paid!
|
||||||
// since bill is paid, 3 possible currencies:
|
// since bill is paid, 3 possible currencies:
|
||||||
// native, currency, foreign currency.
|
|
||||||
// foreign currency amount (converted to native or not) will be ignored.
|
|
||||||
let currentJournal = current.attributes.paid_dates[ii];
|
let currentJournal = current.attributes.paid_dates[ii];
|
||||||
// new array for the currency
|
// new array for the currency
|
||||||
if (!subscriptionData[objectGroupId].payment_info.hasOwnProperty(currentJournal.currency_code)) {
|
if (!subscriptionData[objectGroupId].payment_info.hasOwnProperty(currentJournal.currency_code)) {
|
||||||
@@ -187,15 +181,10 @@ function downloadSubscriptions(params) {
|
|||||||
currency_code: bill.currency_code,
|
currency_code: bill.currency_code,
|
||||||
paid: 0,
|
paid: 0,
|
||||||
unpaid: 0,
|
unpaid: 0,
|
||||||
// native_currency_code: bill.native_currency_code,
|
|
||||||
// native_paid: 0,
|
|
||||||
//native_unpaid: 0,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
const amount = parseFloat(currentJournal.amount) * -1;
|
const amount = parseFloat(currentJournal.amount) * -1;
|
||||||
// const nativeAmount = parseFloat(currentJournal.native_amount) * -1;
|
|
||||||
subscriptionData[objectGroupId].payment_info[currentJournal.currency_code].paid += amount;
|
subscriptionData[objectGroupId].payment_info[currentJournal.currency_code].paid += amount;
|
||||||
// subscriptionData[objectGroupId].payment_info[currentJournal.currency_code].native_paid += nativeAmount;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -221,6 +210,14 @@ export default () => ({
|
|||||||
formatMoney(amount, currencyCode) {
|
formatMoney(amount, currencyCode) {
|
||||||
return formatMoney(amount, currencyCode);
|
return formatMoney(amount, currencyCode);
|
||||||
},
|
},
|
||||||
|
eventListeners: {
|
||||||
|
['@convert-to-native.window'](event){
|
||||||
|
console.log('I heard that! (dashboard/subscriptions)');
|
||||||
|
this.convertToNative = event.detail;
|
||||||
|
this.startSubscriptions();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
startSubscriptions() {
|
startSubscriptions() {
|
||||||
this.loading = true;
|
this.loading = true;
|
||||||
let start = new Date(window.store.get('start'));
|
let start = new Date(window.store.get('start'));
|
||||||
|
@@ -1865,6 +1865,8 @@ return [
|
|||||||
|
|
||||||
// bills:
|
// bills:
|
||||||
'left_to_pay_lc' => 'left to pay',
|
'left_to_pay_lc' => 'left to pay',
|
||||||
|
'less_than_expected' => 'less than expected',
|
||||||
|
'more_than_expected' => 'more than expected',
|
||||||
'skip_help_text' => 'Use the skip field to create bi-monthly (skip = 1) or other custom intervals.',
|
'skip_help_text' => 'Use the skip field to create bi-monthly (skip = 1) or other custom intervals.',
|
||||||
'subscription' => 'Subscription',
|
'subscription' => 'Subscription',
|
||||||
'not_expected_period' => 'Not expected this period',
|
'not_expected_period' => 'Not expected this period',
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
<div class="col" x-data="subscriptions">
|
<div class="col" x-data="subscriptions" x-bind="eventListeners">
|
||||||
<template x-for="group in subscriptions">
|
<template x-for="group in subscriptions">
|
||||||
<div class="card mb-2">
|
<div class="card mb-2">
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
@@ -62,7 +62,16 @@
|
|||||||
<template x-for="transaction in bill.transactions">
|
<template x-for="transaction in bill.transactions">
|
||||||
<li>
|
<li>
|
||||||
<span :title="transaction.amount" x-text="transaction.amount"></span>
|
<span :title="transaction.amount" x-text="transaction.amount"></span>
|
||||||
(<span title="Less or more than expected." x-text="transaction.percentage"></span>%)
|
<template x-if="transaction.percentage < 0">
|
||||||
|
<span>
|
||||||
|
(<span :title="transaction.percentage + '% {{ __("firefly.less_than_expected") }}'" x-text="transaction.percentage"></span>%)
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
<template x-if="transaction.percentage > 0">
|
||||||
|
<span>
|
||||||
|
(<span :title="transaction.percentage + '% {{ __("firefly.more_than_expected") }}'" x-text="transaction.percentage"></span>%)
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
</li>
|
</li>
|
||||||
</template>
|
</template>
|
||||||
</ul>
|
</ul>
|
||||||
|
Reference in New Issue
Block a user