This commit is contained in:
Mateusz Gruszczyński
2026-04-07 10:06:48 +02:00
parent deaa6dfe43
commit ca9c78d88d
36 changed files with 1801 additions and 503 deletions

View File

@@ -1,5 +1,7 @@
import { CommonModule, CurrencyPipe, DatePipe } from '@angular/common';
import { CommonModule } from '@angular/common';
import { Component, OnInit, computed, inject, signal } from '@angular/core';
import { ActivatedRoute, Router, RouterLink } from '@angular/router';
import { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';
import { FormArray, FormBuilder, ReactiveFormsModule, Validators } from '@angular/forms';
import { ImageCroppedEvent, ImageCropperComponent } from 'ngx-image-cropper';
import { CategoriesService } from '../../core/services/categories.service';
@@ -7,7 +9,7 @@ import { ExpensesService } from '../../core/services/expenses.service';
import { MerchantsService } from '../../core/services/merchants.service';
import { ToastService } from '../../core/services/toast.service';
import { UiService } from '../../core/services/ui.service';
import type { DuplicateGroup, Expense, Merchant, Proof } from '../../shared/models';
import type { Expense, Merchant, Proof } from '../../shared/models';
const formatLocalDate = (date: Date) => {
const year = date.getFullYear();
@@ -21,26 +23,38 @@ const today = formatLocalDate(new Date());
@Component({
selector: 'app-expenses',
standalone: true,
imports: [CommonModule, ReactiveFormsModule, CurrencyPipe, DatePipe, ImageCropperComponent],
imports: [CommonModule, ReactiveFormsModule, RouterLink, ImageCropperComponent],
template: `
<div class="page-header d-print-none mb-3 ec-page-header">
<div class="row align-items-center g-3"><div class="col"><h2 class="page-title mb-1">{{ ui.t('expenses.title') }}</h2><div class="text-secondary">{{ ui.t('expenses.subtitle') }}</div></div></div>
<div class="row align-items-center g-3">
<div class="col">
<h2 class="page-title mb-1">{{ ui.t('expenses.title') }}</h2>
<div class="text-secondary">{{ ui.t('expenses.subtitle') }}</div>
</div>
</div>
</div>
@if (duplicateGroups().length) {
<div class="alert alert-warning">
<div class="fw-semibold mb-2">{{ ui.t('expenses.duplicatesTitle') }}</div>
<div class="d-grid gap-1">@for (group of duplicateGroups().slice(0, 3); track group.source.id) { <div>{{ group.source.title }} · {{ group.matches.length }} {{ ui.t('expenses.potentialMatches') }}</div> }</div>
</div>
}
<div class="mb-3">
<nav class="nav nav-pills gap-2">
<a class="nav-link active" [routerLink]="['/expenses/add']">{{ ui.t('action.addExpense') }}</a>
<a class="nav-link" [routerLink]="['/expenses/list']">{{ ui.t('expenses.listTitle') }}</a>
</nav>
</div>
<div class="row row-cards align-items-start">
<div class="col-xl-7">
<div class="col-12">
<div class="card overflow-hidden">
<div class="card-header d-flex justify-content-between align-items-center"><h3 class="card-title">{{ editingExpenseId() ? ui.t('expenses.edit') : ui.t('expenses.new') }}</h3>@if (editingExpenseId()) { <button class="btn btn-outline-secondary btn-sm" type="button" (click)="cancelEdit()">{{ ui.t('action.cancelEdit') }}</button> }</div>
<div class="card-header d-flex justify-content-between align-items-center gap-2 flex-wrap">
<h3 class="card-title mb-0">{{ editingExpenseId() ? ui.t('expenses.edit') : ui.t('expenses.new') }}</h3>
@if (editingExpenseId()) {
<button class="btn btn-outline-secondary btn-sm" type="button" (click)="cancelEdit()">{{ ui.t('action.cancelEdit') }}</button>
}
</div>
<div class="card-body">
<form [formGroup]="expenseForm" (ngSubmit)="submitExpense()" class="d-grid gap-3" novalidate>
@if (submitted() && expenseForm.invalid) { <div class="alert alert-danger mb-0">{{ ui.t('expenses.requiredHint') }}</div> }
@if (submitted() && expenseForm.invalid) {
<div class="alert alert-danger mb-0">{{ ui.t('expenses.requiredHint') }}</div>
}
<div class="row g-3">
<div class="col-md-7"><label class="form-label">{{ ui.t('expenses.field.title') }} <span class="text-danger">*</span></label><input class="form-control" formControlName="title" [class.is-invalid]="expenseForm.controls.title.invalid && (expenseForm.controls.title.touched || submitted())" /></div>
@@ -70,25 +84,38 @@ const today = formatLocalDate(new Date());
</div>
</div></div>
@if (!editingExpenseId()) {
<div class="card bg-body-tertiary overflow-hidden"><div class="card-body d-grid gap-3">
<div class="row g-3">
<div class="col-md-4"><label class="form-label">{{ ui.t('expenses.field.proofType') }}</label><select class="form-select" formControlName="proofType"><option value="RECEIPT">{{ ui.t('proof.receipt') }}</option><option value="INVOICE">{{ ui.t('proof.invoice') }}</option><option value="NOTE">{{ ui.t('proof.note') }}</option><option value="BANK_STATEMENT">{{ ui.t('proof.statement') }}</option><option value="OTHER">{{ ui.t('proof.other') }}</option></select></div>
<div class="col-md-4"><label class="form-label">{{ ui.t('expenses.field.proofLabel') }}</label><input class="form-control" formControlName="proofLabel" /></div>
<div class="col-md-4"><label class="form-label">{{ ui.t('expenses.field.file') }}</label><input class="form-control" type="file" accept="image/*,.pdf" multiple (change)="onProofSelected($event)" /></div>
<div class="col-12"><label class="form-label">{{ ui.t('expenses.field.proofNote') }}</label><textarea class="form-control" rows="2" formControlName="proofNote"></textarea></div>
<div class="card bg-body-tertiary overflow-hidden"><div class="card-body d-grid gap-3">
<div class="row g-3">
<div class="col-md-4"><label class="form-label">{{ ui.t('expenses.field.proofType') }}</label><select class="form-select" formControlName="proofType"><option value="RECEIPT">{{ ui.t('proof.receipt') }}</option><option value="INVOICE">{{ ui.t('proof.invoice') }}</option><option value="NOTE">{{ ui.t('proof.note') }}</option><option value="BANK_STATEMENT">{{ ui.t('proof.statement') }}</option><option value="OTHER">{{ ui.t('proof.other') }}</option></select></div>
<div class="col-md-4"><label class="form-label">{{ ui.t('expenses.field.proofLabel') }}</label><input class="form-control" formControlName="proofLabel" /></div>
<div class="col-md-4"><label class="form-label">{{ ui.t('expenses.field.file') }}</label><input class="form-control" type="file" accept="image/*,.pdf" multiple (change)="onProofSelected($event)" /></div>
<div class="col-12"><label class="form-label">{{ ui.t('expenses.field.proofNote') }}</label><textarea class="form-control" rows="2" formControlName="proofNote"></textarea></div>
</div>
@if (editingExpenseId() && editingProofs().length) {
<div>
<div class="form-label">{{ ui.t('expenses.existingProofs') }}</div>
<div class="d-grid gap-2">
@for (proof of editingProofs(); track proof.id) {
<div class="d-flex justify-content-between align-items-center gap-2 border rounded-3 p-2 bg-white">
<button class="btn btn-link text-start p-0 text-decoration-none flex-grow-1" type="button" (click)="openProof(proof)">{{ proof.label || proof.originalName || ui.t('expenses.proof') }}</button>
<button class="btn btn-sm btn-outline-danger" type="button" (click)="markProofForRemoval(proof)">{{ ui.t('action.delete') }}</button>
</div>
}
</div>
</div>
@if (showCropper()) {
<div><div class="form-label">{{ ui.t('expenses.field.crop') }}</div><image-cropper [imageChangedEvent]="imageChangedEvent()" [maintainAspectRatio]="false" format="png" (imageCropped)="onImageCropped($event)"></image-cropper></div>
}
@if (croppedPreview()) {
<div><div class="form-label">{{ ui.t('expenses.field.cropPreview') }}</div><img class="img-fluid rounded" [src]="croppedPreview()" [alt]="ui.t('expenses.field.cropPreview')" /></div>
}
@if (selectedFiles().length) {
<div><div class="form-label">{{ ui.t('expenses.attachmentsSelected') }}</div><div class="d-flex flex-wrap gap-2">@for (file of selectedFiles(); track file.name + $index) { <span class="badge text-bg-secondary">{{ file.name }}</span> }</div></div>
}
</div></div>
}
}
@if (showCropper()) {
<div><div class="form-label">{{ ui.t('expenses.field.crop') }}</div><image-cropper [imageChangedEvent]="imageChangedEvent()" [maintainAspectRatio]="false" format="png" (imageCropped)="onImageCropped($event)"></image-cropper></div>
}
@if (croppedPreview()) {
<div><div class="form-label">{{ ui.t('expenses.field.cropPreview') }}</div><img class="img-fluid rounded" [src]="croppedPreview()" [alt]="ui.t('expenses.field.cropPreview')" /></div>
}
@if (selectedFiles().length) {
<div><div class="form-label">{{ ui.t('expenses.attachmentsSelected') }}</div><div class="d-flex flex-wrap gap-2">@for (file of selectedFiles(); track file.name + $index) { <span class="badge text-bg-secondary">{{ file.name }}</span> }</div></div>
}
</div></div>
<div class="btn-list flex-wrap">
<button class="btn btn-outline-secondary" type="button" (click)="submitExpense('DRAFT')" [disabled]="saving()">{{ ui.t('action.saveDraft') }}</button>
@@ -99,75 +126,12 @@ const today = formatLocalDate(new Date());
</div>
</div>
<div class="col-xl-5">
<div class="card overflow-hidden mb-3">
<div class="card-header"><h3 class="card-title">{{ ui.t('expenses.filters') }}</h3></div>
<div class="card-body"><form [formGroup]="filterForm" (ngSubmit)="loadExpenses()" class="row g-3 align-items-end">
<div class="col-sm-6"><label class="form-label">{{ ui.t('stats.from') }}</label><input class="form-control" type="date" formControlName="startDate" /></div>
<div class="col-sm-6"><label class="form-label">{{ ui.t('stats.to') }}</label><input class="form-control" type="date" formControlName="endDate" /></div>
<div class="col-sm-6"><label class="form-label">{{ ui.t('expenses.field.category') }}</label><select class="form-select" formControlName="categoryId"><option value="">{{ ui.t('expenses.allCategories') }}</option>@for (category of categories(); track category.id) { <option [value]="category.id">{{ category.name }}</option> }</select></div>
<div class="col-sm-6"><label class="form-label">{{ ui.t('expenses.field.status') }}</label><select class="form-select" formControlName="status"><option value="">{{ ui.t('common.none') }}</option><option value="DRAFT">{{ ui.t('status.draft') }}</option><option value="PENDING">{{ ui.t('status.pending') }}</option><option value="APPROVED">{{ ui.t('status.approved') }}</option><option value="REJECTED">{{ ui.t('status.rejected') }}</option></select></div>
<div class="col-sm-6"><label class="form-label">{{ ui.t('expenses.field.tags') }}</label><input class="form-control" formControlName="tags" /></div>
<div class="col-sm-6"><label class="form-label">{{ ui.t('expenses.search') }}</label><input class="form-control" formControlName="search" /></div>
<div class="col-12"><label class="form-check"><input class="form-check-input" type="checkbox" formControlName="duplicatesOnly" /><span class="form-check-label">{{ ui.t('expenses.duplicatesOnly') }}</span></label></div>
<div class="col-12 d-flex gap-2 flex-wrap"><button class="btn btn-primary" type="submit">{{ ui.t('action.filter') }}</button><button class="btn btn-outline-secondary" type="button" (click)="resetFilters()">{{ ui.t('action.reset') }}</button></div>
</form></div>
</div>
<div class="card overflow-hidden">
<div class="table-responsive">
<table class="table table-vcenter card-table mb-0">
<thead><tr><th>{{ ui.t('table.title') }}</th><th>{{ ui.t('expenses.field.status') }}</th><th class="text-end">{{ ui.t('table.amount') }}</th><th></th></tr></thead>
<tbody>
@for (item of expenses(); track item.id) {
<tr>
<td>
<div class="fw-semibold d-flex align-items-center gap-2 flex-wrap">
{{ item.title }}
@if (item.possibleDuplicate || item.duplicateStatus) {
<span class="badge" [ngClass]="duplicateBadgeClass(item)">{{ duplicateLabel(item) }}</span>
}
@if (item.recurringSourceId) {
<span class="badge text-bg-info">{{ ui.t('recurring.badge') }}</span>
}
</div>
<div class="text-secondary small">{{ item.expenseDate | date:'yyyy-MM-dd' }} · {{ item.category.name }} · {{ item.merchant || ui.t('expenses.noMerchant') }}</div>
@if (item.tags.length) { <div class="mt-1 d-flex flex-wrap gap-1">@for (tag of item.tags; track tag) { <span class="badge text-bg-secondary">#{{ tag }}</span> }</div> }
@if (customFieldEntries(item).length) { <div class="small text-secondary mt-1">@for (field of customFieldEntries(item); track field[0]) { <span class="me-2">{{ field[0] }}: {{ field[1] }}</span> }</div> }
@if (item.proofs.length) { <div class="mt-2 d-flex flex-wrap gap-2">@for (proof of item.proofs; track proof.id) { <button class="btn btn-sm btn-outline-secondary" type="button" (click)="openProof(proof)">{{ proof.label || proof.originalName || ui.t('expenses.proof') }}</button> }</div> }
</td>
<td><span class="badge" [ngClass]="statusBadgeClass(item.status)">{{ ui.t('status.' + item.status.toLowerCase()) }}</span></td>
<td class="text-end">{{ item.amount | currency:'PLN':'symbol':'1.2-2' }}</td>
<td class="text-end">
<div class="btn-list justify-content-end flex-wrap">
@if (item.possibleDuplicate && item.duplicateStatus !== 'CONFIRMED') {
<button class="btn btn-sm btn-outline-success" type="button" (click)="reviewDuplicate(item, 'CONFIRM')">OK</button>
}
@if (item.possibleDuplicate && item.duplicateStatus !== 'DISMISSED') {
<button class="btn btn-sm btn-outline-warning" type="button" (click)="reviewDuplicate(item, 'DISMISS')">X</button>
}
@if (item.duplicateStatus === 'DISMISSED' || item.duplicateStatus === 'CONFIRMED') {
<button class="btn btn-sm btn-outline-secondary" type="button" (click)="reviewDuplicate(item, 'REOPEN')">↺</button>
}
<button class="btn btn-sm btn-outline-primary" type="button" (click)="startEdit(item)">{{ ui.t('action.edit') }}</button>
<button class="btn btn-sm btn-outline-danger" type="button" (click)="removeExpense(item)">{{ ui.t('action.delete') }}</button>
</div>
</td>
</tr>
} @empty { <tr><td colspan="4" class="text-secondary">{{ ui.t('expenses.noItems') }}</td></tr> }
</tbody>
</table>
</div>
</div>
</div>
</div>
@if (merchantModalOpen()) {
<div class="modal modal-blur fade show d-block" tabindex="-1"><div class="modal-dialog modal-dialog-centered"><div class="modal-content"><div class="modal-header"><h5 class="modal-title">{{ ui.t('merchant.new') }}</h5><button class="btn-close" type="button" (click)="closeMerchantModal()"></button></div><form [formGroup]="merchantForm" (ngSubmit)="saveMerchant()"><div class="modal-body"><div class="d-grid gap-3"><div><label class="form-label">{{ ui.t('merchant.name') }}</label><input class="form-control" formControlName="name" /></div><div><label class="form-label">{{ ui.t('merchant.type') }}</label><select class="form-select" formControlName="kind"><option value="MERCHANT">{{ ui.t('merchant.kind.merchant') }}</option><option value="SERVICE_PROVIDER">{{ ui.t('merchant.kind.service') }}</option><option value="OTHER">{{ ui.t('merchant.kind.other') }}</option></select></div><div><label class="form-label">{{ ui.t('merchant.notes') }}</label><textarea class="form-control" rows="3" formControlName="notes"></textarea></div></div></div><div class="modal-footer"><button class="btn btn-ghost-secondary" type="button" (click)="closeMerchantModal()">{{ ui.t('action.cancel') }}</button><button class="btn btn-success" [disabled]="merchantForm.invalid">{{ ui.t('action.saveMerchant') }}</button></div></form></div></div></div><div class="modal-backdrop fade show"></div>
<div class="modal modal-blur fade show d-block" tabindex="-1"><div class="modal-dialog modal-dialog-centered"><div class="modal-content"><div class="modal-header"><h5 class="modal-title">{{ ui.t('merchant.new') }}</h5><button class="btn-close ec-modal-close" type="button" (click)="closeMerchantModal()"></button></div><form [formGroup]="merchantForm" (ngSubmit)="saveMerchant()"><div class="modal-body"><div class="d-grid gap-3"><div><label class="form-label">{{ ui.t('merchant.name') }}</label><input class="form-control" formControlName="name" /></div><div><label class="form-label">{{ ui.t('merchant.type') }}</label><select class="form-select" formControlName="kind"><option value="MERCHANT">{{ ui.t('merchant.kind.merchant') }}</option><option value="SERVICE_PROVIDER">{{ ui.t('merchant.kind.service') }}</option><option value="OTHER">{{ ui.t('merchant.kind.other') }}</option></select></div><div><label class="form-label">{{ ui.t('merchant.notes') }}</label><textarea class="form-control" rows="3" formControlName="notes"></textarea></div></div></div><div class="modal-footer"><button class="btn btn-ghost-secondary" type="button" (click)="closeMerchantModal()">{{ ui.t('action.cancel') }}</button><button class="btn btn-success" [disabled]="merchantForm.invalid">{{ ui.t('action.saveMerchant') }}</button></div></form></div></div></div><div class="modal-backdrop fade show"></div>
}
@if (proofPreview()) {
<div class="modal modal-blur fade show d-block" tabindex="-1"><div class="modal-dialog modal-xl modal-dialog-centered"><div class="modal-content"><div class="modal-header"><h5 class="modal-title">{{ proofPreview()?.label || proofPreview()?.originalName || ui.t('expenses.proof') }}</h5><button class="btn-close" type="button" (click)="closeProofPreview()"></button></div><div class="modal-body">@if (isPdf(proofPreview()!)) { <embed [attr.src]="proofPreview()?.fileUrl" type="application/pdf" style="width:100%;height:75vh;" /> } @else { <img class="img-fluid" [src]="proofPreview()?.fileUrl" [alt]="ui.t('expenses.proof')" /> }</div></div></div></div><div class="modal-backdrop fade show"></div>
<div class="modal modal-blur fade show d-block" tabindex="-1"><div class="modal-dialog modal-xl modal-dialog-centered"><div class="modal-content"><div class="modal-header"><h5 class="modal-title">{{ proofPreview()?.label || proofPreview()?.originalName || ui.t('expenses.proof') }}</h5><button class="btn-close ec-modal-close" type="button" (click)="closeProofPreview()"></button></div><div class="modal-body ec-proof-modal-body">@if (isPdf(proofPreview()!)) { <iframe class="ec-proof-frame" [src]="proofPreviewUrl()"></iframe> } @else { <img class="img-fluid ec-proof-preview" [src]="proofPreview()?.previewUrl || proofPreview()?.fileUrl" [alt]="ui.t('expenses.proof')" /> }</div></div></div></div><div class="modal-backdrop fade show" (click)="closeProofPreview()"></div>
}
`
})
@@ -178,22 +142,33 @@ export class ExpensesComponent implements OnInit {
private readonly merchantsService = inject(MerchantsService);
private readonly expensesService = inject(ExpensesService);
private readonly toast = inject(ToastService);
private readonly route = inject(ActivatedRoute);
private readonly router = inject(Router);
private readonly sanitizer = inject(DomSanitizer);
readonly categories = this.categoriesService.items;
readonly merchants = this.merchantsService.items;
readonly expenses = signal<Expense[]>([]);
readonly duplicateGroups = signal<DuplicateGroup[]>([]);
readonly selectedMerchantId = signal('');
readonly editingExpenseId = signal<string | null>(null);
readonly saving = signal(false);
readonly submitted = signal(false);
readonly merchantModalOpen = signal(false);
readonly proofPreview = signal<Proof | null>(null);
readonly proofPreviewUrl = computed<SafeResourceUrl | null>(() => {
const proof = this.proofPreview();
if (!proof || !this.isPdf(proof)) return null;
const previewUrl = proof.previewUrl || proof.fileUrl;
if (!previewUrl) return null;
const suffix = previewUrl.includes('#') ? '' : '#toolbar=0&navpanes=0&scrollbar=1&view=FitH';
return this.sanitizer.bypassSecurityTrustResourceUrl(`${previewUrl}${suffix}`);
});
readonly selectedFiles = signal<File[]>([]);
readonly imageChangedEvent = signal<Event | null>(null);
readonly croppedFile = signal<File | null>(null);
readonly croppedPreview = signal<string | null>(null);
readonly showCropper = signal(false);
readonly editingProofs = signal<Proof[]>([]);
readonly removedProofIds = signal<string[]>([]);
readonly expenseForm = this.fb.nonNullable.group({
title: ['', [Validators.required, Validators.minLength(2)]],
@@ -211,7 +186,6 @@ export class ExpensesComponent implements OnInit {
customFields: this.fb.array([])
});
readonly filterForm = this.fb.nonNullable.group({ startDate: [''], endDate: [''], categoryId: [''], search: [''], status: [''], tags: [''], duplicatesOnly: [false] });
readonly merchantForm = this.fb.nonNullable.group({ name: ['', [Validators.required, Validators.minLength(2)]], kind: ['MERCHANT' as Merchant['kind'], Validators.required], notes: [''] });
get customFields() { return this.expenseForm.controls.customFields as FormArray; }
@@ -220,26 +194,24 @@ export class ExpensesComponent implements OnInit {
ngOnInit() {
this.categoriesService.ensureLoaded(true);
this.merchantsService.ensureLoaded(true);
this.loadExpenses();
this.loadDuplicates();
this.route.queryParamMap.subscribe((params) => {
const editId = params.get('edit');
if (editId) this.loadExpenseForEdit(editId);
else this.cancelEdit(false);
});
}
addCustomField(key = '', value = '') { this.customFields.push(this.fb.group({ key: [key], value: [value] })); }
removeCustomField(index: number) { this.customFields.removeAt(index); }
customFieldEntries(item: Expense) { return Object.entries(item.customFields || {}); }
loadExpenses() {
const raw = this.filterForm.getRawValue();
this.expensesService.list({ startDate: raw.startDate || undefined, endDate: raw.endDate || undefined, categoryId: raw.categoryId || undefined, search: raw.search || undefined, status: raw.status || undefined, tags: raw.tags || undefined, duplicatesOnly: raw.duplicatesOnly || undefined }).subscribe({ next: (response) => this.expenses.set(response.items) });
}
loadDuplicates() {
this.expensesService.duplicates().subscribe({ next: (response) => this.duplicateGroups.set(response.items) });
}
resetFilters() {
this.filterForm.reset({ startDate: '', endDate: '', categoryId: '', search: '', status: '', tags: '', duplicatesOnly: false });
this.loadExpenses();
private loadExpenseForEdit(id: string) {
this.expensesService.getById(id).subscribe({
next: (response) => this.startEdit(response.item),
error: (error) => {
this.toast.error(error.error?.message ?? this.ui.t('expenses.saveError'));
this.cancelEdit();
}
});
}
selectMerchant(id: string) {
@@ -283,6 +255,11 @@ export class ExpensesComponent implements OnInit {
this.croppedPreview.set(event.objectUrl ?? null);
}
markProofForRemoval(proof: Proof) {
this.removedProofIds.update((ids) => Array.from(new Set([...ids, proof.id])));
this.editingProofs.update((items) => items.filter((item) => item.id !== proof.id));
}
submitExpense(forcedStatus?: Expense['status']) {
this.submitted.set(true);
this.expenseForm.markAllAsTouched();
@@ -295,21 +272,6 @@ export class ExpensesComponent implements OnInit {
const status = forcedStatus ?? (raw.status as Expense['status']);
this.saving.set(true);
if (this.editingExpenseId()) {
this.expensesService.update(this.editingExpenseId()!, { title: raw.title, amount: raw.amount, expenseDate: raw.expenseDate, categoryId: raw.categoryId, merchant: raw.merchant, paymentMethod: raw.paymentMethod as Expense['paymentMethod'], description: raw.description, currency: 'PLN', status, tags, customFields }).subscribe({
next: (response) => {
this.finishSave(response.warnings);
this.toast.success(this.ui.t('expenses.saved'));
this.cancelEdit();
},
error: (error) => {
this.saving.set(false);
this.toast.error(error.error?.message ?? this.ui.t('expenses.saveError'));
}
});
return;
}
const formData = new FormData();
formData.set('title', raw.title);
formData.set('amount', String(raw.amount));
@@ -325,6 +287,7 @@ export class ExpensesComponent implements OnInit {
formData.set('proofType', raw.proofType);
formData.set('proofLabel', raw.proofLabel);
formData.set('proofNote', raw.proofNote);
formData.set('removeProofIds', JSON.stringify(this.removedProofIds()));
const selected = this.selectedFiles();
if (this.croppedFile()) {
@@ -334,39 +297,42 @@ export class ExpensesComponent implements OnInit {
selected.forEach((file) => formData.append('proofFiles', file));
}
this.expensesService.create(formData).subscribe({
const request = this.editingExpenseId()
? this.expensesService.update(this.editingExpenseId()!, formData)
: this.expensesService.create(formData);
request.subscribe({
next: (response) => {
this.finishSave(response.warnings);
this.toast.success(status === 'DRAFT' ? this.ui.t('expenses.draftSaved') : this.ui.t('expenses.added'));
this.saving.set(false);
this.submitted.set(false);
response.warnings?.forEach((warning) => this.toast.warning(warning));
const wasEditing = Boolean(this.editingExpenseId());
this.toast.success(wasEditing ? this.ui.t('expenses.saved') : status === 'DRAFT' ? this.ui.t('expenses.draftSaved') : this.ui.t('expenses.added'));
this.resetForm();
if (wasEditing) this.router.navigate(['/expenses/add']);
},
error: (error) => {
this.saving.set(false);
this.toast.error(error.error?.message ?? this.ui.t('expenses.addError'));
this.toast.error(error.error?.message ?? (this.editingExpenseId() ? this.ui.t('expenses.saveError') : this.ui.t('expenses.addError')));
}
});
}
private finishSave(warnings?: string[]) {
this.saving.set(false);
this.submitted.set(false);
warnings?.forEach((warning) => this.toast.warning(warning));
this.resetForm();
this.loadExpenses();
this.loadDuplicates();
}
startEdit(item: Expense) {
this.editingExpenseId.set(item.id);
this.editingProofs.set(item.proofs || []);
this.removedProofIds.set([]);
this.submitted.set(false);
this.customFields.clear();
Object.entries(item.customFields || {}).forEach(([key, value]) => this.addCustomField(key, value));
this.expenseForm.patchValue({ title: item.title, amount: item.amount, expenseDate: item.expenseDate, categoryId: item.category.id, merchant: item.merchant ?? '', paymentMethod: item.paymentMethod ?? '', description: item.description ?? '', status: item.status, tagsText: (item.tags || []).join(', '), proofType: 'RECEIPT', proofLabel: '', proofNote: '' });
}
cancelEdit() {
cancelEdit(navigate = true) {
this.editingExpenseId.set(null);
this.submitted.set(false);
this.resetForm();
if (navigate) this.router.navigate(['/expenses/add']);
}
private resetForm() {
@@ -376,51 +342,14 @@ export class ExpensesComponent implements OnInit {
this.selectedFiles.set([]);
this.croppedFile.set(null);
this.croppedPreview.set(null);
this.imageChangedEvent.set(null);
this.showCropper.set(false);
}
removeExpense(item: Expense) {
this.expensesService.delete(item.id).subscribe({
next: () => {
this.toast.success(this.ui.t('expenses.deleted'));
this.loadExpenses();
this.loadDuplicates();
},
error: (error) => this.toast.error(error.error?.message ?? this.ui.t('expenses.deleteError'))
});
}
reviewDuplicate(item: Expense, action: 'CONFIRM' | 'DISMISS' | 'REOPEN') {
this.expensesService.reviewDuplicate(item.id, action).subscribe({
next: () => {
if (action === 'CONFIRM') this.toast.success(this.ui.t('expenses.duplicateConfirmed'));
if (action === 'DISMISS') this.toast.success(this.ui.t('expenses.duplicateDismissed'));
if (action === 'REOPEN') this.toast.success(this.ui.t('expenses.duplicateReopened'));
this.loadExpenses();
this.loadDuplicates();
},
error: (error) => this.toast.error(error.error?.message ?? this.ui.t('toast.error'))
});
this.editingProofs.set([]);
this.removedProofIds.set([]);
}
openProof(proof: Proof) { this.proofPreview.set(proof); }
closeMerchantModal() { this.merchantModalOpen.set(false); }
closeProofPreview() { this.proofPreview.set(null); }
isPdf(proof: Proof) { return (proof.mimeType || '').includes('pdf'); }
statusBadgeClass(status: string) {
return ({ DRAFT: 'text-bg-secondary', PENDING: 'text-bg-warning', APPROVED: 'text-bg-success', REJECTED: 'text-bg-danger' } as Record<string, string>)[status] || 'text-bg-secondary';
}
duplicateBadgeClass(item: Expense) {
const state = item.duplicateStatus ?? (item.possibleDuplicate ? 'OPEN' : null);
return ({ OPEN: 'text-bg-warning', CONFIRMED: 'text-bg-danger', DISMISSED: 'text-bg-secondary' } as Record<string, string>)[state || 'OPEN'] || 'text-bg-warning';
}
duplicateLabel(item: Expense) {
const state = item.duplicateStatus ?? (item.possibleDuplicate ? 'OPEN' : null);
if (state === 'CONFIRMED') return this.ui.t('expenses.duplicateStatus.confirmed');
if (state === 'DISMISSED') return this.ui.t('expenses.duplicateStatus.dismissed');
return this.ui.t('expenses.duplicateStatus.open');
}
isPdf(proof: Proof) { return (proof.mimeType || '').toLowerCase().includes('pdf'); }
}