From 0c2028cde4f3b4f1f1cd90c90b30769ea78fd90c Mon Sep 17 00:00:00 2001 From: Suyash Patil <127177049+suyashpatil78@users.noreply.github.com> Date: Mon, 11 Mar 2024 22:19:27 +0530 Subject: [PATCH] feat: removed old my-expenses page (deprecation) (#2814) * feat: removed old my-expenses page (deprecation) * feat: removed old my-expenses page (deprecation) - Part 2 (#2815) * feat: removed old my-expenses page (deprecation) - Part 2 * feat: removed old my-expenses page (deprecation) - Part 3 (#2816) * feat: removed old my-expenses page (deprecation) - Part 3 * feat: removed old my-expenses page (deprecation) - Part 4 (#2817) * feat: removed old my-expenses page (deprecation) - Part 4 * fix: correction in card filters logic (#2818) * major pr comments as well as merge * setting branch coverage to 91% --- .github/workflows/unit-tests.yml | 2 +- .../guards/my-expenses-guard.guard.spec.ts | 78 - .../core/guards/my-expenses-guard.guard.ts | 28 - .../core/mock-data/expense-filters.data.ts | 2 +- .../core/mock-data/modal-controller.data.ts | 12 +- .../models}/expense-filters.model.ts | 2 +- .../models/platform/expense-filters.model.ts | 2 +- .../models/report-filters.model.ts} | 13 - .../v1/shared/expenses.service.spec.ts | 4 +- .../platform/v1/shared/expenses.service.ts | 9 +- src/app/core/services/tracking.service.ts | 4 +- src/app/core/services/transaction.service.ts | 2 +- .../dashboard/stats/stats.component.spec.ts | 26 - .../fyle/dashboard/stats/stats.component.ts | 29 +- .../dashboard/tasks/tasks-3.component.spec.ts | 2 +- .../fyle/dashboard/tasks/tasks.component.ts | 2 +- src/app/fyle/fyle-routing.module.ts | 11 - .../add-txn-to-report-dialog.component.html | 64 - .../add-txn-to-report-dialog.component.scss | 63 - ...add-txn-to-report-dialog.component.spec.ts | 131 - .../add-txn-to-report-dialog.component.ts | 42 - .../my-expenses-routing.module.ts | 17 - .../my-expenses-v2/my-expenses-v2.module.ts | 45 - .../my-expenses-v2/my-expenses-v2.page.html | 265 -- .../my-expenses-v2/my-expenses-v2.page.scss | 329 -- .../my-expenses-v2.page.spec.ts | 3071 ----------------- .../my-expenses-v2/my-expenses-v2.page.ts | 1650 --------- .../my-expenses.service.spec.ts | 521 --- .../my-expenses-v2/my-expenses.service.ts | 530 --- .../add-expense-popover.component.html | 0 .../my-expenses/my-expenses-filters.model.ts | 13 - .../fyle/my-expenses/my-expenses.page.html | 23 +- .../fyle/my-expenses/my-expenses.page.spec.ts | 1442 ++++---- src/app/fyle/my-expenses/my-expenses.page.ts | 756 ++-- .../my-expenses/my-expenses.service.spec.ts | 50 +- .../fyle/my-expenses/my-expenses.service.ts | 56 +- .../fyle/my-reports/my-reports.page.spec.ts | 2 +- .../card-detail/card-detail.component.spec.ts | 11 +- .../card-detail/card-detail.component.ts | 25 +- 39 files changed, 1423 insertions(+), 7911 deletions(-) delete mode 100644 src/app/core/guards/my-expenses-guard.guard.spec.ts delete mode 100644 src/app/core/guards/my-expenses-guard.guard.ts rename src/app/{fyle/my-expenses => core/models}/expense-filters.model.ts (75%) rename src/app/{fyle/my-expenses-v2/my-expenses-filters.model.ts => core/models/report-filters.model.ts} (51%) delete mode 100644 src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.html delete mode 100644 src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.scss delete mode 100644 src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.spec.ts delete mode 100644 src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.ts delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses-routing.module.ts delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses-v2.module.ts delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses-v2.page.html delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses-v2.page.scss delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses-v2.page.spec.ts delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses-v2.page.ts delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses.service.spec.ts delete mode 100644 src/app/fyle/my-expenses-v2/my-expenses.service.ts rename src/app/fyle/{my-expenses-v2 => my-expenses}/add-expense-popover/add-expense-popover.component.html (100%) diff --git a/.github/workflows/unit-tests.yml b/.github/workflows/unit-tests.yml index 6f5d1eb646..5bf94db8f3 100644 --- a/.github/workflows/unit-tests.yml +++ b/.github/workflows/unit-tests.yml @@ -44,7 +44,7 @@ jobs: if (( $(echo "$lines < 95.0" | bc -l) || \ $(echo "$statements < 95.0" | bc -l) || \ - $(echo "$branches < 94.0" | bc -l) || \ + $(echo "$branches < 91.0" | bc -l) || \ $(echo "$functions < 94.0" | bc -l) )); then echo "Code Coverage Percentage is below 95%" exit 1 diff --git a/src/app/core/guards/my-expenses-guard.guard.spec.ts b/src/app/core/guards/my-expenses-guard.guard.spec.ts deleted file mode 100644 index ab6efdcd6a..0000000000 --- a/src/app/core/guards/my-expenses-guard.guard.spec.ts +++ /dev/null @@ -1,78 +0,0 @@ -import { TestBed } from '@angular/core/testing'; - -import { MyExpensesGuardGuard } from './my-expenses-guard.guard'; -import { ActivatedRoute, Router } from '@angular/router'; -import { OrgSettingsService } from '../services/org-settings.service'; -import { Observable, of } from 'rxjs'; -import { orgSettingsWithV2ExpensesPage, orgSettingsWoV2ExpensesPage } from '../mock-data/org-settings.data'; - -describe('MyExpensesGuardGuard', () => { - let guard: MyExpensesGuardGuard; - let router: jasmine.SpyObj; - let activatedRoute: jasmine.SpyObj; - let orgSettingsSerivce: jasmine.SpyObj; - - beforeEach(() => { - const orgSettingsSerivceSpy = jasmine.createSpyObj('OrgSettingsService', ['get']); - const routerSpy = jasmine.createSpyObj('Router', ['navigate']); - - TestBed.configureTestingModule({ - providers: [ - { - provide: OrgSettingsService, - useValue: orgSettingsSerivceSpy, - }, - { - provide: Router, - useValue: routerSpy, - }, - { - provide: ActivatedRoute, - useValue: { - snapshot: { - data: { - url: '/enterprise/dashboard', - root: null, - }, - }, - }, - }, - ], - }); - guard = TestBed.inject(MyExpensesGuardGuard); - router = TestBed.inject(Router) as jasmine.SpyObj; - orgSettingsSerivce = TestBed.inject(OrgSettingsService) as jasmine.SpyObj; - activatedRoute = TestBed.inject(ActivatedRoute) as jasmine.SpyObj; - }); - - it('should be created', () => { - expect(guard).toBeTruthy(); - }); - - describe('canActivate():', () => { - it('should navigate to v2 page if settings is enabled', (done) => { - orgSettingsSerivce.get.and.returnValue(of(orgSettingsWithV2ExpensesPage)); - - const result = guard.canActivate(activatedRoute.snapshot, { url: '/test', root: null }) as Observable; - - result.subscribe(() => { - expect(orgSettingsSerivce.get).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses_v2']); - done(); - }); - }); - - it('should navigate to old page if setting is disbaled', (done) => { - orgSettingsSerivce.get.and.returnValue(of(orgSettingsWoV2ExpensesPage)); - - const result = guard.canActivate(activatedRoute.snapshot, { url: '/test', root: null }) as Observable; - - result.subscribe((res) => { - expect(res).toBeTrue(); - expect(orgSettingsSerivce.get).toHaveBeenCalledTimes(1); - expect(router.navigate).not.toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses_v2']); - done(); - }); - }); - }); -}); diff --git a/src/app/core/guards/my-expenses-guard.guard.ts b/src/app/core/guards/my-expenses-guard.guard.ts deleted file mode 100644 index 1770ccc7aa..0000000000 --- a/src/app/core/guards/my-expenses-guard.guard.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { Injectable } from '@angular/core'; -import { ActivatedRouteSnapshot, CanActivate, Router, RouterStateSnapshot, UrlTree } from '@angular/router'; -import { Observable, map, tap } from 'rxjs'; -import { OrgSettingsService } from '../services/org-settings.service'; - -import { OrgSettings } from '../models/org-settings.model'; - -@Injectable({ - providedIn: 'root', -}) -export class MyExpensesGuardGuard implements CanActivate { - constructor(private orgSettingsSerivce: OrgSettingsService, private router: Router) {} - - canActivate( - route: ActivatedRouteSnapshot, - state: RouterStateSnapshot - ): Observable | Promise | boolean | UrlTree { - return this.orgSettingsSerivce.get().pipe( - map((orgSettings: OrgSettings) => { - if (orgSettings.mobile_app_my_expenses_beta_enabled) { - this.router.navigate(['/', 'enterprise', 'my_expenses_v2']); - } - - return true; - }) - ); - } -} diff --git a/src/app/core/mock-data/expense-filters.data.ts b/src/app/core/mock-data/expense-filters.data.ts index 7970e9e150..643bbc5732 100644 --- a/src/app/core/mock-data/expense-filters.data.ts +++ b/src/app/core/mock-data/expense-filters.data.ts @@ -1,7 +1,7 @@ -import { ExpenseFilters } from 'src/app/fyle/my-expenses/expense-filters.model'; import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; import { ExpenseType } from '../enums/expense-type.enum'; import { FilterState } from '../enums/filter-state.enum'; +import { ExpenseFilters } from '../models/expense-filters.model'; export const expenseFiltersData1: Partial = { state: ['DRAFT', 'READY_TO_REPORT'], diff --git a/src/app/core/mock-data/modal-controller.data.ts b/src/app/core/mock-data/modal-controller.data.ts index 395a575fc7..f404ca377d 100644 --- a/src/app/core/mock-data/modal-controller.data.ts +++ b/src/app/core/mock-data/modal-controller.data.ts @@ -6,8 +6,7 @@ import { CreateNewReportComponent as createReportV2 } from 'src/app/shared/compo import { CreateNewReportComponent } from 'src/app/shared/components/create-new-report/create-new-report.component'; import { Mode } from '@ionic/core'; import { fyModalProperties } from './model-properties.data'; -import { AddTxnToReportDialogComponent } from 'src/app/fyle/my-expenses/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; -import { AddTxnToReportDialogComponent as v2 } from 'src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; +import { AddTxnToReportDialogComponent as v2 } from 'src/app/fyle/my-expenses/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; import { PopupAlertComponent } from 'src/app/shared/components/popup-alert/popup-alert.component'; import { FilterOptions } from 'src/app/shared/components/fy-filters/filter-options.interface'; import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; @@ -75,15 +74,6 @@ export const newReportModalParams2 = { ...fyModalProperties, }; -export const addExpenseToReportModalParams = { - component: AddTxnToReportDialogComponent, - componentProps: { - txId: '12345', - }, - mode: 'ios', - ...fyModalProperties, -}; - export const addExpenseToReportModalParams2 = { component: v2, componentProps: { diff --git a/src/app/fyle/my-expenses/expense-filters.model.ts b/src/app/core/models/expense-filters.model.ts similarity index 75% rename from src/app/fyle/my-expenses/expense-filters.model.ts rename to src/app/core/models/expense-filters.model.ts index b66825de9b..aece467047 100644 --- a/src/app/fyle/my-expenses/expense-filters.model.ts +++ b/src/app/core/models/expense-filters.model.ts @@ -1,4 +1,4 @@ -import { Filters } from './my-expenses-filters.model'; +import { Filters } from 'src/app/fyle/my-expenses/my-expenses-filters.model'; export interface ExpenseFilters extends Omit { state: string | string[]; diff --git a/src/app/core/models/platform/expense-filters.model.ts b/src/app/core/models/platform/expense-filters.model.ts index 61b127f46e..cd64ceb917 100644 --- a/src/app/core/models/platform/expense-filters.model.ts +++ b/src/app/core/models/platform/expense-filters.model.ts @@ -1,4 +1,4 @@ -import { Filters } from 'src/app/fyle/my-expenses-v2/my-expenses-filters.model'; +import { Filters } from 'src/app/fyle/my-expenses/my-expenses-filters.model'; export interface ExpenseFilters extends Omit { state: string | string[]; diff --git a/src/app/fyle/my-expenses-v2/my-expenses-filters.model.ts b/src/app/core/models/report-filters.model.ts similarity index 51% rename from src/app/fyle/my-expenses-v2/my-expenses-filters.model.ts rename to src/app/core/models/report-filters.model.ts index ddd0353bc5..80d21e62cd 100644 --- a/src/app/fyle/my-expenses-v2/my-expenses-filters.model.ts +++ b/src/app/core/models/report-filters.model.ts @@ -1,16 +1,3 @@ -export type Filters = Partial<{ - state: string[]; - date: string; - customDateStart: Date; - customDateEnd: Date; - receiptsAttached: string; - type: string[]; - sortParam: string; - sortDir: string; - cardNumbers: string[]; - splitExpense: string; -}>; - export type ReportFilters = Partial<{ state: string | string[]; date: string; diff --git a/src/app/core/services/platform/v1/shared/expenses.service.spec.ts b/src/app/core/services/platform/v1/shared/expenses.service.spec.ts index 496a344956..32e807a99b 100644 --- a/src/app/core/services/platform/v1/shared/expenses.service.spec.ts +++ b/src/app/core/services/platform/v1/shared/expenses.service.spec.ts @@ -28,6 +28,8 @@ import { import { ExpenseState } from 'src/app/core/models/expense-state.enum'; import { AccountType } from 'src/app/core/models/platform/v1/account.model'; import { Expense } from 'src/app/core/models/platform/v1/expense.model'; +import { GetExpenseQueryParam } from 'src/app/core/models/platform/v1/get-expenses-query.model'; +import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; import { ExpensesService } from './expenses.service'; import { cloneDeep } from 'lodash'; import { DateService } from '../../../date.service'; @@ -470,7 +472,7 @@ describe('ExpensesService', () => { const result = service.generateCardNumberParams({}, expenseFiltersData1); expect(result).toEqual({ - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(1234,5678)', + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.("1234","5678")', }); }); diff --git a/src/app/core/services/platform/v1/shared/expenses.service.ts b/src/app/core/services/platform/v1/shared/expenses.service.ts index 3677c970df..9ed030708f 100644 --- a/src/app/core/services/platform/v1/shared/expenses.service.ts +++ b/src/app/core/services/platform/v1/shared/expenses.service.ts @@ -9,9 +9,9 @@ import { PaymentModeSummary } from 'src/app/core/models/payment-mode-summary.mod import { AccountType } from 'src/app/core/models/platform/v1/account.model'; import { Expense } from 'src/app/core/models/platform/v1/expense.model'; import { GetExpenseQueryParam } from 'src/app/core/models/platform/v1/get-expenses-query.model'; -import { ExpenseFilters } from 'src/app/fyle/my-expenses/expense-filters.model'; import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; import { DateService } from '../../../date.service'; +import { ExpenseFilters } from 'src/app/core/models/expense-filters.model'; @Injectable({ providedIn: 'root', @@ -200,11 +200,8 @@ export class ExpensesService { ): Record { const newQueryParamsCopy = cloneDeep(newQueryParams); if (filters.cardNumbers?.length > 0) { - let cardNumberString = ''; - cardNumberString = filters.cardNumbers.join(','); - cardNumberString = cardNumberString.slice(0, cardNumberString.length); - newQueryParamsCopy['matched_corporate_card_transactions->0->corporate_card_number'] = - 'in.(' + cardNumberString + ')'; + const cardNumberString = filters.cardNumbers.map((cardNumber) => `"${cardNumber}"`).join(','); + newQueryParamsCopy['matched_corporate_card_transactions->0->corporate_card_number'] = `in.(${cardNumberString})`; } return newQueryParamsCopy; diff --git a/src/app/core/services/tracking.service.ts b/src/app/core/services/tracking.service.ts index d20ed841ad..08a04754da 100644 --- a/src/app/core/services/tracking.service.ts +++ b/src/app/core/services/tracking.service.ts @@ -31,12 +31,12 @@ import { EnrollingNonRTFCardProperties, } from '../models/tracking-properties.model'; import { ExpenseView } from '../models/expense-view.enum'; -import { ExpenseFilters } from 'src/app/fyle/my-expenses/expense-filters.model'; -import { ReportFilters } from 'src/app/fyle/my-expenses-v2/my-expenses-filters.model'; import { TaskFilters } from '../models/task-filters.model'; import { OrgCategory } from '../models/v1/org-category.model'; import { TeamReportsFilters } from '../models/team-reports-filters.model'; import { forkJoin, from } from 'rxjs'; +import { ExpenseFilters } from '../models/expense-filters.model'; +import { ReportFilters } from '../models/report-filters.model'; @Injectable({ providedIn: 'root', diff --git a/src/app/core/services/transaction.service.ts b/src/app/core/services/transaction.service.ts index 9bf7d8030a..f19112d952 100644 --- a/src/app/core/services/transaction.service.ts +++ b/src/app/core/services/transaction.service.ts @@ -18,7 +18,6 @@ import { UserEventService } from './user-event.service'; import { UndoMerge } from '../models/undo-merge.model'; import { cloneDeep } from 'lodash'; import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; -import { ExpenseFilters } from 'src/app/fyle/my-expenses/expense-filters.model'; import { PAGINATION_SIZE } from 'src/app/constants'; import { PaymentModesService } from './payment-modes.service'; import { OrgSettingsService } from './org-settings.service'; @@ -42,6 +41,7 @@ import { PlatformMissingMandatoryFieldsResponse } from '../models/platform/platf import { AccountType } from '../enums/account-type.enum'; import { Expense as PlatformExpense } from '../models/platform/v1/expense.model'; import { CorporateCardTransactionRes } from '../models/platform/v1/corporate-card-transaction-res.model'; +import { ExpenseFilters } from '../models/expense-filters.model'; enum FilterState { READY_TO_REPORT = 'READY_TO_REPORT', diff --git a/src/app/fyle/dashboard/stats/stats.component.spec.ts b/src/app/fyle/dashboard/stats/stats.component.spec.ts index 138f1cd04f..ae385b4f72 100644 --- a/src/app/fyle/dashboard/stats/stats.component.spec.ts +++ b/src/app/fyle/dashboard/stats/stats.component.spec.ts @@ -356,7 +356,6 @@ describe('StatsComponent', () => { describe('goToExpensesPage():', () => { it('goToExpensesPage(): should navigate to expenses page with query params', () => { - component.redirectToNewPage$ = of(false); component.goToExpensesPage('COMPLETE'); expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses'], { @@ -368,7 +367,6 @@ describe('StatsComponent', () => { }); it('goToExpensesPage(): should navigate to expenses page with query params', () => { - component.redirectToNewPage$ = of(false); component.goToExpensesPage('INCOMPLETE'); expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses'], { @@ -378,30 +376,6 @@ describe('StatsComponent', () => { }); expect(trackingService.dashboardOnIncompleteExpensesClick).toHaveBeenCalledTimes(1); }); - - it('goToExpensesPage(): should navigate to v2 expenses page with query params', () => { - component.redirectToNewPage$ = of(true); - component.goToExpensesPage('COMPLETE'); - - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses_v2'], { - queryParams: { - filters: JSON.stringify({ state: ['READY_TO_REPORT'] }), - }, - }); - expect(trackingService.dashboardOnUnreportedExpensesClick).toHaveBeenCalledTimes(1); - }); - - it('goToExpensesPage(): should navigate to v2 expenses page with query params', () => { - component.redirectToNewPage$ = of(true); - component.goToExpensesPage('INCOMPLETE'); - - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses_v2'], { - queryParams: { - filters: JSON.stringify({ state: ['DRAFT'] }), - }, - }); - expect(trackingService.dashboardOnIncompleteExpensesClick).toHaveBeenCalledTimes(1); - }); }); describe('trackDashboardLaunchTime():', () => { diff --git a/src/app/fyle/dashboard/stats/stats.component.ts b/src/app/fyle/dashboard/stats/stats.component.ts index cef00620d9..47db5fe852 100644 --- a/src/app/fyle/dashboard/stats/stats.component.ts +++ b/src/app/fyle/dashboard/stats/stats.component.ts @@ -200,23 +200,20 @@ export class StatsComponent implements OnInit { } goToExpensesPage(state: string): void { - this.redirectToNewPage$.subscribe((redirect) => { - const endpoint = redirect ? 'my_expenses_v2' : 'my_expenses'; - if (state === 'COMPLETE') { - const queryParams: Params = { filters: JSON.stringify({ state: ['READY_TO_REPORT'] }) }; - this.router.navigate(['/', 'enterprise', endpoint], { - queryParams, - }); + if (state === 'COMPLETE') { + const queryParams: Params = { filters: JSON.stringify({ state: ['READY_TO_REPORT'] }) }; + this.router.navigate(['/', 'enterprise', 'my_expenses'], { + queryParams, + }); - this.trackingService.dashboardOnUnreportedExpensesClick(); - } else { - const queryParams: Params = { filters: JSON.stringify({ state: ['DRAFT'] }) }; - this.router.navigate(['/', 'enterprise', endpoint], { - queryParams, - }); - this.trackingService.dashboardOnIncompleteExpensesClick(); - } - }); + this.trackingService.dashboardOnUnreportedExpensesClick(); + } else { + const queryParams: Params = { filters: JSON.stringify({ state: ['DRAFT'] }) }; + this.router.navigate(['/', 'enterprise', 'my_expenses'], { + queryParams, + }); + this.trackingService.dashboardOnIncompleteExpensesClick(); + } } private trackDashboardLaunchTime(): void { diff --git a/src/app/fyle/dashboard/tasks/tasks-3.component.spec.ts b/src/app/fyle/dashboard/tasks/tasks-3.component.spec.ts index 544c2abfaa..9e1fd2fcff 100644 --- a/src/app/fyle/dashboard/tasks/tasks-3.component.spec.ts +++ b/src/app/fyle/dashboard/tasks/tasks-3.component.spec.ts @@ -19,7 +19,7 @@ import { cloneDeep, noop } from 'lodash'; import { snackbarPropertiesRes2 } from 'src/app/core/mock-data/snackbar-properties.data'; import { ToastMessageComponent } from 'src/app/shared/components/toast-message/toast-message.component'; import { ToastType } from 'src/app/core/enums/toast-type.enum'; -import { AddTxnToReportDialogComponent } from '../../my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; +import { AddTxnToReportDialogComponent } from '../../my-expenses/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; import { ExpensesService } from 'src/app/core/services/platform/v1/spender/expenses.service'; import { expenseData } from 'src/app/core/mock-data/platform/v1/expense.data'; import { unreportedExpensesQueryParams } from 'src/app/core/mock-data/platform/v1/expenses-query-params.data'; diff --git a/src/app/fyle/dashboard/tasks/tasks.component.ts b/src/app/fyle/dashboard/tasks/tasks.component.ts index 53fa8a7b76..63a3fd0a24 100644 --- a/src/app/fyle/dashboard/tasks/tasks.component.ts +++ b/src/app/fyle/dashboard/tasks/tasks.component.ts @@ -22,7 +22,7 @@ import { FilterOptionType } from 'src/app/shared/components/fy-filters/filter-op import { FilterOptions } from 'src/app/shared/components/fy-filters/filter-options.interface'; import { FyFiltersComponent } from 'src/app/shared/components/fy-filters/fy-filters.component'; import { ToastMessageComponent } from 'src/app/shared/components/toast-message/toast-message.component'; -import { AddTxnToReportDialogComponent } from '../../my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; +import { AddTxnToReportDialogComponent } from '../../my-expenses/add-txn-to-report-dialog/add-txn-to-report-dialog.component'; import { FilterPill } from 'src/app/shared/components/fy-filter-pills/filter-pill.interface'; import { SelectedFilters } from 'src/app/shared/components/fy-filters/selected-filters.interface'; import { ExpensesService } from 'src/app/core/services/platform/v1/spender/expenses.service'; diff --git a/src/app/fyle/fyle-routing.module.ts b/src/app/fyle/fyle-routing.module.ts index bb5eb72463..dc504f58f5 100644 --- a/src/app/fyle/fyle-routing.module.ts +++ b/src/app/fyle/fyle-routing.module.ts @@ -1,21 +1,14 @@ import { NgModule } from '@angular/core'; import { Routes, RouterModule } from '@angular/router'; -import { MyExpensesGuardGuard } from '../core/guards/my-expenses-guard.guard'; -import { BetaPageFeatureFlagGuard } from '../core/guards/beta-page-feature-flag.guard'; const routes: Routes = [ { path: 'my_dashboard', loadChildren: () => import('./dashboard/dashboard.module').then((m) => m.DashboardPageModule), }, - { - path: 'my_expenses_v2', - loadChildren: () => import('./my-expenses-v2/my-expenses-v2.module').then((m) => m.MyExpensesV2PageModule), - }, { path: 'my_expenses', loadChildren: () => import('./my-expenses/my-expenses.module').then((m) => m.MyExpensesPageModule), - canActivate: [MyExpensesGuardGuard], }, { path: 'my_advances', @@ -142,10 +135,6 @@ const routes: Routes = [ loadChildren: () => import('./manage-corporate-cards/manage-corporate-cards.module').then((m) => m.ManageCorporateCardsPageModule), }, - { - path: 'my-expenses', - loadChildren: () => import('./my-expenses/my-expenses.module').then((m) => m.MyExpensesPageModule), - }, ]; export const fyleRoutes = routes; diff --git a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.html b/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.html deleted file mode 100644 index f0f87e0172..0000000000 --- a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.html +++ /dev/null @@ -1,64 +0,0 @@ - -
- - - - - - Add to Report - - - - - -
-
-
-
-
-
-
- - - -
{{ report.rp_purpose }}
-
- {{ report.rp_num_transactions }} Expense{{ report.rp_num_transactions > 1 ? 's' : '' }} -
-
- -
- {{ reportCurrencySymbol }} - {{ - report.rp_amount || 0 | humanizeCurrency : report.rp_currency : true - }} -
-
-
- {{ report.rp_state | reportState : data.isNewReportsFlowEnabled | snakeCaseToSpaceCase | titlecase }} -
-
-
-
-
-
-
-
- - - -
diff --git a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.scss b/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.scss deleted file mode 100644 index 54c7dc3ebe..0000000000 --- a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.scss +++ /dev/null @@ -1,63 +0,0 @@ -@import '../../../../theme/colors.scss'; -$reports_sent_back_color: #da1e28; - -.report-list { - &--header { - font-size: 24px; - line-height: 1.3; - color: $black; - margin-bottom: 24px; - font-weight: 500; - - &--row-icon-container { - display: flex; - align-items: center; - justify-content: center; - } - } - - &--list { - margin-top: 10px; - } - - &--currency-amount-container { - margin-bottom: 4px; - } - - &--currency { - color: $black-light; - font-weight: 500; - font-size: 14px; - line-height: 1.3; - margin-right: 2px; - } - - &--amount { - color: $black; - font-size: 20px; - line-height: 1.3; - font-weight: 500; - } - - &--purpose { - font-weight: 500; - color: $black; - font-size: 18px; - line-height: 1.3; - } - - &--count { - color: $blue-black; - line-height: 1.3; - margin-top: 2px; - } - - &--divider { - border-bottom: 1px solid $grey-lighter; - margin-bottom: 14px; - } - - &--state { - margin-left: auto; - } -} diff --git a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.spec.ts b/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.spec.ts deleted file mode 100644 index 0612a3497b..0000000000 --- a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.spec.ts +++ /dev/null @@ -1,131 +0,0 @@ -import { CurrencyPipe } from '@angular/common'; -import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing'; -import { MatBottomSheet, MatBottomSheetModule, MAT_BOTTOM_SHEET_DATA } from '@angular/material/bottom-sheet'; -import { Router, RouterModule } from '@angular/router'; -import { RouterTestingModule } from '@angular/router/testing'; -import { IonicModule } from '@ionic/angular'; -import { of } from 'rxjs'; -import { click, getElementBySelector, getTextContent } from 'src/app/core/dom-helpers'; -import { apiExtendedReportRes } from 'src/app/core/mock-data/report.data'; -import { CurrencyService } from 'src/app/core/services/currency.service'; -import { FyZeroStateComponent } from 'src/app/shared/components/fy-zero-state/fy-zero-state.component'; -import { FyCurrencyPipe } from 'src/app/shared/pipes/fy-currency.pipe'; -import { HumanizeCurrencyPipe } from 'src/app/shared/pipes/humanize-currency.pipe'; -import { ReportState } from 'src/app/shared/pipes/report-state.pipe'; -import { SnakeCaseToSpaceCase } from 'src/app/shared/pipes/snake-case-to-space-case.pipe'; -import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog.component'; - -describe('AddTxnToReportDialogComponent', () => { - let component: AddTxnToReportDialogComponent; - let fixture: ComponentFixture; - let currencyService: jasmine.SpyObj; - let matBottomsheet: jasmine.SpyObj; - let router: jasmine.SpyObj; - - beforeEach(waitForAsync(() => { - const routerSpy = jasmine.createSpyObj('Router', ['navigate']); - const currencyServiceSpy = jasmine.createSpyObj('CurrencyService', ['getHomeCurrency']); - const matBottomsheetSpy = jasmine.createSpyObj('MatBottomSheet', ['dismiss']); - TestBed.configureTestingModule({ - declarations: [ - AddTxnToReportDialogComponent, - FyZeroStateComponent, - HumanizeCurrencyPipe, - ReportState, - SnakeCaseToSpaceCase, - ], - imports: [IonicModule.forRoot(), RouterTestingModule, RouterModule, MatBottomSheetModule], - providers: [ - FyCurrencyPipe, - CurrencyPipe, - { - provide: Router, - useValue: routerSpy, - }, - { - provide: CurrencyService, - useValue: currencyServiceSpy, - }, - { - provide: MatBottomSheet, - useValue: matBottomsheetSpy, - }, - { - provide: MAT_BOTTOM_SHEET_DATA, - useValue: { openReports: apiExtendedReportRes, isNewReportsFlowEnabled: true }, - }, - ], - }).compileComponents(); - - fixture = TestBed.createComponent(AddTxnToReportDialogComponent); - component = fixture.componentInstance; - router = TestBed.inject(Router) as jasmine.SpyObj; - currencyService = TestBed.inject(CurrencyService) as jasmine.SpyObj; - matBottomsheet = TestBed.inject(MatBottomSheet) as jasmine.SpyObj; - currencyService.getHomeCurrency.and.returnValue(of('USD')); - fixture.detectChanges(); - })); - - it('should create', () => { - expect(component).toBeTruthy(); - }); - - it('closeAddToReportModal(): should close Add To Report modal', () => { - matBottomsheet.dismiss.and.callThrough(); - - component.closeAddToReportModal(); - expect(matBottomsheet.dismiss).toHaveBeenCalledTimes(1); - }); - - it('addTransactionToReport(): should add txn to report', () => { - matBottomsheet.dismiss.and.callThrough(); - - component.addTransactionToReport(apiExtendedReportRes[0]); - expect(matBottomsheet.dismiss).toHaveBeenCalledOnceWith({ report: apiExtendedReportRes[0] }); - }); - - it('onClickCreateReportTask(): should navigate to create report page', () => { - matBottomsheet.dismiss.and.callThrough(); - router.navigate.and.callThrough(); - - component.onClickCreateReportTask(); - expect(matBottomsheet.dismiss).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_create_report']); - }); - - it('should display report information correctly', () => { - component.openReports = [apiExtendedReportRes[0]]; - fixture.detectChanges(); - - expect(getTextContent(getElementBySelector(fixture, '.report-list--purpose'))).toEqual('#8: Jan 2023'); - expect(getTextContent(getElementBySelector(fixture, '.report-list--count'))).toEqual('1 Expense'); - expect(getTextContent(getElementBySelector(fixture, '.report-list--currency'))).toEqual('$'); - expect(getTextContent(getElementBySelector(fixture, '.report-list--amount'))).toEqual('116.90'); - expect(getTextContent(getElementBySelector(fixture, '.report-list--state'))).toEqual('Submitted'); - }); - - it('should call addTransactionToReport() when clicked', () => { - spyOn(component, 'addTransactionToReport'); - component.openReports = [apiExtendedReportRes[0]]; - - const reportCard = getElementBySelector(fixture, '[data-testid="report"]') as HTMLElement; - click(reportCard); - expect(component.addTransactionToReport).toHaveBeenCalledOnceWith(apiExtendedReportRes[0]); - }); - - it('should call closeAddToReportModal() when clicked', () => { - spyOn(component, 'closeAddToReportModal'); - - const closeIcon = getElementBySelector(fixture, '.report-list--header--row-icon') as HTMLElement; - click(closeIcon); - expect(component.closeAddToReportModal).toHaveBeenCalledTimes(1); - }); - - it('should call onClickCreateReportTask() when clicked', () => { - spyOn(component, 'onClickCreateReportTask'); - - const addToReportButton = getElementBySelector(fixture, '[data-testid="addIcon"]') as HTMLElement; - click(addToReportButton); - expect(component.onClickCreateReportTask).toHaveBeenCalledTimes(1); - }); -}); diff --git a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.ts b/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.ts deleted file mode 100644 index 919be030a0..0000000000 --- a/src/app/fyle/my-expenses-v2/add-txn-to-report-dialog/add-txn-to-report-dialog.component.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { Component, OnInit, Input, Inject } from '@angular/core'; -import { getCurrencySymbol } from '@angular/common'; -import { MatBottomSheet, MAT_BOTTOM_SHEET_DATA } from '@angular/material/bottom-sheet'; -import { ExtendedReport } from 'src/app/core/models/report.model'; -import { CurrencyService } from 'src/app/core/services/currency.service'; -import { Router } from '@angular/router'; -@Component({ - selector: 'app-add-txn-to-report-dialog', - templateUrl: './add-txn-to-report-dialog.component.html', - styleUrls: ['./add-txn-to-report-dialog.component.scss'], -}) -export class AddTxnToReportDialogComponent implements OnInit { - @Input() openReports; - - reportCurrencySymbol: string; - - constructor( - private currencyService: CurrencyService, - @Inject(MAT_BOTTOM_SHEET_DATA) public data: { openReports: ExtendedReport[]; isNewReportsFlowEnabled: boolean }, - private matBottomsheet: MatBottomSheet, - private router: Router - ) {} - - closeAddToReportModal() { - this.matBottomsheet.dismiss(); - } - - onClickCreateReportTask() { - this.matBottomsheet.dismiss(); - this.router.navigate(['/', 'enterprise', 'my_create_report']); - } - - addTransactionToReport(report: ExtendedReport) { - this.matBottomsheet.dismiss({ report }); - } - - ngOnInit() { - this.currencyService.getHomeCurrency().subscribe((homeCurrency) => { - this.reportCurrencySymbol = getCurrencySymbol(homeCurrency, 'wide'); - }); - } -} diff --git a/src/app/fyle/my-expenses-v2/my-expenses-routing.module.ts b/src/app/fyle/my-expenses-v2/my-expenses-routing.module.ts deleted file mode 100644 index a777dd0063..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses-routing.module.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { NgModule } from '@angular/core'; -import { Routes, RouterModule } from '@angular/router'; - -import { MyExpensesV2Page } from './my-expenses-v2.page'; - -const routes: Routes = [ - { - path: '', - component: MyExpensesV2Page, - }, -]; - -@NgModule({ - imports: [RouterModule.forChild(routes)], - exports: [RouterModule], -}) -export class MyExpensesV2PageRoutingModule {} diff --git a/src/app/fyle/my-expenses-v2/my-expenses-v2.module.ts b/src/app/fyle/my-expenses-v2/my-expenses-v2.module.ts deleted file mode 100644 index 9f265f0071..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses-v2.module.ts +++ /dev/null @@ -1,45 +0,0 @@ -import { NgModule } from '@angular/core'; -import { CommonModule } from '@angular/common'; -import { FormsModule, ReactiveFormsModule } from '@angular/forms'; -import { IonicModule } from '@ionic/angular'; -import { MyExpensesV2PageRoutingModule } from './my-expenses-routing.module'; -import { MyExpensesV2Page } from './my-expenses-v2.page'; -import { MatFormFieldModule } from '@angular/material/form-field'; -import { MatInputModule } from '@angular/material/input'; -import { MatButtonModule } from '@angular/material/button'; -import { MatRadioModule } from '@angular/material/radio'; -import { MatNativeDateModule, MatRippleModule } from '@angular/material/core'; -import { MatDatepickerModule } from '@angular/material/datepicker'; -import { MatIconModule } from '@angular/material/icon'; -import { SharedModule } from 'src/app/shared/shared.module'; -import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog/add-txn-to-report-dialog.component'; -import { MatBottomSheetModule } from '@angular/material/bottom-sheet'; -import { MatSnackBarModule } from '@angular/material/snack-bar'; -import { MatCheckboxModule } from '@angular/material/checkbox'; - -@NgModule({ - imports: [ - CommonModule, - FormsModule, - IonicModule, - MyExpensesV2PageRoutingModule, - MatInputModule, - MatFormFieldModule, - MatButtonModule, - MatRadioModule, - FormsModule, - ReactiveFormsModule, - MatRippleModule, - MatDatepickerModule, - MatNativeDateModule, - MatIconModule, - MatButtonModule, - MatCheckboxModule, - MatBottomSheetModule, - MatSnackBarModule, - SharedModule, - MatCheckboxModule, - ], - declarations: [MyExpensesV2Page, AddTxnToReportDialogComponent], -}) -export class MyExpensesV2PageModule {} diff --git a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.html b/src/app/fyle/my-expenses-v2/my-expenses-v2.page.html deleted file mode 100644 index 1e2a7754c4..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.html +++ /dev/null @@ -1,265 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - -
- -
-
-
-
-
- - - - - - - - - {{allExpensesStats?.count}} {{(allExpensesStats?.count > 1) ? 'expenses' : 'expense'}} worth - {{(allExpensesStats.amount || 0) | humanizeCurrency: homeCurrency }} - - - - - - - - - - - - - - - - Select All - - - -
- - -
- - - - - -
- -
-
- - - -
- Create First Expenses -
- -

Ready to add your first expense?

-
Click a picture of a receipt or add the details yourself.
-
- -
-

Ready to add your expenses?

-
Click a picture of a receipt or add the details yourself.
-
-
-
- -
-
No Expenses Found
-
No expenses were found for the selected filter.
-
-
- - -
- -
-
- - -
- -
-
-
- -
-
- - -
-
- - -
- - -
-
- - - - - -
- - - - - - - - -
- - - - - - - - - - - - diff --git a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.scss b/src/app/fyle/my-expenses-v2/my-expenses-v2.page.scss deleted file mode 100644 index e375ac68e8..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.scss +++ /dev/null @@ -1,329 +0,0 @@ -@import '../../../theme/colors'; - -.my-expenses { - &--toolbar-header { - border-bottom: 1px solid #e6e6e6; - } - - &--header-btn { - --padding-start: 8px !important; - --padding-end: 8px !important; - } - - &--header-btn--skeleton-loader { - margin: 12px 8px; - border-radius: 5px; - width: 24px; - height: 24px; - } - - &--filter-pills { - position: fixed; - width: 100%; - background-color: $white; - z-index: 99; - } - - &--filter-pills-container { - display: block; - min-height: 52px; - } - - &--toolbar { - align-items: center; - display: inline-flex; - margin: 8px 16px; - width: 90%; - } - - &--multiselect-title { - padding-left: 12px; - padding-right: 0px; - } - - &--content { - --background: #{$white}; - --padding-bottom: 82px; - } - - &--select-all-container { - --border-style: none; - } - - &--state-toolbar { - margin-bottom: -2px !important; - } - - &--state { - display: flex; - justify-content: center; - margin: 0 8px; - border: none; - } - - &--stats { - color: black; - position: relative; - top: 8px; - } - - &--amount { - font-size: 24px; - font-weight: 500; - } - - &--homeCurrency { - font-size: 18px; - font-weight: 500; - } - - &--count { - font-size: 16px; - font-weight: 500; - color: #ababab; - margin-bottom: 12px; - } - - &--cta-container { - justify-content: space-between; - display: flex; - width: 100%; - } - - &--shimmers { - overflow-y: scroll; - } - - &--state-pill { - width: 47%; - border: 1px solid $blue; - padding: 8px; - text-align: center; - color: $blue; - white-space: nowrap; - &__left { - border-radius: 4px 0 0 4px; - } - - &__right { - border-radius: 0 4px 4px 0; - } - - &__selected { - background-color: $blue; - color: white; - } - } - - &--simple-search-container { - display: flex; - align-items: center; - height: 62px; - padding-left: 12px; - mat-form-field { - ::ng-deep .mat-form-field-outline { - color: $black-light !important; - } - } - } - - &--simple-search { - width: 100%; - font-size: 14px; - - &-icon { - font-size: 18px; - margin: auto 8px auto 0; - color: $black-light; - height: auto; - } - - &-close { - margin: -10px; - font-size: 18px; - } - } - - &--simple-search-block { - display: flex; - vertical-align: baseline; - } - - &--footer-container { - background: white; - padding: 16px; - } - - &--filters { - margin: 0 16px; - padding: 8px; - box-shadow: rgba(50, 50, 93, 0.25) 0px 2px 5px -1px, rgba(0, 0, 0, 0.3) 0px 1px 3px -1px; - border-radius: 4px; - background-color: #5c98e5; - color: white; - display: flex; - justify-content: space-between; - align-items: center; - } - - &--filter-button { - box-shadow: rgba(50, 50, 93, 0.25) 0px 2px 5px -1px, rgba(0, 0, 0, 0.3) 0px 1px 3px -1px; - border-radius: 4px; - margin-left: 9px; - - &-icon { - fill: #1978f7; - } - - &__applied { - ::after { - content: '•'; - font-size: 48px; - position: absolute; - top: -8px; - color: #f36; - } - } - } - - &--offline-header { - font-size: 16px; - margin-bottom: 0; - line-height: 1; - font-weight: 500; - padding: 0 12px; - } - - &--offline-sub-header { - color: #4a4a4a; - line-height: 1.5; - font-size: 12px; - padding: 0 12px; - } - - &--offline-message { - padding: 8px 0; - } - - &--syncing { - margin: 16px 16px 0 16px; - } - - &--zero-state { - display: flex; - flex-direction: column; - height: 90%; - justify-content: center; - text-align: center; - - &-img { - margin: 0 20px 20px 20px; - } - - &-header { - margin: 12px; - font-size: 18px; - display: block; - margin-bottom: 8px; - color: $dark-blue; - line-height: 1.5; - font-weight: normal; - } - - &-content { - color: $blue-black; - font-size: 14px; - font-weight: normal; - } - - &-subheader { - color: $gray-3; - display: block; - font-size: 16px; - font-weight: 300; - line-height: 1.5; - padding: 0 8px; - - &__title { - color: #4a4a4a; - font-weight: 500; - } - } - } - - &--needs-receipt { - &-header { - color: #4a4a4a; - font-size: 20px; - padding-bottom: 12px; - font-weight: 500; - } - - &-sub-header { - font-size: 16px; - font-weight: 500; - line-height: 1.43; - color: #ababab; - margin-bottom: 16px; - } - } - - &--body { - padding-bottom: 20px; - background-color: $white; - } - - &--zero-states-body { - height: 100%; - } - - &--footer-toolbar { - --min-height: 120px; - } - - &--footer-conatiner { - display: flex; - justify-content: space-around; - padding: 14px 36px; - position: fixed; - bottom: 0; - width: 100%; - background-color: $pure-white; - box-shadow: 0 -2px 12px $pink-shadow; - z-index: 999; - } - - &--cta-button { - font-weight: 500; - width: 100%; - } - - &--secondary-cta { - height: auto; - background-color: transparent; - color: $black; - font-size: 16px; - line-height: 1.25; - &__disabled { - opacity: 0.2; - } - } - - &--primary-cta { - min-height: 52px; - background: $pink-gradient; - color: $pure-white; - font-size: 14px; - line-height: 1.3; - border-radius: 4px; - width: 100%; - - &__disabled { - opacity: 0.8; - } - } - - &--merge-icon { - color: $brand-primary; - } -} diff --git a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.spec.ts b/src/app/fyle/my-expenses-v2/my-expenses-v2.page.spec.ts deleted file mode 100644 index 13f9e3c9fc..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.spec.ts +++ /dev/null @@ -1,3071 +0,0 @@ -import { ComponentFixture, TestBed, fakeAsync, tick, waitForAsync } from '@angular/core/testing'; -import { ActionSheetController, IonicModule, ModalController, NavController, PopoverController } from '@ionic/angular'; - -import { HttpClientTestingModule } from '@angular/common/http/testing'; -import { NO_ERRORS_SCHEMA } from '@angular/core'; -import { MatBottomSheet, MatBottomSheetRef } from '@angular/material/bottom-sheet'; -import { MatSnackBar, MatSnackBarRef } from '@angular/material/snack-bar'; -import { By } from '@angular/platform-browser'; -import { ActivatedRoute, Router } from '@angular/router'; -import { RouterTestingModule } from '@angular/router/testing'; -import { clone, cloneDeep } from 'lodash'; -import { BehaviorSubject, Subscription, finalize, noop, of, tap, throwError } from 'rxjs'; -import { getElementRef } from 'src/app/core/dom-helpers'; -import { - expectedActionSheetButtonRes, - expectedActionSheetButtonsWithMileage, - expectedActionSheetButtonsWithPerDiem, -} from 'src/app/core/mock-data/action-sheet-options.data'; -import { allowedExpenseTypes } from 'src/app/core/mock-data/allowed-expense-types.data'; -import { apiAuthRes } from 'src/app/core/mock-data/auth-reponse.data'; -import { cardAggregateStatParam } from 'src/app/core/mock-data/card-aggregate-stats.data'; -import { expectedAssignedCCCStats } from 'src/app/core/mock-data/ccc-expense.details.data'; -import { - expectedCriticalPolicyViolationPopoverParams, - expectedCriticalPolicyViolationPopoverParams2, - expectedCriticalPolicyViolationPopoverParams3, -} from 'src/app/core/mock-data/critical-policy-violation-popover.data'; -import { expenseFiltersData1, expenseFiltersData2 } from 'src/app/core/mock-data/expense-filters.data'; -import { - apiExpenseRes, - expectedFormattedTransaction, - expenseData1, - expenseData2, - expenseList4, - expenseListwithoutID, -} from 'src/app/core/mock-data/expense.data'; -import { - cardFilterPill, - creditTxnFilterPill, - dateFilterPill, - expectedFilterPill1, - expectedFilterPill2, - filterTypeMappings, - receiptsAttachedFilterPill, - sortFilterPill, - splitExpenseFilterPill, - stateFilterPill, - typeFilterPill, -} from 'src/app/core/mock-data/filter-pills.data'; -import { filterOptions1 } from 'src/app/core/mock-data/filter.data'; -import { - expectedCurrentParamsCannotReportState, - expectedCurrentParamsDraftState, - expectedCurrentParamsWithDraftCannotReportState, - expectedCurrentParamsWoFilterState, -} from 'src/app/core/mock-data/get-expenses-query-params-with-filters.data'; -import { - addExpenseToReportModalParams, - addExpenseToReportModalParams2, - modalControllerParams, - modalControllerParams2, - newReportModalParams, - newReportModalParams2, - openFromComponentConfig, - popoverControllerParams, -} from 'src/app/core/mock-data/modal-controller.data'; -import { fyModalProperties } from 'src/app/core/mock-data/model-properties.data'; -import { mileagePerDiemPlatformCategoryData } from 'src/app/core/mock-data/org-category.data'; -import { orgSettingsParamsWithSimplifiedReport, orgSettingsRes } from 'src/app/core/mock-data/org-settings.data'; -import { orgUserSettingsData } from 'src/app/core/mock-data/org-user-settings.data'; -import { - apiExpenses1, - apiExpenses2, - expenseData, - mileageExpenseWithDistance, - perDiemExpenseWithSingleNumDays, -} from 'src/app/core/mock-data/platform/v1/expense.data'; -import { reportUnflattenedData } from 'src/app/core/mock-data/report-v1.data'; -import { apiExtendedReportRes, expectedReportSingleResponse } from 'src/app/core/mock-data/report.data'; -import { selectedFilters1, selectedFilters2 } from 'src/app/core/mock-data/selected-filters.data'; -import { - snackbarPropertiesRes, - snackbarPropertiesRes2, - snackbarPropertiesRes3, - snackbarPropertiesRes4, -} from 'src/app/core/mock-data/snackbar-properties.data'; -import { transactionDatum1, transactionDatum3 } from 'src/app/core/mock-data/stats-response.data'; -import { txnList } from 'src/app/core/mock-data/transaction.data'; -import { unflattenedTxnData } from 'src/app/core/mock-data/unflattened-txn.data'; -import { unformattedTxnData } from 'src/app/core/mock-data/unformatted-transaction.data'; -import { expectedUniqueCardStats } from 'src/app/core/mock-data/unique-cards-stats.data'; -import { uniqueCardsParam } from 'src/app/core/mock-data/unique-cards.data'; -import { AdvancesStates } from 'src/app/core/models/advances-states.model'; -import { BackButtonActionPriority } from 'src/app/core/models/back-button-action-priority.enum'; -import { Expense } from 'src/app/core/models/expense.model'; -import { ExtendedReport } from 'src/app/core/models/report.model'; -import { ApiV2Service } from 'src/app/core/services/api-v2.service'; -import { CategoriesService } from 'src/app/core/services/categories.service'; -import { CorporateCreditCardExpenseService } from 'src/app/core/services/corporate-credit-card-expense.service'; -import { CurrencyService } from 'src/app/core/services/currency.service'; -import { LoaderService } from 'src/app/core/services/loader.service'; -import { ModalPropertiesService } from 'src/app/core/services/modal-properties.service'; -import { NetworkService } from 'src/app/core/services/network.service'; -import { OrgSettingsService } from 'src/app/core/services/org-settings.service'; -import { OrgUserSettingsService } from 'src/app/core/services/org-user-settings.service'; -import { PlatformHandlerService } from 'src/app/core/services/platform-handler.service'; -import { ExpensesService as SharedExpenseService } from 'src/app/core/services/platform/v1/shared/expenses.service'; -import { ExpensesService } from 'src/app/core/services/platform/v1/spender/expenses.service'; -import { PopupService } from 'src/app/core/services/popup.service'; -import { ReportService } from 'src/app/core/services/report.service'; -import { SnackbarPropertiesService } from 'src/app/core/services/snackbar-properties.service'; -import { StorageService } from 'src/app/core/services/storage.service'; -import { TasksService } from 'src/app/core/services/tasks.service'; -import { TokenService } from 'src/app/core/services/token.service'; -import { TrackingService } from 'src/app/core/services/tracking.service'; -import { TransactionService } from 'src/app/core/services/transaction.service'; -import { TransactionsOutboxService } from 'src/app/core/services/transactions-outbox.service'; -import { FyDeleteDialogComponent } from 'src/app/shared/components/fy-delete-dialog/fy-delete-dialog.component'; -import { HeaderState } from 'src/app/shared/components/fy-header/header-state.enum'; -import { ToastMessageComponent } from 'src/app/shared/components/toast-message/toast-message.component'; -import { MaskNumber } from 'src/app/shared/pipes/mask-number.pipe'; -import { ReportState } from 'src/app/shared/pipes/report-state.pipe'; -import { environment } from 'src/environments/environment'; -import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog/add-txn-to-report-dialog.component'; -import { MyExpensesV2Page } from './my-expenses-v2.page'; -import { MyExpensesService } from './my-expenses.service'; -import { completeStats, incompleteStats } from 'src/app/core/mock-data/platform/v1/expenses-stats.data'; - -describe('MyExpensesV2Page', () => { - let component: MyExpensesV2Page; - let fixture: ComponentFixture; - let tasksService: jasmine.SpyObj; - let currencyService: jasmine.SpyObj; - let reportService: jasmine.SpyObj; - let apiV2Service: jasmine.SpyObj; - let transactionService: jasmine.SpyObj; - let orgSettingsService: jasmine.SpyObj; - let activatedRoute: jasmine.SpyObj; - let router: jasmine.SpyObj; - let navController: jasmine.SpyObj; - let networkService: jasmine.SpyObj; - let transactionOutboxService: jasmine.SpyObj; - let matBottomsheet: jasmine.SpyObj; - let matSnackBar: jasmine.SpyObj; - let myExpenseService: jasmine.SpyObj; - let tokenService: jasmine.SpyObj; - let actionSheetController: jasmine.SpyObj; - let modalProperties: jasmine.SpyObj; - let storageService: jasmine.SpyObj; - let corporateCreditCardService: jasmine.SpyObj; - let orgUserSettingsService: jasmine.SpyObj; - let categoriesService: jasmine.SpyObj; - let platformHandlerService: jasmine.SpyObj; - let trackingService: jasmine.SpyObj; - let modalController: jasmine.SpyObj; - let loaderService: jasmine.SpyObj; - let popupService: jasmine.SpyObj; - let popoverController: jasmine.SpyObj; - let snackbarProperties: jasmine.SpyObj; - let inputElement: HTMLInputElement; - let sharedExpenseService: jasmine.SpyObj; - let expensesService: jasmine.SpyObj; - - beforeEach(waitForAsync(() => { - const tasksServiceSpy = jasmine.createSpyObj('TasksService', ['getReportsTaskCount', 'getExpensesTaskCount']); - const currencyServiceSpy = jasmine.createSpyObj('CurrencyService', ['getHomeCurrency']); - const reportServiceSpy = jasmine.createSpyObj('ReportService', [ - 'getMyReportsCount', - 'getMyReports', - 'clearTransactionCache', - 'getAllExtendedReports', - 'addTransactions', - ]); - const apiV2ServiceSpy = jasmine.createSpyObj('ApiV2Service', ['extendQueryParamsForTextSearch']); - const transactionServiceSpy = jasmine.createSpyObj('TransactionService', [ - 'getTransactionStats', - 'getMyExpensesCount', - 'getMyExpenses', - 'getPaginatedETxncCount', - 'clearCache', - 'generateCardNumberParams', - 'generateDateParams', - 'generateReceiptAttachedParams', - 'generateStateFilters', - 'generateTypeFilters', - 'setSortParams', - 'generateSplitExpenseParams', - 'delete', - 'getReportableExpenses', - 'isMergeAllowed', - 'excludeCCCExpenses', - 'isCriticalPolicyViolatedExpense', - 'isExpenseInDraft', - 'getETxnUnflattened', - 'getAllExpenses', - 'getDeleteDialogBody', - 'getExpenseDeletionMessage', - 'getCCCExpenseMessage', - 'deleteBulk', - ]); - const orgSettingsServiceSpy = jasmine.createSpyObj('OrgSettingsService', ['get']); - const categoriesServiceSpy = jasmine.createSpyObj('CategoriesService', ['getMileageOrPerDiemCategories']); - const navControllerSpy = jasmine.createSpyObj('NavController', ['back']); - const networkServiceSpy = jasmine.createSpyObj('NetworkService', ['isOnline', 'connectivityWatcher']); - const activatedRouteSpy = { - snapshot: { - params: { - navigateBack: false, - }, - }, - }; - const transactionOutboxServiceSpy = jasmine.createSpyObj('TransactionOutboxService', [ - 'getPendingTransactions', - 'sync', - 'deleteOfflineExpense', - 'deleteBulkOfflineExpenses', - ]); - const matBottomsheetSpy = jasmine.createSpyObj('MatBottomSheet', ['dismiss', 'open']); - const matSnackBarSpy = jasmine.createSpyObj('MatSnackBar', ['openFromComponent']); - const myExpensesServiceSpy = jasmine.createSpyObj('MyExpensesService', [ - 'generateStateFilterPills', - 'generateReceiptsAttachedFilterPills', - 'generateDateFilterPills', - 'generateTypeFilterPills', - 'generateSortFilterPills', - 'generateCardFilterPills', - 'generateSplitExpenseFilterPills', - 'convertFilters', - 'generateSelectedFilters', - 'getFilters', - 'convertSelectedOptionsToExpenseFilters', - ]); - const tokenServiceSpy = jasmine.createSpyObj('TokenService', ['getClusterDomain']); - const actionSheetControllerSpy = jasmine.createSpyObj('ActionSheetController', ['create']); - const modalPropertiesSpy = jasmine.createSpyObj('ModalPropertiesService', ['getModalDefaultProperties']); - const storageServiceSpy = jasmine.createSpyObj('StorageService', ['get', 'set', 'post']); - const corporateCreditCardServiceSpy = jasmine.createSpyObj('CorporateCreditCardExpenseService', [ - 'getExpenseDetailsInCards', - 'getAssignedCards', - ]); - const orgUserSettingsServiceSpy = jasmine.createSpyObj('OrgUserSettingsService', ['get']); - const platformHandlerServiceSpy = jasmine.createSpyObj('PlatformHandlerService', ['registerBackButtonAction']); - const trackingServiceSpy = jasmine.createSpyObj('TrackingService', [ - 'createFirstExpense', - 'myExpensesActionSheetAction', - 'tasksPageOpened', - 'footerHomeTabClicked', - 'myExpensesFilterApplied', - 'deleteExpense', - 'clickAddToReport', - 'showToastMessage', - 'addToReport', - 'clickCreateReport', - 'myExpensesBulkDeleteExpenses', - 'spenderSelectedPendingTxnFromMyExpenses', - ]); - const modalControllerSpy = jasmine.createSpyObj('ModalController', ['create']); - const loaderServiceSpy = jasmine.createSpyObj('LoaderService', ['showLoader', 'hideLoader']); - const popupServiceSpy = jasmine.createSpyObj('PopupService', ['showPopup']); - const popoverControllerSpy = jasmine.createSpyObj('PopoverController', ['create']); - const snackbarPropertiesSpy = jasmine.createSpyObj('SnackbarPropertiesService', ['setSnackbarProperties']); - const expensesServiceSpy = jasmine.createSpyObj('ExpensesService', [ - 'getExpensesCount', - 'getExpenses', - 'getAllExpenses', - 'getExpenseById', - 'getExpenseStats', - ]); - const sharedExpenseServiceSpy = jasmine.createSpyObj('SharedExpenseService', [ - 'generateCardNumberParams', - 'generateDateParams', - 'generateReceiptAttachedParams', - 'generateStateFilters', - 'generateTypeFilters', - 'setSortParams', - 'generateSplitExpenseParams', - 'getReportableExpenses', - 'excludeCCCExpenses', - 'isMergeAllowed', - 'isCriticalPolicyViolatedExpense', - 'isExpenseInDraft', - 'getExpenseDeletionMessage', - 'getCCCExpenseMessage', - 'getDeleteDialogBody', - 'restrictPendingTransactionsEnabled', - 'doesExpenseHavePendingCardTransaction', - ]); - - TestBed.configureTestingModule({ - declarations: [MyExpensesV2Page, ReportState, MaskNumber], - imports: [IonicModule.forRoot(), RouterTestingModule, HttpClientTestingModule], - providers: [ - { provide: TasksService, useValue: tasksServiceSpy }, - { provide: CurrencyService, useValue: currencyServiceSpy }, - { provide: ReportService, useValue: reportServiceSpy }, - { provide: ApiV2Service, useValue: apiV2ServiceSpy }, - { provide: TransactionService, useValue: transactionServiceSpy }, - { provide: OrgSettingsService, useValue: orgSettingsServiceSpy }, - { provide: ActivatedRoute, useValue: activatedRouteSpy }, - { provide: Router, useValue: jasmine.createSpyObj('Router', ['navigate', 'createUrlTree']) }, - { - provide: NavController, - useValue: navControllerSpy, - }, - { - provide: NetworkService, - useValue: networkServiceSpy, - }, - { - provide: ReportService, - useValue: reportServiceSpy, - }, - { - provide: TransactionsOutboxService, - useValue: transactionOutboxServiceSpy, - }, - { - provide: MatBottomSheet, - useValue: matBottomsheetSpy, - }, - { - provide: MatSnackBar, - useValue: matSnackBarSpy, - }, - { - provide: MyExpensesService, - useValue: myExpensesServiceSpy, - }, - { - provide: TokenService, - useValue: tokenServiceSpy, - }, - { - provide: ActionSheetController, - useValue: actionSheetControllerSpy, - }, - { - provide: ModalPropertiesService, - useValue: modalPropertiesSpy, - }, - { - provide: StorageService, - useValue: storageServiceSpy, - }, - { - provide: CorporateCreditCardExpenseService, - useValue: corporateCreditCardServiceSpy, - }, - { - provide: OrgUserSettingsService, - useValue: orgUserSettingsServiceSpy, - }, - { - provide: PlatformHandlerService, - useValue: platformHandlerServiceSpy, - }, - { - provide: TrackingService, - useValue: trackingServiceSpy, - }, - { - provide: ModalController, - useValue: modalControllerSpy, - }, - { - provide: LoaderService, - useValue: loaderServiceSpy, - }, - { - provide: PopupService, - useValue: popupServiceSpy, - }, - { - provide: PopoverController, - useValue: popoverControllerSpy, - }, - { - provide: SnackbarPropertiesService, - useValue: snackbarPropertiesSpy, - }, - { - provide: CategoriesService, - useValue: categoriesServiceSpy, - }, - { - provide: ExpensesService, - useValue: expensesServiceSpy, - }, - { - provide: SharedExpenseService, - useValue: sharedExpenseServiceSpy, - }, - ReportState, - MaskNumber, - ], - schemas: [NO_ERRORS_SCHEMA], - }).compileComponents(); - - fixture = TestBed.createComponent(MyExpensesV2Page); - component = fixture.componentInstance; - - activatedRoute = TestBed.inject(ActivatedRoute) as jasmine.SpyObj; - activatedRoute.snapshot.params = {}; - activatedRoute.snapshot.queryParams = {}; - router = TestBed.inject(Router) as jasmine.SpyObj; - navController = TestBed.inject(NavController) as jasmine.SpyObj; - currencyService = TestBed.inject(CurrencyService) as jasmine.SpyObj; - reportService = TestBed.inject(ReportService) as jasmine.SpyObj; - tasksService = TestBed.inject(TasksService) as jasmine.SpyObj; - orgSettingsService = TestBed.inject(OrgSettingsService) as jasmine.SpyObj; - categoriesService = TestBed.inject(CategoriesService) as jasmine.SpyObj; - apiV2Service = TestBed.inject(ApiV2Service) as jasmine.SpyObj; - transactionService = TestBed.inject(TransactionService) as jasmine.SpyObj; - networkService = TestBed.inject(NetworkService) as jasmine.SpyObj; - reportService = TestBed.inject(ReportService) as jasmine.SpyObj; - transactionOutboxService = TestBed.inject(TransactionsOutboxService) as jasmine.SpyObj; - matBottomsheet = TestBed.inject(MatBottomSheet) as jasmine.SpyObj; - matSnackBar = TestBed.inject(MatSnackBar) as jasmine.SpyObj; - myExpenseService = TestBed.inject(MyExpensesService) as jasmine.SpyObj; - tokenService = TestBed.inject(TokenService) as jasmine.SpyObj; - actionSheetController = TestBed.inject(ActionSheetController) as jasmine.SpyObj; - modalProperties = TestBed.inject(ModalPropertiesService) as jasmine.SpyObj; - storageService = TestBed.inject(StorageService) as jasmine.SpyObj; - corporateCreditCardService = TestBed.inject( - CorporateCreditCardExpenseService - ) as jasmine.SpyObj; - orgUserSettingsService = TestBed.inject(OrgUserSettingsService) as jasmine.SpyObj; - platformHandlerService = TestBed.inject(PlatformHandlerService) as jasmine.SpyObj; - trackingService = TestBed.inject(TrackingService) as jasmine.SpyObj; - modalController = TestBed.inject(ModalController) as jasmine.SpyObj; - loaderService = TestBed.inject(LoaderService) as jasmine.SpyObj; - popupService = TestBed.inject(PopupService) as jasmine.SpyObj; - popoverController = TestBed.inject(PopoverController) as jasmine.SpyObj; - snackbarProperties = TestBed.inject(SnackbarPropertiesService) as jasmine.SpyObj; - expensesService = TestBed.inject(ExpensesService) as jasmine.SpyObj; - sharedExpenseService = TestBed.inject(SharedExpenseService) as jasmine.SpyObj; - - component.loadExpenses$ = new BehaviorSubject({}); - })); - - it('should create', () => { - expect(component).toBeTruthy(); - }); - - it('ngOnInit(): should invoke setupNetworkWatcher', () => { - spyOn(component, 'setupNetworkWatcher'); - component.ngOnInit(); - expect(component.setupNetworkWatcher).toHaveBeenCalledTimes(1); - }); - - describe('ionViewWillEnter(): ', () => { - let backButtonSubscription: Subscription; - - beforeEach(() => { - component.isConnected$ = of(true); - backButtonSubscription = new Subscription(); - tasksService.getExpensesTaskCount.and.returnValue(of(10)); - platformHandlerService.registerBackButtonAction.and.returnValue(backButtonSubscription); - orgUserSettingsService.get.and.returnValue(of(orgUserSettingsData)); - orgSettingsService.get.and.returnValue(of(orgSettingsRes)); - categoriesService.getMileageOrPerDiemCategories.and.returnValue(of(mileagePerDiemPlatformCategoryData)); - corporateCreditCardService.getAssignedCards.and.returnValue(of(expectedAssignedCCCStats)); - spyOn(component, 'getCardDetail').and.returnValue(expectedUniqueCardStats); - spyOn(component, 'syncOutboxExpenses'); - spyOn(component, 'setAllExpensesCountAndAmount'); - spyOn(component, 'clearFilters'); - spyOn(component, 'setupActionSheet'); - tokenService.getClusterDomain.and.resolveTo(apiAuthRes.cluster_domain); - currencyService.getHomeCurrency.and.returnValue(of('USD')); - expensesService.getExpenseStats.and.returnValue(of(completeStats)); - expensesService.getExpensesCount.and.returnValue(of(10)); - expensesService.getExpenses.and.returnValue(of(apiExpenses1)); - - reportService.getAllExtendedReports.and.returnValue(of(apiExtendedReportRes)); - spyOn(component, 'doRefresh'); - spyOn(component, 'backButtonAction'); - - spyOn(component, 'formatTransactions').and.returnValue(apiExpenseRes); - spyOn(component, 'addNewFiltersToParams').and.returnValue({ pageNumber: 1, sortDir: 'desc' }); - spyOn(component, 'generateFilterPills').and.returnValue(creditTxnFilterPill); - component.simpleSearchInput = getElementRef(fixture, '.my-expenses--simple-search-input'); - inputElement = component.simpleSearchInput.nativeElement; - }); - - it('should set isNewReportsFlowEnabled, isInstaFyleEnabled, isBulkFyleEnabled, isMileageEnabled and isPerDiemEnabled to true if orgSettings and orgUserSettings properties are enabled', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - expect(component.expensesTaskCount).toBe(10); - expect(component.isNewReportsFlowEnabled).toBeFalse(); - - expect(orgUserSettingsService.get).toHaveBeenCalledTimes(1); - expect(orgSettingsService.get).toHaveBeenCalledTimes(1); - component.isInstaFyleEnabled$.subscribe((isInstaFyleEnabled) => { - expect(isInstaFyleEnabled).toBeTrue(); - }); - component.isBulkFyleEnabled$.subscribe((isBulkFyleEnabled) => { - expect(isBulkFyleEnabled).toBeTrue(); - }); - component.isMileageEnabled$.subscribe((isMileageEnabled) => { - expect(isMileageEnabled).toBeTrue(); - }); - component.isPerDiemEnabled$.subscribe((isPerDiemEnabled) => { - expect(isPerDiemEnabled).toBeTrue(); - }); - })); - - it('should set isNewReportsFlowEnabled, isInstaFyleEnabled, isBulkFyleEnabled, isMileageEnabled and isPerDiemEnabled to false if orgSettings and orgUserSettings properties are disabled', fakeAsync(() => { - const mockOrgUserSettingsData = cloneDeep(orgUserSettingsData); - const mockOrgSettingsData = cloneDeep(orgSettingsRes); - mockOrgUserSettingsData.insta_fyle_settings.enabled = false; - mockOrgUserSettingsData.bulk_fyle_settings.enabled = false; - mockOrgSettingsData.mileage.enabled = false; - mockOrgSettingsData.per_diem.enabled = false; - orgUserSettingsService.get.and.returnValue(of(mockOrgUserSettingsData)); - orgSettingsService.get.and.returnValue(of(mockOrgSettingsData)); - - component.ionViewWillEnter(); - tick(500); - expect(component.expensesTaskCount).toBe(10); - expect(component.isNewReportsFlowEnabled).toBeFalse(); - - expect(orgUserSettingsService.get).toHaveBeenCalledTimes(1); - expect(orgSettingsService.get).toHaveBeenCalledTimes(1); - component.isInstaFyleEnabled$.subscribe((isInstaFyleEnabled) => { - expect(isInstaFyleEnabled).toBeFalse(); - }); - component.isBulkFyleEnabled$.subscribe((isBulkFyleEnabled) => { - expect(isBulkFyleEnabled).toBeFalse(); - }); - component.isMileageEnabled$.subscribe((isMileageEnabled) => { - expect(isMileageEnabled).toBeFalse(); - }); - component.isPerDiemEnabled$.subscribe((isPerDiemEnabled) => { - expect(isPerDiemEnabled).toBeFalse(); - }); - })); - - it('should set isNewReportsFlowEnabled, isInstaFyleEnabled, isBulkFyleEnabled, isMileageEnabled and isPerDiemEnabled to false if orgSettings and orgUserSettings properties are not allowed', fakeAsync(() => { - const mockOrgUserSettingsData = cloneDeep(orgUserSettingsData); - mockOrgUserSettingsData.insta_fyle_settings.allowed = false; - mockOrgUserSettingsData.bulk_fyle_settings.allowed = false; - orgUserSettingsService.get.and.returnValue(of(mockOrgUserSettingsData)); - - component.ionViewWillEnter(); - tick(500); - expect(component.expensesTaskCount).toBe(10); - expect(component.isNewReportsFlowEnabled).toBeFalse(); - - expect(orgUserSettingsService.get).toHaveBeenCalledTimes(1); - expect(orgSettingsService.get).toHaveBeenCalledTimes(1); - component.isInstaFyleEnabled$.subscribe((isInstaFyleEnabled) => { - expect(isInstaFyleEnabled).toBeFalse(); - }); - component.isBulkFyleEnabled$.subscribe((isBulkFyleEnabled) => { - expect(isBulkFyleEnabled).toBeTrue(); - }); - component.isMileageEnabled$.subscribe((isMileageEnabled) => { - expect(isMileageEnabled).toBeTrue(); - }); - component.isPerDiemEnabled$.subscribe((isPerDiemEnabled) => { - expect(isPerDiemEnabled).toBeTrue(); - }); - })); - - it('should set isInstaFyleEnabled, isBulkFyleEnabled, isMileageEnabled and isPerDiemEnabled to undefined if orgUserSettings and orgSettings are undefined', fakeAsync(() => { - orgUserSettingsService.get.and.returnValue(of(undefined)); - orgSettingsService.get.and.returnValue(of(undefined)); - - component.ionViewWillEnter(); - tick(500); - expect(component.expensesTaskCount).toBe(10); - expect(component.isNewReportsFlowEnabled).toBeFalse(); - - expect(orgUserSettingsService.get).toHaveBeenCalledTimes(1); - expect(orgSettingsService.get).toHaveBeenCalledTimes(1); - component.isInstaFyleEnabled$.subscribe((isInstaFyleEnabled) => { - expect(isInstaFyleEnabled).toBeUndefined(); - }); - component.isBulkFyleEnabled$.subscribe((isBulkFyleEnabled) => { - expect(isBulkFyleEnabled).toBeUndefined(); - }); - component.isMileageEnabled$.subscribe((isMileageEnabled) => { - expect(isMileageEnabled).toBeUndefined(); - }); - component.isPerDiemEnabled$.subscribe((isPerDiemEnabled) => { - expect(isPerDiemEnabled).toBeUndefined(); - }); - })); - - it('should set hardwareBackButton and expenseTaskCount successfully', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - expect(component.hardwareBackButton).toEqual(backButtonSubscription); - expect(platformHandlerService.registerBackButtonAction).toHaveBeenCalledOnceWith( - BackButtonActionPriority.MEDIUM, - component.backButtonAction - ); - expect(tasksService.getExpensesTaskCount).toHaveBeenCalledTimes(1); - expect(component.expensesTaskCount).toBe(10); - })); - - it('should set isNewReportFlowEnabled to true if simplified_report_closure_settings is defined ', fakeAsync(() => { - orgSettingsService.get.and.returnValue(of(orgSettingsParamsWithSimplifiedReport)); - - component.ionViewWillEnter(); - tick(500); - - expect(component.isNewReportsFlowEnabled).toBeTrue(); - })); - - it('should call setupActionSheet once', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(component.setupActionSheet).toHaveBeenCalledOnceWith(orgSettingsRes, allowedExpenseTypes); - })); - - it('should update cardNumbers by calling getCardDetail', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(corporateCreditCardService.getAssignedCards).toHaveBeenCalledTimes(1); - expect(component.getCardDetail).toHaveBeenCalledOnceWith(expectedAssignedCCCStats.cardDetails); - expect(component.cardNumbers).toEqual([ - { label: '****8698', value: '8698' }, - { label: '****869', value: '869' }, - ]); - })); - - it('should update headerState, reviewMode and set isLoading to false after 500ms', fakeAsync(() => { - component.ionViewWillEnter(); - - expect(component.headerState).toEqual(HeaderState.base); - expect(component.reviewMode).toBeFalse(); - expect(component.isLoading).toBeTrue(); - - tick(500); - - expect(component.isLoading).toBeFalse(); - })); - - it('should set clusterDomain, ROUTER_API_ENDPOINT, navigateBack, simpleSearchText, currentPageNumber, selectionMode and selectedElements', fakeAsync(() => { - component.simpleSearchText = 'example'; - component.currentPageNumber = 2; - component.selectionMode = true; - component.ionViewWillEnter(); - tick(500); - - expect(component.ROUTER_API_ENDPOINT).toEqual(environment.ROUTER_API_ENDPOINT); - expect(tokenService.getClusterDomain).toHaveBeenCalledTimes(1); - expect(component.clusterDomain).toEqual(apiAuthRes.cluster_domain); - expect(component.navigateBack).toBeFalse(); - expect(component.simpleSearchText).toEqual(''); - expect(component.currentPageNumber).toBe(1); - expect(component.selectionMode).toBeFalse(); - expect(component.selectedElements).toEqual([]); - })); - - it('should call syncOutboxExpenses twice if isConnected is true', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(component.syncOutboxExpenses).toHaveBeenCalledTimes(2); - })); - - it('should call syncOutboxExpenses once if isConnected is false', fakeAsync(() => { - component.isConnected$ = of(false); - component.ionViewWillEnter(); - tick(500); - - expect(component.syncOutboxExpenses).toHaveBeenCalledTimes(1); - })); - - it('should set homeCurrency and homeCurrencySymbol correctly', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(currencyService.getHomeCurrency).toHaveBeenCalledTimes(1); - component.homeCurrency$.subscribe((currency) => { - expect(currency).toEqual('USD'); - }); - expect(component.homeCurrencySymbol).toEqual('$'); - })); - - it('should update loadData$ if user types something in input', fakeAsync(() => { - component.ionViewWillEnter(); - expect(inputElement.value).toEqual(''); - inputElement.value = 'example'; - inputElement.dispatchEvent(new Event('keyup')); - tick(500); - - component.loadExpenses$.subscribe((loadData) => { - expect(loadData).toEqual({ pageNumber: 1, searchString: 'example' }); - }); - })); - - it('should call extendQueryParamsForTextSearch and getMyExpensesCount whenever loadData$ value changes', fakeAsync(() => { - component.ionViewWillEnter(); - expect(inputElement.value).toEqual(''); - inputElement.value = 'example'; - inputElement.dispatchEvent(new Event('keyup')); - tick(500); - - expect(expensesService.getExpensesCount).toHaveBeenCalledTimes(5); - expect(expensesService.getExpensesCount).toHaveBeenCalledWith({ - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - }); - expect(expensesService.getExpenses).toHaveBeenCalledTimes(2); - expect(expensesService.getExpenses).toHaveBeenCalledWith({ - offset: 0, - limit: 10, - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - order: 'spent_at.desc,created_at.desc,id.desc', - }); - - expect(component.acc).toEqual(apiExpenses1); - })); - - it('should not call getMyExpenses if count is less than (params.pageNumber - 1) * 10', fakeAsync(() => { - transactionService.getMyExpensesCount.and.returnValue(of(0)); - component.ionViewWillEnter(); - expect(inputElement.value).toEqual(''); - inputElement.value = 'example'; - inputElement.dispatchEvent(new Event('keyup')); - tick(500); - - expect(expensesService.getExpensesCount).toHaveBeenCalledTimes(5); - expect(expensesService.getExpensesCount).toHaveBeenCalledWith({ - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - }); - expect(component.clusterDomain).toEqual(apiAuthRes.cluster_domain); - expect(transactionService.getMyExpenses).not.toHaveBeenCalled(); - expect(component.acc).toEqual(apiExpenses1); - })); - - it('should call getMyExpenseCount with order if sortDir and sortParam are defined', fakeAsync(() => { - component.ionViewWillEnter(); - component.loadExpenses$.next({ - pageNumber: 1, - sortDir: 'asc', - sortParam: 'approvalDate', - }); - tick(500); - - expect(expensesService.getExpenses).toHaveBeenCalledTimes(2); - expect(expensesService.getExpenses).toHaveBeenCalledWith({ - offset: 0, - limit: 10, - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - order: 'spent_at.desc,created_at.desc,id.desc', - }); - })); - - it('should set myExpenses$, count$, isNewUser$ and isInfiniteScrollRequired', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - component.myExpenses$.subscribe((myExpenses) => { - expect(myExpenses).toEqual(apiExpenses1); - }); - component.count$.subscribe((count) => { - expect(count).toBe(10); - }); - component.isNewUser$.subscribe((isNewUser) => { - expect(isNewUser).toBeFalse(); - }); - component.isInfiniteScrollRequired$.subscribe((isInfiniteScrollReq) => { - expect(isInfiniteScrollReq).toBeTrue(); - }); - })); - - it('should call setAllExpensesCountAndAmount once', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(component.setAllExpensesCountAndAmount).toHaveBeenCalledTimes(1); - })); - - it('should update allExpenseCountHeader$ and draftExpensesCount$ based on loadData', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - component.allExpenseCountHeader$.subscribe((allExpenseCountHeader) => { - expect(expensesService.getExpenseStats).toHaveBeenCalledWith({ - state: 'in.(COMPLETE,DRAFT)', - report_id: 'is.null', - }); - expect(allExpenseCountHeader).toBe(3); - }); - component.draftExpensesCount$.subscribe((draftExpensesCount) => { - expect(expensesService.getExpenseStats).toHaveBeenCalledWith({ - report_id: 'is.null', - state: 'in.(DRAFT)', - }); - expect(draftExpensesCount).toBe(3); - }); - expect(expensesService.getExpenseStats).toHaveBeenCalledTimes(2); - })); - - it('should navigate relative to activatedRoute and call clearFilters if snapshot.params and queryParams are undefined', fakeAsync(() => { - component.filters = { - state: [AdvancesStates.paid, AdvancesStates.cancelled], - }; - const stringifiedFilters = JSON.stringify(component.filters); - component.ionViewWillEnter(); - tick(500); - - expect(router.navigate).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledWith([], { - relativeTo: activatedRoute, - queryParams: { - filters: stringifiedFilters, - }, - replaceUrl: true, - }); - expect(component.clearFilters).toHaveBeenCalledTimes(1); - })); - - it('should update filters and filterPills if activatedRoute contains filters', fakeAsync(() => { - activatedRoute.snapshot.queryParams.filters = '{"sortDir": "desc"}'; - component.ionViewWillEnter(); - tick(500); - expect(component.clearFilters).not.toHaveBeenCalled(); - component.filters = { - state: [AdvancesStates.paid, AdvancesStates.cancelled], - }; - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((loadData) => { - expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); - }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - })); - - it('should update filters and filterPills if activatedRoute state is equal to needsreceipt', fakeAsync(() => { - activatedRoute.snapshot.params.state = 'needsreceipt'; - component.filters = { - state: [AdvancesStates.paid, AdvancesStates.cancelled], - }; - component.ionViewWillEnter(); - tick(500); - expect(component.clearFilters).not.toHaveBeenCalled(); - - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((loadData) => { - expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); - }); - expect(component.generateFilterPills).toHaveBeenCalledTimes(1); - expect(component.filterPills).toEqual(creditTxnFilterPill); - })); - - it('should update filters and filterPills if activatedRoute state is equal to policyviolated', fakeAsync(() => { - activatedRoute.snapshot.params.state = 'policyviolated'; - component.filters = { - state: [AdvancesStates.paid, AdvancesStates.cancelled], - }; - component.ionViewWillEnter(); - tick(500); - - expect(component.clearFilters).not.toHaveBeenCalled(); - expect(component.filters).toEqual({ - is_policy_flagged: 'eq.true', - or: '(policy_amount.is.null,policy_amount.gt.0.0001)', - state: 'POLICY_VIOLATED', - }); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((loadData) => { - expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); - }); - expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ - state: 'POLICY_VIOLATED', - is_policy_flagged: 'eq.true', - or: '(policy_amount.is.null,policy_amount.gt.0.0001)', - }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - })); - - it('should update filters and filterPills if activatedRoute state is equal to cannotreport', fakeAsync(() => { - activatedRoute.snapshot.params.state = 'cannotreport'; - component.filters = { - state: [AdvancesStates.paid, AdvancesStates.cancelled], - }; - component.ionViewWillEnter(); - tick(500); - - expect(component.clearFilters).not.toHaveBeenCalled(); - expect(component.filters).toEqual({ - policy_amount: 'lt.0.0001', - state: 'CANNOT_REPORT', - }); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((loadData) => { - expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); - }); - expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ - policy_amount: 'lt.0.0001', - state: 'CANNOT_REPORT', - }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - })); - - it('should set openReports$ and call doRefresh', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(reportService.getAllExtendedReports).toHaveBeenCalledOnceWith({ - queryParams: { - rp_state: 'in.(DRAFT,APPROVER_PENDING,APPROVER_INQUIRY)', - }, - }); - component.openReports$.subscribe((openReports) => { - expect(openReports).toEqual(apiExtendedReportRes); - }); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - - it('should set openReports$ and call doRefresh if report_approvals is defined', fakeAsync(() => { - const extendedReportResWithReportApproval = [expectedReportSingleResponse]; - reportService.getAllExtendedReports.and.returnValue(of(extendedReportResWithReportApproval)); - component.ionViewWillEnter(); - tick(500); - - expect(reportService.getAllExtendedReports).toHaveBeenCalledOnceWith({ - queryParams: { - rp_state: 'in.(DRAFT,APPROVER_PENDING,APPROVER_INQUIRY)', - }, - }); - component.openReports$.subscribe((openReports) => { - expect(openReports).toEqual(extendedReportResWithReportApproval); - }); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - - it('should return an empty array if no expenses are found and search is empty', fakeAsync(() => { - expensesService.getExpensesCount.and.returnValue(of(0)); - component.ionViewWillEnter(); - component.loadExpenses$.next({ - pageNumber: 1, - searchString: '', - sortParam: 'category->name', - sortDir: 'asc', - }); - tick(500); - - component.myExpenses$.subscribe((res) => { - expect(res).toEqual([]); - }); - - expect(expensesService.getExpenses).not.toHaveBeenCalled(); - })); - }); - - it('HeaderState(): should return the headerState', () => { - expect(component.HeaderState).toEqual(HeaderState); - }); - - describe('clearText', () => { - let dispatchEventSpy: jasmine.Spy; - beforeEach(() => { - component.isSearchBarFocused = false; - component.simpleSearchInput = getElementRef(fixture, '.my-expenses--simple-search-input'); - inputElement = component.simpleSearchInput.nativeElement; - dispatchEventSpy = spyOn(inputElement, 'dispatchEvent'); - }); - it('should clear the search text and dispatch keyup event', () => { - component.clearText('onSimpleSearchCancel'); - - expect(component.simpleSearchText).toBe(''); - expect(inputElement.value).toBe(''); - expect(dispatchEventSpy).toHaveBeenCalledWith(new Event('keyup')); - expect(component.isSearchBarFocused).toBe(true); - }); - - it('should clear the search text and not toggle isSearchBarFocused when isFromCancel is not specified', () => { - component.clearText(''); - - expect(component.simpleSearchText).toBe(''); - expect(inputElement.value).toBe(''); - expect(dispatchEventSpy).toHaveBeenCalledWith(new Event('keyup')); - expect(component.isSearchBarFocused).toBe(false); - }); - }); - - it('onSearchBarFocus(): should set isSearchBarFocused to true', () => { - component.isSearchBarFocused = false; - component.simpleSearchInput = getElementRef(fixture, '.my-expenses--simple-search-input'); - inputElement = component.simpleSearchInput.nativeElement; - inputElement.dispatchEvent(new Event('focus')); - expect(component.isSearchBarFocused).toBeTrue(); - }); - - it('formatTransactions(): should format transactions correctly', () => { - const unformattedTransactions = unformattedTxnData; - const formattedTransactions = component.formatTransactions(unformattedTransactions); - - expect(formattedTransactions.length).toBe(unformattedTransactions.length); - expect(formattedTransactions).toEqual(expectedFormattedTransaction); - }); - - describe('switchSelectionMode(): ', () => { - beforeEach(() => { - component.selectionMode = true; - component.loadExpenses$ = new BehaviorSubject({ - searchString: 'example', - }); - component.headerState = HeaderState.simpleSearch; - component.allExpensesStats$ = of({ count: 10, amount: 1000 }); - spyOn(component, 'selectExpense'); - spyOn(component, 'setAllExpensesCountAndAmount'); - }); - it('should set headerState to simpleSearch if searchString is defined in loadData', () => { - component.switchSelectionMode(); - - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toBe(HeaderState.simpleSearch); - expect(component.selectedElements).toEqual([]); - expect(component.setAllExpensesCountAndAmount).toHaveBeenCalledTimes(1); - expect(component.selectExpense).not.toHaveBeenCalled(); - }); - - it('should set headerState to base if searchString is defined in loadData and if expense is selected', () => { - component.loadExpenses$ = new BehaviorSubject({}); - - component.switchSelectionMode(apiExpenses1[0]); - - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toBe(HeaderState.base); - expect(component.selectedElements).toEqual([]); - expect(component.setAllExpensesCountAndAmount).toHaveBeenCalledTimes(1); - expect(component.selectExpense).toHaveBeenCalledOnceWith(apiExpenses1[0]); - }); - - it('should update allExpensesStats$ and headerState if selectionMode is false', () => { - component.selectionMode = false; - - component.switchSelectionMode(); - - expect(component.selectionMode).toBeTrue(); - expect(component.headerState).toBe(HeaderState.multiselect); - expect(component.setAllExpensesCountAndAmount).not.toHaveBeenCalled(); - expect(component.selectExpense).not.toHaveBeenCalled(); - component.allExpensesStats$.subscribe((stats) => { - expect(stats.count).toBe(0); - expect(stats.amount).toBe(0); - }); - }); - }); - - describe('switchOutboxSelectionMode(): ', () => { - beforeEach(() => { - component.selectionMode = true; - component.loadExpenses$ = new BehaviorSubject({ - searchString: 'example', - }); - component.headerState = HeaderState.simpleSearch; - component.allExpensesStats$ = of({ count: 10, amount: 1000 }); - spyOn(component, 'selectExpense'); - spyOn(component, 'setAllExpensesCountAndAmount'); - spyOn(component, 'setOutboxExpenseStatsOnSelect'); - }); - it('should set headerState to simpleSearch if searchString is defined in loadData', () => { - component.switchOutboxSelectionMode(); - - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toBe(HeaderState.simpleSearch); - expect(component.selectedOutboxExpenses).toEqual([]); - expect(component.selectExpense).not.toHaveBeenCalled(); - }); - - it('should set headerState to base if searchString is defined in loadData and if expense is selected', () => { - component.loadExpenses$ = new BehaviorSubject({}); - transactionService.getReportableExpenses.and.returnValue([]); - transactionService.excludeCCCExpenses.and.returnValue([]); - - component.switchOutboxSelectionMode(apiExpenseRes[0]); - - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toBe(HeaderState.base); - expect(component.selectedOutboxExpenses.length).toEqual(1); - expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(2); - }); - - it('should update allExpensesStats$ and headerState if selectionMode is false', () => { - component.selectionMode = false; - - component.switchOutboxSelectionMode(); - - expect(component.selectionMode).toBeTrue(); - expect(component.headerState).toBe(HeaderState.multiselect); - expect(component.setOutboxExpenseStatsOnSelect).not.toHaveBeenCalled(); - expect(component.selectExpense).not.toHaveBeenCalled(); - component.allExpensesStats$.subscribe((stats) => { - expect(stats.count).toBe(0); - expect(stats.amount).toBe(0); - }); - }); - }); - - it('sendFirstExpenseCreatedEvent(): should store the first expense created event', fakeAsync(() => { - component.allExpensesStats$ = of({ - count: 0, - amount: 0, - }); - storageService.get.and.resolveTo(false); - component.sendFirstExpenseCreatedEvent(); - tick(100); - expect(storageService.get).toHaveBeenCalledOnceWith('isFirstExpenseCreated'); - expect(trackingService.createFirstExpense).toHaveBeenCalledTimes(1); - expect(storageService.set).toHaveBeenCalledOnceWith('isFirstExpenseCreated', true); - })); - - describe('setAllExpensesCountAndAmount(): ', () => { - it('should call expensesService.getExpenseStats if loadExpenses contains queryParams', () => { - component.loadExpenses$ = new BehaviorSubject({ - queryParams: { - 'matched_corporate_card_transactions->0->corporate_card_number': '8698', - }, - }); - - expensesService.getExpenseStats.and.returnValue(of(completeStats)); - component.setAllExpensesCountAndAmount(); - component.allExpensesStats$.subscribe((allExpenseStats) => { - expect(expensesService.getExpenseStats).toHaveBeenCalledOnceWith({ - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - or: ['(matched_corporate_card_transactions->0->corporate_card_number.8698)'], - }); - expect(allExpenseStats).toEqual({ - count: 3, - amount: 30, - }); - }); - }); - - it('should call expensesService.getExpenseStats and initialize queryParams to empty object if loadData.queryParams is falsy', () => { - component.loadExpenses$ = new BehaviorSubject({ - queryParams: null, - }); - expensesService.getExpenseStats.and.returnValue(of(incompleteStats)); - component.setAllExpensesCountAndAmount(); - component.allExpensesStats$.subscribe((allExpenseStats) => { - expect(expensesService.getExpenseStats).toHaveBeenCalledOnceWith({ - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - }); - expect(allExpenseStats).toEqual({ - count: incompleteStats.data.count, - amount: incompleteStats.data.total_amount, - }); - }); - }); - - it('should handle error in getExpenseStats and complete the observable', () => { - component.loadExpenses$ = new BehaviorSubject({ - queryParams: { - 'matched_corporate_card_transactions->0->corporate_card_number': '8698', - }, - }); - expensesService.getExpenseStats.and.returnValue(throwError(() => new Error('error message'))); - component.setAllExpensesCountAndAmount(); - component.allExpensesStats$.subscribe({ - error: (err) => { - expect(err.message).toEqual('error message'); - }, - }); - expect(expensesService.getExpenseStats).toHaveBeenCalledOnceWith({ - report_id: 'is.null', - state: 'in.(COMPLETE,DRAFT)', - or: ['(matched_corporate_card_transactions->0->corporate_card_number.8698)'], - }); - }); - }); - describe('setupActionSheet()', () => { - it('should update actionSheetButtons', () => { - spyOn(component, 'actionSheetButtonsHandler'); - component.setupActionSheet(orgSettingsRes, allowedExpenseTypes); - expect(component.actionSheetButtons).toEqual(expectedActionSheetButtonRes); - }); - - it('should update actionSheetButtons without mileage', () => { - spyOn(component, 'actionSheetButtonsHandler'); - const mockAllowedExpenseTypes = clone(allowedExpenseTypes); - mockAllowedExpenseTypes.mileage = false; - component.setupActionSheet(orgSettingsRes, mockAllowedExpenseTypes); - expect(component.actionSheetButtons).toEqual(expectedActionSheetButtonsWithPerDiem); - }); - - it('should update actionSheetButtons without Per Diem', () => { - spyOn(component, 'actionSheetButtonsHandler'); - const mockAllowedExpenseTypes = clone(allowedExpenseTypes); - mockAllowedExpenseTypes.perDiem = false; - component.setupActionSheet(orgSettingsRes, mockAllowedExpenseTypes); - expect(component.actionSheetButtons).toEqual(expectedActionSheetButtonsWithMileage); - }); - }); - - describe('actionSheetButtonsHandler():', () => { - it('should call trackingService and navigate to add_edit_per_diem if action is add per diem', () => { - const handler = component.actionSheetButtonsHandler('Add Per Diem', 'add_edit_per_diem'); - handler(); - expect(trackingService.myExpensesActionSheetAction).toHaveBeenCalledOnceWith({ - Action: 'Add Per Diem', - }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_per_diem', - { - navigate_back: true, - }, - ]); - }); - it('should call trackingService and navigate to add_edit_mileage if action is add mileage', () => { - const handler = component.actionSheetButtonsHandler('Add Mileage', 'add_edit_mileage'); - handler(); - expect(trackingService.myExpensesActionSheetAction).toHaveBeenCalledOnceWith({ - Action: 'Add Mileage', - }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_mileage', - { - navigate_back: true, - }, - ]); - }); - it('should call trackingService and navigate to add_edit_expense if action is add expense', () => { - const handler = component.actionSheetButtonsHandler('Add Expense', 'add_edit_expense'); - handler(); - expect(trackingService.myExpensesActionSheetAction).toHaveBeenCalledOnceWith({ - Action: 'Add Expense', - }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_expense', - { - navigate_back: true, - }, - ]); - }); - it('should call trackingService and navigate to camera_overlay if action is capture receipts', () => { - const handler = component.actionSheetButtonsHandler('capture receipts', 'camera_overlay'); - handler(); - expect(trackingService.myExpensesActionSheetAction).toHaveBeenCalledOnceWith({ - Action: 'capture receipts', - }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'camera_overlay', - { - navigate_back: true, - }, - ]); - }); - }); - - it('getCardDetail(): should call corporateCreditCardService.getExpenseDetailsInCards method', () => { - corporateCreditCardService.getExpenseDetailsInCards.and.returnValue(expectedUniqueCardStats); - const getCardDetailRes = component.getCardDetail(cardAggregateStatParam); - - expect(getCardDetailRes).toEqual(expectedUniqueCardStats); - expect(corporateCreditCardService.getExpenseDetailsInCards).toHaveBeenCalledOnceWith( - uniqueCardsParam, - cardAggregateStatParam - ); - }); - - it('ionViewWillLeave(): should unsubscribe hardwareBackButton and update onPageExit', () => { - component.hardwareBackButton = new Subscription(); - const unsubscribeSpy = spyOn(component.hardwareBackButton, 'unsubscribe'); - const onPageNextSpy = spyOn(component.onPageExit$, 'next'); - component.ionViewWillLeave(); - expect(unsubscribeSpy).toHaveBeenCalledTimes(1); - expect(onPageNextSpy).toHaveBeenCalledOnceWith(null); - }); - - describe('backButtonAction(): ', () => { - beforeEach(() => { - spyOn(component, 'switchSelectionMode'); - spyOn(component, 'onSimpleSearchCancel'); - }); - it('should call switchSelectionMode when headerState is HeaderState.multiselect', () => { - component.headerState = HeaderState.multiselect; - - component.backButtonAction(); - - expect(component.switchSelectionMode).toHaveBeenCalledTimes(1); - expect(component.onSimpleSearchCancel).not.toHaveBeenCalled(); - expect(navController.back).not.toHaveBeenCalled(); - }); - - it('should call onSimpleSearchCancel when headerState is HeaderState.simpleSearch', () => { - component.headerState = HeaderState.simpleSearch; - - component.backButtonAction(); - - expect(component.onSimpleSearchCancel).toHaveBeenCalledTimes(1); - expect(component.switchSelectionMode).not.toHaveBeenCalled(); - expect(navController.back).not.toHaveBeenCalled(); - }); - - it('should call navController.back when headerState is neither HeaderState.multiselect nor HeaderState.simpleSearch', () => { - component.headerState = HeaderState.base; - - component.backButtonAction(); - - expect(navController.back).toHaveBeenCalled(); - expect(component.switchSelectionMode).not.toHaveBeenCalled(); - expect(component.onSimpleSearchCancel).not.toHaveBeenCalled(); - }); - }); - - it('setupNetworkWatcher(): should update isConnected$ and call connectivityWatcher', () => { - networkService.isOnline.and.returnValue(of(true)); - component.setupNetworkWatcher(); - expect(networkService.connectivityWatcher).toHaveBeenCalledTimes(1); - expect(networkService.isOnline).toHaveBeenCalledTimes(1); - component.isConnected$.subscribe((isConnected) => { - expect(isConnected).toBeTrue(); - }); - }); - - describe('loadData(): ', () => { - beforeEach(() => { - component.currentPageNumber = 2; - component.loadExpenses$ = new BehaviorSubject({ - pageNumber: 2, - }); - }); - it('should increment currentPageNumber and emit updated params and call complete() after 1s', fakeAsync(() => { - const mockEvent = { target: { complete: jasmine.createSpy('complete') } }; - - component.loadData(mockEvent); - - expect(component.currentPageNumber).toBe(3); - expect(component.loadExpenses$.getValue().pageNumber).toBe(3); - tick(1000); - expect(mockEvent.target.complete).toHaveBeenCalledTimes(1); - })); - - it('should increment currentPageNumber and emit updated params if target is not defined', () => { - const mockEvent = {}; - - component.loadData(mockEvent); - - expect(component.currentPageNumber).toBe(3); - expect(component.loadExpenses$.getValue().pageNumber).toBe(3); - }); - - it('should increment currentPageNumber and emit updated params if event if undefined', () => { - const mockEvent = undefined; - - component.loadData(mockEvent); - - expect(component.currentPageNumber).toBe(3); - expect(component.loadExpenses$.getValue().pageNumber).toBe(3); - }); - }); - - describe('doRefresh():', () => { - beforeEach(() => { - transactionService.clearCache.and.returnValue(of(null)); - component.currentPageNumber = 2; - component.loadExpenses$ = new BehaviorSubject({ - pageNumber: 2, - }); - spyOn(component, 'setExpenseStatsOnSelect'); - }); - it('should refresh data if ionRefresher event is not passed as an argument', fakeAsync(() => { - component.doRefresh(); - tick(1000); - - expect(component.selectedElements).toEqual([]); - expect(transactionService.clearCache).toHaveBeenCalledTimes(1); - expect(component.currentPageNumber).toBe(1); - expect(component.loadExpenses$.getValue().pageNumber).toBe(1); - })); - - it('should refresh data and call complete if ionRefresher event if present and selectionMode is true', fakeAsync(() => { - component.selectionMode = true; - const mockEvent = { target: { complete: jasmine.createSpy('complete') } }; - - component.doRefresh(mockEvent); - tick(1000); - - expect(component.selectedElements).toEqual([]); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.clearCache).toHaveBeenCalledTimes(1); - expect(component.currentPageNumber).toBe(1); - expect(component.loadExpenses$.getValue().pageNumber).toBe(1); - expect(mockEvent.target.complete).toHaveBeenCalledTimes(1); - })); - - it('should refresh data if target is not defined', fakeAsync(() => { - const mockEvent = {}; - - component.doRefresh(mockEvent); - tick(1000); - - expect(component.selectedElements).toEqual([]); - expect(transactionService.clearCache).toHaveBeenCalledTimes(1); - expect(component.currentPageNumber).toBe(1); - expect(component.loadExpenses$.getValue().pageNumber).toBe(1); - })); - }); - - it('syncOutboxExpenses(): should call transactionoutboxService and do a refresh', fakeAsync(() => { - spyOn(component, 'formatTransactions').and.returnValues(apiExpenseRes, []); - transactionOutboxService.getPendingTransactions.and.returnValues(txnList, []); - transactionOutboxService.sync.and.resolveTo(undefined); - spyOn(component, 'doRefresh'); - - component.syncOutboxExpenses(); - tick(100); - - expect(component.pendingTransactions).toEqual(apiExpenseRes); - expect(component.formatTransactions).toHaveBeenCalledTimes(2); - expect(transactionOutboxService.getPendingTransactions).toHaveBeenCalledTimes(2); - expect(transactionOutboxService.sync).toHaveBeenCalledTimes(1); - expect(component.syncing).toBeFalse(); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - - describe('generateFilterPills(): ', () => { - beforeEach(() => { - myExpenseService.generateStateFilterPills.and.callFake((filterPill, filters) => { - filterPill.push(stateFilterPill); - }); - myExpenseService.generateReceiptsAttachedFilterPills.and.callFake((filterPill, filters) => { - filterPill.push(receiptsAttachedFilterPill); - }); - myExpenseService.generateDateFilterPills.and.returnValue(dateFilterPill); - myExpenseService.generateTypeFilterPills.and.callFake((filters, filterPill) => { - filterPill.push(typeFilterPill); - }); - myExpenseService.generateSortFilterPills.and.callFake((filters, filterPill) => { - filterPill.push(sortFilterPill); - }); - myExpenseService.generateCardFilterPills.and.callFake((filterPill, filters) => { - filterPill.push(cardFilterPill); - }); - myExpenseService.generateSplitExpenseFilterPills.and.callFake((filterPill, filters) => { - filterPill.push(splitExpenseFilterPill); - }); - }); - it('should return filterPills based on the properties present in filters', () => { - const filterPillRes = component.generateFilterPills(expenseFiltersData1); - expect(filterPillRes).toEqual(expectedFilterPill1); - }); - - it('should return filterPills if state, type and cardNumbers properties are not present in filters passed as argument', () => { - const filterPillRes = component.generateFilterPills(expenseFiltersData2); - expect(filterPillRes).toEqual(expectedFilterPill2); - }); - }); - - describe('addNewFiltersToParams(): ', () => { - beforeEach(() => { - sharedExpenseService.generateCardNumberParams.and.returnValue({ - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - or: [], - }); - sharedExpenseService.generateDateParams.and.returnValue({ - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }); - sharedExpenseService.generateReceiptAttachedParams.and.returnValue({ - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }); - sharedExpenseService.generateStateFilters.and.returnValue({ - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }); - sharedExpenseService.generateTypeFilters.and.returnValue({ - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }); - sharedExpenseService.setSortParams.and.returnValue({ sortDir: 'asc' }); - sharedExpenseService.generateSplitExpenseParams.and.returnValue({ - or: ['(is_split.eq.true)'], - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - }); - }); - - it('should update queryParams if filter state is not defined', () => { - component.filters = {}; - - const currentParams = component.addNewFiltersToParams(); - - expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( - { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, - component.filters - ); - expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - - expect(currentParams).toEqual(expectedCurrentParamsWoFilterState); - expect(component.reviewMode).toBeFalse(); - }); - - it('should update queryParams if filter state includes only DRAFT', () => { - component.filters = { - state: ['DRAFT'], - }; - - const currentParams = component.addNewFiltersToParams(); - - expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( - { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, - component.filters - ); - expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - - expect(currentParams).toEqual(expectedCurrentParamsDraftState); - expect(component.reviewMode).toBeTrue(); - }); - - it('should update queryParams if filter state includes only CANNOT_REPORT', () => { - component.filters = { - state: ['CANNOT_REPORT'], - }; - - const currentParams = component.addNewFiltersToParams(); - - expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( - { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, - component.filters - ); - expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - - expect(currentParams).toEqual(expectedCurrentParamsCannotReportState); - expect(component.reviewMode).toBeTrue(); - }); - - it('should update queryParams if filter state includes both DRAFT and CANNOT_REPORT', () => { - component.filters = { - state: ['DRAFT', 'CANNOT_REPORT'], - }; - - const currentParams = component.addNewFiltersToParams(); - - expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( - { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, - component.filters - ); - expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { - 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', - and: '(spent_at.gte.March,spent_at.lt.April)', - or: [], - }, - component.filters - ); - - expect(currentParams).toEqual(expectedCurrentParamsWithDraftCannotReportState); - expect(component.reviewMode).toBeTrue(); - }); - - it('should set reviewMode to false if filter state is APPROVED', () => { - component.filters = { - state: ['APPROVED'], - }; - - const currentParams = component.addNewFiltersToParams(); - expect(component.reviewMode).toBeFalse(); - }); - }); - - describe('openFilters(): ', () => { - beforeEach(() => { - myExpenseService.getFilters.and.returnValue(cloneDeep(filterOptions1)); - const filterPopoverSpy = jasmine.createSpyObj('filterPopover', ['present', 'onWillDismiss']); - filterPopoverSpy.onWillDismiss.and.resolveTo({ data: selectedFilters2 }); - modalController.create.and.resolveTo(filterPopoverSpy); - component.filters = { - state: [], - }; - myExpenseService.generateSelectedFilters.and.returnValue(selectedFilters1); - component.loadExpenses$ = new BehaviorSubject({ - pageNumber: 1, - }); - component.currentPageNumber = 2; - myExpenseService.convertSelectedOptionsToExpenseFilters.and.returnValue({ sortDir: 'asc', splitExpense: 'YES' }); - spyOn(component, 'addNewFiltersToParams').and.returnValue({ searchString: 'example' }); - spyOn(component, 'generateFilterPills').and.returnValue([ - { - label: 'Transactions Type', - type: 'string', - value: 'Credit', - }, - ]); - }); - - it('should call modalController and myExpensesService', fakeAsync(() => { - component.cardNumbers = [ - { - label: 'ABC', - value: '1234', - }, - ]; - - component.openFilters('approvalDate'); - tick(200); - - expect(modalController.create).toHaveBeenCalledOnceWith(modalControllerParams); - expect(myExpenseService.convertSelectedOptionsToExpenseFilters).toHaveBeenCalledOnceWith(selectedFilters2); - expect(component.filters).toEqual({ sortDir: 'asc', splitExpense: 'YES' }); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((loadData) => { - expect(loadData).toEqual({ searchString: 'example' }); - }); - - expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ sortDir: 'asc', splitExpense: 'YES' }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - expect(trackingService.myExpensesFilterApplied).toHaveBeenCalledOnceWith({ sortDir: 'asc', splitExpense: 'YES' }); - })); - - it('should call modalController and myExpensesService if cardNumbers is undefined', fakeAsync(() => { - component.cardNumbers = undefined; - - component.openFilters('approvalDate'); - tick(200); - - expect(modalController.create).toHaveBeenCalledOnceWith(modalControllerParams2); - - expect(myExpenseService.convertSelectedOptionsToExpenseFilters).toHaveBeenCalledOnceWith(selectedFilters2); - expect(component.filters).toEqual({ sortDir: 'asc', splitExpense: 'YES' }); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((loadExpenses) => { - expect(loadExpenses).toEqual({ searchString: 'example' }); - }); - expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ sortDir: 'asc', splitExpense: 'YES' }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - expect(trackingService.myExpensesFilterApplied).toHaveBeenCalledOnceWith({ sortDir: 'asc', splitExpense: 'YES' }); - })); - }); - - it('clearFilters(): should clear the filters and call generateFilterPills', () => { - component.filters = { - sortDir: 'asc', - sortParam: 'category->name', - }; - component.currentPageNumber = 3; - spyOn(component, 'addNewFiltersToParams').and.returnValue({ - pageNumber: 1, - searchString: 'example', - }); - - spyOn(component, 'generateFilterPills').and.returnValue(creditTxnFilterPill); - - component.clearFilters(); - - expect(component.filters).toEqual({}); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((data) => { - expect(data).toEqual({ - pageNumber: 1, - searchString: 'example', - }); - }); - expect(component.generateFilterPills).toHaveBeenCalledOnceWith({}); - expect(component.filterPills).toEqual(creditTxnFilterPill); - }); - - it('setState(): should pageNumber to 1 and update isLoading correctly', fakeAsync(() => { - spyOn(component, 'addNewFiltersToParams').and.returnValue({ - pageNumber: 1, - searchString: 'example', - }); - component.loadExpenses$ = new BehaviorSubject({ - pageNumber: 1, - }); - - component.setState(); - - expect(component.isLoading).toBeTrue(); - expect(component.currentPageNumber).toBe(1); - component.loadExpenses$.subscribe((data) => { - expect(data).toEqual({ - pageNumber: 1, - searchString: 'example', - }); - }); - tick(500); - expect(component.isLoading).toBeFalse(); - })); - - describe('selectExpense(): ', () => { - beforeEach(() => { - sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); - component.allExpensesCount = 2; - spyOn(component, 'setExpenseStatsOnSelect'); - component.selectedElements = cloneDeep(apiExpenses1); - sharedExpenseService.isMergeAllowed.and.returnValue(true); - sharedExpenseService.excludeCCCExpenses.and.returnValue(apiExpenses1); - }); - - it('should remove an expense from selectedElements if it is present in selectedElements', () => { - sharedExpenseService.getReportableExpenses.and.returnValue([]); - const expense = apiExpenses1[0]; - component.selectedElements = cloneDeep(apiExpenses1); - - component.selectExpense(expense); - - expect(component.selectedElements).toEqual([apiExpenses1[1]]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeFalse(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([apiExpenses1[1]]); - expect(component.isMergeAllowed).toBeTrue(); - }); - - it('should remove an expense from selectedElements if it is present in selectedElements', () => { - sharedExpenseService.getReportableExpenses.and.returnValue([]); - component.allExpensesCount = 3; - - component.selectedElements = cloneDeep(cloneDeep(apiExpenses1)); - - component.selectExpense(expenseData); - - expect(component.selectedElements).toEqual([...apiExpenses1, expenseData]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeTrue(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([...apiExpenses1, expenseData]); - expect(component.isMergeAllowed).toBeTrue(); - }); - - it('should remove an expense from selectedElements if it is present in selectedElements and allExpenseCount is not equal to length of selectedElements', () => { - sharedExpenseService.getReportableExpenses.and.returnValue([]); - - component.selectedElements = cloneDeep(apiExpenses1); - - component.selectExpense(apiExpenses1[0]); - - expect(component.selectedElements).toEqual([apiExpenses1[1]]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeFalse(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([apiExpenses1[1]]); - expect(component.isMergeAllowed).toBeTrue(); - }); - - it('should update expenseToBeDeleted if selectedElements is an array of atleast 1', () => { - sharedExpenseService.excludeCCCExpenses.and.returnValue([apiExpenses1[1]]); - component.selectedElements = cloneDeep(apiExpenses1); - component.selectExpense(apiExpenses1[0]); - - expect(component.selectedElements).toEqual([apiExpenses1[1]]); - expect(component.expensesToBeDeleted).toEqual([apiExpenses1[1]]); - expect(component.cccExpenses).toBe(0); - expect(component.selectAll).toBeFalse(); - }); - - it('should remove an expense from selectedElements if it is present in selectedElements and tx_id is not present in expense', () => { - sharedExpenseService.getReportableExpenses.and.returnValue([]); - component.allExpensesCount = 0; - const expense = cloneDeep(apiExpenses1[0]); - expense.id = undefined; - component.selectedElements = cloneDeep(apiExpenses1); - component.selectedElements[0].id = undefined; - - component.selectExpense(expense); - - expect(component.selectedElements).toEqual([apiExpenses1[1]]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeFalse(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([apiExpenses1[1]]); - expect(component.isMergeAllowed).toBeTrue(); - }); - }); - - it('setExpenseStatsOnSelect(): should update allExpenseStats$', () => { - component.selectedElements = apiExpenses1; - component.setExpenseStatsOnSelect(); - component.allExpensesStats$.subscribe((expenseStats) => { - expect(expenseStats).toEqual({ - count: 2, - amount: 25, - }); - }); - }); - - describe('goToTransaction():', () => { - it('should navigate to add_edit_mileage page if category is mileage', () => { - component.goToTransaction({ - expense: { ...expenseData, category: { ...expenseData.category, name: 'mileage' } }, - expenseIndex: 1, - }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_mileage', - { id: expenseData.id, persist_filters: true }, - ]); - }); - - it('should navigate to add_edit_per_diem if category is per diem', () => { - component.goToTransaction({ - expense: { ...expenseData, category: { ...expenseData.category, name: 'per diem' } }, - expenseIndex: 1, - }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_per_diem', - { id: expenseData.id, persist_filters: true }, - ]); - }); - - it('should navigate to add_edit_expense if category is something else', () => { - component.goToTransaction({ expense: apiExpenses1[0], expenseIndex: 1 }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_expense', - { id: apiExpenses1[0].id, persist_filters: true }, - ]); - }); - - it('should not navigate to any other page, if category is not present', () => { - component.goToTransaction({ expense: { ...apiExpenses1[0], category: null }, expenseIndex: 1 }); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_expense', - { id: apiExpenses1[0].id, persist_filters: true }, - ]); - }); - }); - - describe('openCriticalPolicyViolationPopOver():', () => { - beforeEach(() => { - const criticalPolicyViolationPopOverSpy = jasmine.createSpyObj('criticalPolicyViolationPopOver', [ - 'present', - 'onWillDismiss', - ]); - criticalPolicyViolationPopOverSpy.onWillDismiss.and.resolveTo({ data: { action: 'continue' } }); - popoverController.create.and.resolveTo(criticalPolicyViolationPopOverSpy); - spyOn(component, 'showOldReportsMatBottomSheet'); - spyOn(component, 'showNewReportModal'); - }); - - it('should open popoverController and call showOldReportsMatBottomSheet', fakeAsync(() => { - component.openCriticalPolicyViolationPopOver({ - title: '2 Draft Expenses blocking the way', - message: '2 expenses are in draft state.', - reportType: 'oldReport', - }); - tick(100); - - expect(popoverController.create).toHaveBeenCalledOnceWith(popoverControllerParams); - - expect(component.showOldReportsMatBottomSheet).toHaveBeenCalledTimes(1); - expect(component.showNewReportModal).not.toHaveBeenCalled(); - })); - - it('should open popoverController and call showNewReportModal', fakeAsync(() => { - component.openCriticalPolicyViolationPopOver({ - title: '2 Draft Expenses blocking the way', - message: '2 expenses are in draft state.', - reportType: 'newReport', - }); - tick(100); - - expect(popoverController.create).toHaveBeenCalledOnceWith(popoverControllerParams); - - expect(component.showOldReportsMatBottomSheet).not.toHaveBeenCalled(); - expect(component.showNewReportModal).toHaveBeenCalledTimes(1); - })); - }); - - it('showNonReportableExpenseSelectedToast(): should call matSnackbar and call trackingService', () => { - snackbarProperties.setSnackbarProperties.and.returnValue(snackbarPropertiesRes); - component.showNonReportableExpenseSelectedToast('Please select one or more expenses to be reported'); - - expect(matSnackBar.openFromComponent).toHaveBeenCalledOnceWith(ToastMessageComponent, openFromComponentConfig); - expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('failure', { - message: 'Please select one or more expenses to be reported', - }); - expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ - ToastContent: 'Please select one or more expenses to be reported', - }); - }); - - describe('openCreateReportWithSelectedIds(): ', () => { - beforeEach(() => { - spyOn(component, 'showNonReportableExpenseSelectedToast'); - spyOn(component, 'openCriticalPolicyViolationPopOver'); - spyOn(component, 'showOldReportsMatBottomSheet'); - spyOn(component, 'showNewReportModal'); - spyOn(component, 'unreportableExpenseExceptionHandler'); - spyOn(component, 'reportableExpenseDialogHandler'); - }); - - describe('when restrictPendingTransactionsEnabled is false', () => { - beforeEach(() => { - // sharedExpenseService.restrictPendingTransactionsEnabled.and.returnValues(false); - }); - it('should call showNonReportableExpenseSelectedToast and return if selectedElement length is zero', fakeAsync(() => { - const expenses = cloneDeep(apiExpenses1); - component.selectedElements = expenses.map((expense) => { - return { ...expense, id: null }; - }); - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - 'Please select one or more expenses to be reported' - ); - expect(component.openCriticalPolicyViolationPopOver).not.toHaveBeenCalled(); - expect(component.showOldReportsMatBottomSheet).not.toHaveBeenCalled(); - expect(component.showNewReportModal).not.toHaveBeenCalled(); - })); - it('should call unreportableExpenseExceptionHandler if none of the reportable expenses are selected', fakeAsync(() => { - component.selectedElements = cloneDeep(apiExpenses1); - sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(true, true); - sharedExpenseService.isExpenseInDraft.and.returnValues(false, true); - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - expect(sharedExpenseService.isCriticalPolicyViolatedExpense).toHaveBeenCalledTimes(2); - expect(sharedExpenseService.isCriticalPolicyViolatedExpense).toHaveBeenCalledWith(apiExpenses1[0]); - expect(sharedExpenseService.isCriticalPolicyViolatedExpense).toHaveBeenCalledWith(apiExpenses1[1]); - - expect(sharedExpenseService.isExpenseInDraft).toHaveBeenCalledTimes(2); - expect(sharedExpenseService.isExpenseInDraft).toHaveBeenCalledWith(apiExpenses1[0]); - expect(sharedExpenseService.isExpenseInDraft).toHaveBeenCalledWith(apiExpenses1[1]); - - component.isReportableExpensesSelected = false; - - expect(component.unreportableExpenseExceptionHandler).toHaveBeenCalledOnceWith(1, 2, 0); - })); - - it('should call showOldReportsMatBottomSheet if reportType is oldReport', fakeAsync(() => { - component.selectedElements = cloneDeep(apiExpenses1); - component.isReportableExpensesSelected = true; - sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(false, false); - sharedExpenseService.isExpenseInDraft.and.returnValues(false, false); - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - expect(trackingService.addToReport).toHaveBeenCalled(); - expect(component.showOldReportsMatBottomSheet).toHaveBeenCalledOnceWith(); - })); - - it('should call showOldReportsMatBottomSheet if reportType is newReport', fakeAsync(() => { - component.selectedElements = cloneDeep(apiExpenses1); - component.isReportableExpensesSelected = true; - sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(false, false); - sharedExpenseService.isExpenseInDraft.and.returnValues(false, false); - component.openCreateReportWithSelectedIds('newReport'); - tick(100); - expect(trackingService.addToReport).toHaveBeenCalled(); - expect(component.showNewReportModal).toHaveBeenCalledOnceWith(); - })); - - it('should call reportableExpenseDialogHandler if totalUnreportableCount greater than 0', fakeAsync(() => { - component.selectedElements = cloneDeep(apiExpenses1); - component.isReportableExpensesSelected = true; - sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(false, false); - sharedExpenseService.isExpenseInDraft.and.returnValues(false, true); - component.openCreateReportWithSelectedIds('newReport'); - tick(100); - expect(trackingService.addToReport).toHaveBeenCalled(); - expect(component.reportableExpenseDialogHandler).toHaveBeenCalledWith(1, 0, 0, 'newReport'); - })); - }); - - describe('when restrictPendingTransactionsEnabled is true', () => { - beforeEach(() => { - component.restrictPendingTransactionsEnabled = true; - }); - it('should call showNonReportableExpenseSelectedToast and return if selectedElement length is zero', fakeAsync(() => { - const expenses = cloneDeep(apiExpenses1); - component.selectedElements = expenses.map((expense) => { - return { ...expense, id: null }; - }); - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - 'Please select one or more expenses to be reported' - ); - expect(component.openCriticalPolicyViolationPopOver).not.toHaveBeenCalled(); - expect(component.showOldReportsMatBottomSheet).not.toHaveBeenCalled(); - expect(component.showNewReportModal).not.toHaveBeenCalled(); - })); - it('should call doesExpenseHavePendingCardTransaction', fakeAsync(() => { - component.selectedElements = cloneDeep(apiExpenses1); - sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(true, true); - sharedExpenseService.isExpenseInDraft.and.returnValues(false, true); - component.restrictPendingTransactionsEnabled = true; - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - expect(sharedExpenseService.doesExpenseHavePendingCardTransaction).toHaveBeenCalledTimes(2); - expect(sharedExpenseService.doesExpenseHavePendingCardTransaction).toHaveBeenCalledWith(apiExpenses1[0]); - expect(sharedExpenseService.doesExpenseHavePendingCardTransaction).toHaveBeenCalledWith(apiExpenses1[1]); - component.isReportableExpensesSelected = false; - expect(component.unreportableExpenseExceptionHandler).toHaveBeenCalledOnceWith(1, 2, 0); - })); - }); - }); - - describe('unreportableExpenseExceptionHandler():', () => { - beforeEach(() => { - spyOn(component, 'showNonReportableExpenseSelectedToast'); - // sharedExpenseService.restrictPendingTransactionsEnabled.and.returnValues(true); - }); - it('should call showNonReportableExpenseSelectedToast when mix of expense types are selected', () => { - component.unreportableExpenseExceptionHandler(1, 1, 1); - expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - "You can't add draft expenses and expenses with critical policy violation & pending transactions." - ); - }); - - it('should call showNonReportableExpenseSelectedToast when mix of draft and policy violation types are selected', () => { - component.unreportableExpenseExceptionHandler(1, 1, 0); - expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - "You can't add draft expenses & expenses with critical policy violations to a report." - ); - }); - }); - - describe('reportableExpenseDialogHandler():', () => { - beforeEach(() => { - spyOn(component, 'openCriticalPolicyViolationPopOver'); - // sharedExpenseService.restrictPendingTransactionsEnabled.and.returnValues(true); - }); - describe('reportableExpenseDialogHandler():', () => { - it('should set proper message when only draft count is greater than 0', () => { - component.reportableExpenseDialogHandler(1, 0, 0, 'newReport'); - expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ - title: "Can't add these expenses...", - message: '1 expense is in draft state.', - reportType: 'newReport', - }); - }); - - it('should set proper message when only policy violation count is greater than 0', () => { - component.reportableExpenseDialogHandler(0, 1, 0, 'newReport'); - expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ - title: "Can't add these expenses...", - message: '1 expense with Critical Policy Violations.', - reportType: 'newReport', - }); - }); - - it('should set proper message when only pendingTransactionsCount count is greater than 0', () => { - component.reportableExpenseDialogHandler(0, 0, 1, 'newReport'); - expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ - title: "Can't add these expenses...", - message: '1 expense with pending transactions.', - reportType: 'newReport', - }); - }); - - it('should set proper message when policy violation and pendingTransactionsCount count is greater than 0', () => { - component.reportableExpenseDialogHandler(0, 1, 1, 'newReport'); - expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ - title: "Can't add these expenses...", - message: '1 expense with pending transactions.

1 expense with Critical Policy Violations.', - reportType: 'newReport', - }); - }); - }); - }); - - it('showNewReportModal(): should open modalController and call showAddToReportSuccessToast', fakeAsync(() => { - component.selectedElements = apiExpenses1; - sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); - const addExpenseToNewReportModalSpy = jasmine.createSpyObj('addExpenseToNewReportModal', [ - 'present', - 'onDidDismiss', - ]); - addExpenseToNewReportModalSpy.onDidDismiss.and.resolveTo({ - data: { report: apiExtendedReportRes[0], message: 'new report is created' }, - }); - modalController.create.and.resolveTo(addExpenseToNewReportModalSpy); - modalProperties.getModalDefaultProperties.and.returnValue(fyModalProperties); - spyOn(component, 'showAddToReportSuccessToast'); - - component.showNewReportModal(); - tick(100); - expect(modalController.create).toHaveBeenCalledOnceWith(newReportModalParams2); - expect(component.showAddToReportSuccessToast).toHaveBeenCalledOnceWith({ - report: apiExtendedReportRes[0], - message: 'new report is created', - }); - })); - - it('openCreateReport(): should navigate to my_create_report', () => { - component.openCreateReport(); - - expect(trackingService.clickCreateReport).toHaveBeenCalledTimes(1); - - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_create_report']); - }); - - describe('openReviewExpenses(): ', () => { - beforeEach(() => { - component.loadExpenses$ = new BehaviorSubject({ pageNumber: 1 }); - - component.selectedElements = apiExpenses1; - expensesService.getAllExpenses.and.returnValue(of(apiExpenses1)); - spyOn(component, 'filterExpensesBySearchString').and.returnValue(true); - - expensesService.getExpenseById.withArgs(apiExpenses1[0].id).and.returnValue(of(apiExpenses1[0])); - expensesService.getExpenseById.withArgs(apiExpenses1[1].id).and.returnValue(of(apiExpenses1[1])); - loaderService.showLoader.and.resolveTo(); - loaderService.hideLoader.and.resolveTo(true); - }); - - it('should call getAllExpenses if sortParams and sortDir is undefined in loadData$ and selectedElement length is zero', fakeAsync(() => { - component.selectedElements = []; - component.openReviewExpenses(); - tick(100); - - expect(expensesService.getAllExpenses).toHaveBeenCalledOnceWith({ - queryParams: Object({ report_id: 'is.null', state: 'in.(COMPLETE,DRAFT)' }), - order: 'spent_at.desc,created_at.desc,id.desc', - }); - expect(component.filterExpensesBySearchString).not.toHaveBeenCalled(); - })); - - it('should call getAllExpenses and filterExpensesBySearchString if searchString, sortParams and sortDir are defined in loadData$ and selectedElement length is zero', fakeAsync(() => { - component.loadExpenses$ = new BehaviorSubject({ - sortDir: 'asc', - sortParam: 'category->name', - searchString: 'example', - }); - component.selectedElements = []; - component.openReviewExpenses(); - tick(100); - - expect(expensesService.getAllExpenses).toHaveBeenCalledOnceWith({ - queryParams: { report_id: 'is.null', state: 'in.(COMPLETE,DRAFT)' }, - order: 'category->name.asc', - }); - expect(component.filterExpensesBySearchString).toHaveBeenCalledTimes(2); - expect(component.filterExpensesBySearchString).toHaveBeenCalledWith(apiExpenses1[0], 'example'); - })); - - it('should navigate to add_edit_mileage if org_category is mileage and selectedElement length is greater than zero', fakeAsync(() => { - component.selectedElements = [mileageExpenseWithDistance, apiExpenses1[1]]; - expensesService.getAllExpenses.and.returnValue(of([mileageExpenseWithDistance, apiExpenses1[1]])); - expensesService.getExpenseById.and.returnValue(of(mileageExpenseWithDistance)); - component.openReviewExpenses(); - tick(100); - - expect(loaderService.showLoader).toHaveBeenCalledTimes(1); - expect(expensesService.getExpenseById).toHaveBeenCalledOnceWith(mileageExpenseWithDistance.id); - expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_mileage', - { id: 'txcSFe6efB6R', txnIds: JSON.stringify(['txcSFe6efB6R', 'tx5WDG9lxBDT']), activeIndex: 0 }, - ]); - })); - - it('should navigate to add_edit_per_diem if org_category is Per Diem and selectedElement length is greater than zero', fakeAsync(() => { - component.selectedElements = [perDiemExpenseWithSingleNumDays, apiExpenses1[1]]; - expensesService.getAllExpenses.and.returnValue(of([perDiemExpenseWithSingleNumDays, apiExpenses1[1]])); - expensesService.getExpenseById.and.returnValue(of(perDiemExpenseWithSingleNumDays)); - - component.openReviewExpenses(); - tick(100); - - expect(loaderService.showLoader).toHaveBeenCalledTimes(1); - expect(expensesService.getExpenseById).toHaveBeenCalledOnceWith(perDiemExpenseWithSingleNumDays.id); - expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_per_diem', - { id: 'txcSFe6efB6R', txnIds: JSON.stringify(['txcSFe6efB6R', 'tx5WDG9lxBDT']), activeIndex: 0 }, - ]); - })); - - it('should navigate to add_edit_expense if org_category is not amongst mileage and per diem and selectedElement length is greater than zero', fakeAsync(() => { - component.selectedElements = apiExpenses1; - expensesService.getAllExpenses.and.returnValue(of(apiExpenses1)); - expensesService.getExpenseById.and.returnValue(of(apiExpenses1[0])); - component.openReviewExpenses(); - tick(100); - - expect(loaderService.showLoader).toHaveBeenCalledTimes(1); - expect(expensesService.getExpenseById).toHaveBeenCalledOnceWith(apiExpenses1[0].id); - expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'add_edit_expense', - { id: 'txDDLtRaflUW', txnIds: '["txDDLtRaflUW","tx5WDG9lxBDT"]', activeIndex: 0 }, - ]); - })); - }); - - describe('filterExpensesBySearchString(): ', () => { - it('should return true if expense consist of searchString', () => { - const expectedFilteredExpenseRes = component.filterExpensesBySearchString(expenseData, 'usvKA4X8Ugcr'); - - expect(expectedFilteredExpenseRes).toBeTrue(); - }); - - it('should return false if expense does not consist of searchString', () => { - const expectedFilteredExpenseRes = component.filterExpensesBySearchString(expenseData, 'Software'); - - expect(expectedFilteredExpenseRes).toBeFalse(); - }); - }); - - it('onAddTransactionToReport(): should open modalController and doRefresh', fakeAsync(() => { - const addExpenseToReportModalSpy = jasmine.createSpyObj('addExpenseToReportModal', ['present', 'onDidDismiss']); - addExpenseToReportModalSpy.onDidDismiss.and.resolveTo({ data: { reload: true } }); - modalController.create.and.resolveTo(addExpenseToReportModalSpy); - modalProperties.getModalDefaultProperties.and.returnValue(fyModalProperties); - spyOn(component, 'doRefresh'); - - component.onAddTransactionToReport({ tx_id: '12345' }); - tick(100); - - expect(modalController.create).toHaveBeenCalledOnceWith(addExpenseToReportModalParams2); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - - describe('showAddToReportSuccessToast():', () => { - let expensesAddedToReportSnackBarSpy: jasmine.SpyObj>; - beforeEach(() => { - expensesAddedToReportSnackBarSpy = jasmine.createSpyObj('expensesAddedToReportSnackBar', ['onAction']); - expensesAddedToReportSnackBarSpy.onAction.and.returnValue(of(undefined)); - matSnackBar.openFromComponent.and.returnValue(expensesAddedToReportSnackBarSpy); - snackbarProperties.setSnackbarProperties.and.returnValue(snackbarPropertiesRes2); - spyOn(component, 'doRefresh'); - }); - - it('should navigate to my_view_report and open matSnackbar', () => { - component.showAddToReportSuccessToast({ - message: 'Expense added to report successfully', - report: apiExtendedReportRes[0], - }); - - expect(matSnackBar.openFromComponent).toHaveBeenCalledOnceWith(ToastMessageComponent, { - ...snackbarPropertiesRes2, - panelClass: ['msb-success-with-camera-icon'], - }); - expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('success', { - message: 'Expense added to report successfully', - redirectionText: 'View Report', - }); - expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ - ToastContent: 'Expense added to report successfully', - }); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toEqual(HeaderState.base); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'my_view_report', - { id: 'rprAfNrce73O', navigateBack: true }, - ]); - }); - - it('should navigate to my_view_report with newly created report id in case of adding it to new report and open matSnackbar', () => { - component.showAddToReportSuccessToast({ - message: 'Expense added to report successfully', - report: reportUnflattenedData, - }); - - expect(matSnackBar.openFromComponent).toHaveBeenCalledOnceWith(ToastMessageComponent, { - ...snackbarPropertiesRes2, - panelClass: ['msb-success-with-camera-icon'], - }); - expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('success', { - message: 'Expense added to report successfully', - redirectionText: 'View Report', - }); - expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ - ToastContent: 'Expense added to report successfully', - }); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toEqual(HeaderState.base); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'my_view_report', - { id: 'rp6LK3ghVatB', navigateBack: true }, - ]); - }); - }); - - it('addTransactionsToReport(): should show loader call reportService and hide the loader', (done) => { - loaderService.showLoader.and.resolveTo(); - loaderService.hideLoader.and.resolveTo(true); - - reportService.addTransactions.and.returnValue(of()); - component - .addTransactionsToReport(apiExtendedReportRes[0], ['tx5fBcPBAxLv']) - .pipe( - tap((updatedReport) => { - expect(loaderService.showLoader).toHaveBeenCalledOnceWith('Adding transaction to report'); - expect(reportService.addTransactions).toHaveBeenCalledOnceWith('rprAfNrce73O', ['tx5fBcPBAxLv']); - expect(updatedReport).toEqual(apiExtendedReportRes[0]); - }), - finalize(() => { - expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); - }) - ) - .subscribe(noop); - done(); - }); - - describe('showOldReportsMatBottomSheet(): ', () => { - beforeEach(() => { - component.selectedElements = apiExpenses1; - component.isNewReportsFlowEnabled = true; - component.openReports$ = of(apiExtendedReportRes); - sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); - spyOn(component, 'showAddToReportSuccessToast'); - }); - - it('should call matBottomSheet.open and call showAddToReportSuccessToast if data.report is defined', () => { - spyOn(component, 'addTransactionsToReport').and.returnValue(of(apiExtendedReportRes[0])); - - matBottomsheet.open.and.returnValue({ - afterDismissed: () => - of({ - report: apiExtendedReportRes[0], - }), - } as MatBottomSheetRef); - - component.showOldReportsMatBottomSheet(); - - expect(matBottomsheet.open).toHaveBeenCalledOnceWith(AddTxnToReportDialogComponent, { - data: { openReports: apiExtendedReportRes, isNewReportsFlowEnabled: true }, - panelClass: ['mat-bottom-sheet-1'], - }); - expect(component.addTransactionsToReport).toHaveBeenCalledOnceWith(apiExtendedReportRes[0], [ - 'txDDLtRaflUW', - 'tx5WDG9lxBDT', - ]); - expect(component.showAddToReportSuccessToast).toHaveBeenCalledOnceWith({ - message: 'Expenses added to report successfully', - report: apiExtendedReportRes[0], - }); - }); - - it('should call matBottomSheet.open and call showAddToReportSuccessToast if data.report is defined and rp_state is draft', () => { - const mockReportData = cloneDeep(apiExtendedReportRes); - mockReportData[0].rp_state = 'DRAFT'; - component.openReports$ = of(mockReportData); - spyOn(component, 'addTransactionsToReport').and.returnValue(of(mockReportData[0])); - matBottomsheet.open.and.returnValue({ - afterDismissed: () => - of({ - report: mockReportData[0], - }), - } as MatBottomSheetRef); - - component.showOldReportsMatBottomSheet(); - expect(matBottomsheet.open).toHaveBeenCalledOnceWith(AddTxnToReportDialogComponent, { - data: { openReports: mockReportData, isNewReportsFlowEnabled: true }, - panelClass: ['mat-bottom-sheet-1'], - }); - - expect(component.addTransactionsToReport).toHaveBeenCalledOnceWith(mockReportData[0], [ - 'txDDLtRaflUW', - 'tx5WDG9lxBDT', - ]); - expect(component.showAddToReportSuccessToast).toHaveBeenCalledOnceWith({ - message: 'Expenses added to an existing draft report', - report: mockReportData[0], - }); - }); - - it('should call matBottomSheet.open and should not call showAddToReportSuccessToast if data.report is null', () => { - spyOn(component, 'addTransactionsToReport'); - matBottomsheet.open.and.returnValue({ - afterDismissed: () => - of({ - report: null, - }), - } as MatBottomSheetRef); - - component.showOldReportsMatBottomSheet(); - expect(matBottomsheet.open).toHaveBeenCalledOnceWith(AddTxnToReportDialogComponent, { - data: { openReports: apiExtendedReportRes, isNewReportsFlowEnabled: true }, - panelClass: ['mat-bottom-sheet-1'], - }); - - expect(component.addTransactionsToReport).not.toHaveBeenCalled(); - expect(component.showAddToReportSuccessToast).not.toHaveBeenCalled(); - }); - }); - - it('openActionSheet(): should open actionSheetController', fakeAsync(() => { - const actionSheetSpy = jasmine.createSpyObj('actionSheet', ['present']); - component.actionSheetButtons = []; - actionSheetController.create.and.returnValue(actionSheetSpy); - - component.openActionSheet(); - tick(100); - - expect(actionSheetController.create).toHaveBeenCalledOnceWith({ - header: 'ADD EXPENSE', - mode: 'md', - cssClass: 'fy-action-sheet', - buttons: [], - }); - })); - - describe('deleteSelectedExpenses(): ', () => { - beforeEach(() => { - component.pendingTransactions = []; - component.expensesToBeDeleted = apiExpenses1; - }); - it('should update selectedElements and call deleteBulk method if expenseToBeDeleted is defined', () => { - component.deleteSelectedExpenses([]); - expect(transactionOutboxService.deleteBulkOfflineExpenses).not.toHaveBeenCalledOnceWith([], []); - expect(component.selectedElements).toEqual(apiExpenses1); - expect(transactionService.deleteBulk).toHaveBeenCalledOnceWith(['txDDLtRaflUW', 'tx5WDG9lxBDT']); - }); - - it('should not call deleteBulk method if tx_id is not present in expensesToBeDeleted', () => { - const mockExpensesWithoutId = cloneDeep([apiExpenses1[0]]); - mockExpensesWithoutId[0].id = undefined; - component.expensesToBeDeleted = mockExpensesWithoutId; - component.deleteSelectedExpenses(null); - expect(transactionOutboxService.deleteBulkOfflineExpenses).not.toHaveBeenCalledOnceWith([], []); - expect(component.selectedElements).toEqual([]); - expect(transactionService.deleteBulk).not.toHaveBeenCalled(); - }); - - it('should delete outbox expenses', () => { - component.deleteSelectedExpenses(expenseList4); - - expect(transactionOutboxService.deleteBulkOfflineExpenses).toHaveBeenCalledOnceWith( - component.pendingTransactions, - expenseList4 - ); - }); - }); - - describe('openDeleteExpensesPopover(): ', () => { - beforeEach(() => { - sharedExpenseService.getExpenseDeletionMessage.and.returnValue('You are about to delete this expense'); - sharedExpenseService.getCCCExpenseMessage.and.returnValue( - 'There are 2 corporate credit cards which can be deleted' - ); - sharedExpenseService.getDeleteDialogBody.and.returnValue('Once deleted, the action cannot be undone'); - component.expensesToBeDeleted = apiExpenses1; - component.cccExpenses = 1; - transactionService.deleteBulk.and.returnValue(of(txnList)); - snackbarProperties.setSnackbarProperties.and.returnValue(snackbarPropertiesRes3); - spyOn(component, 'doRefresh'); - component.expensesToBeDeleted = cloneDeep(apiExpenses1); - component.selectedElements = cloneDeep(apiExpenses1); - }); - - it('should open a popover and get data of expenses on dismiss', fakeAsync(() => { - const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); - deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); - popoverController.create.and.resolveTo(deletePopOverSpy); - - component.openDeleteExpensesPopover(); - tick(100); - - expect(popoverController.create).toHaveBeenCalledOnceWith({ - component: FyDeleteDialogComponent, - cssClass: 'delete-dialog', - backdropDismiss: false, - componentProps: { - header: 'Delete Expense', - body: 'Once deleted, the action cannot be undone', - ctaText: 'Exclude and Delete', - disableDelete: false, - deleteMethod: jasmine.any(Function), - }, - }); - })); - - it('should open a popover and get data of expenses on dismiss if expensesToBeDeleted and cccExpenses are zero', fakeAsync(() => { - const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); - deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); - popoverController.create.and.resolveTo(deletePopOverSpy); - component.cccExpenses = 0; - component.expensesToBeDeleted = []; - - spyOn(component, 'deleteSelectedExpenses').and.callThrough(); - - component.openDeleteExpensesPopover(); - tick(100); - - expect(popoverController.create).toHaveBeenCalledOnceWith({ - component: FyDeleteDialogComponent, - cssClass: 'delete-dialog', - backdropDismiss: false, - componentProps: { - header: 'Delete Expense', - body: 'Once deleted, the action cannot be undone', - ctaText: 'Delete', - disableDelete: true, - deleteMethod: jasmine.any(Function), - }, - }); - })); - - it('should open a popover and delete offline expenses', fakeAsync(() => { - component.outboxExpensesToBeDeleted = expenseListwithoutID; - const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); - deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); - popoverController.create.and.resolveTo(deletePopOverSpy); - - component.openDeleteExpensesPopover(); - tick(1000); - - expect(popoverController.create).toHaveBeenCalledOnceWith({ - component: FyDeleteDialogComponent, - cssClass: 'delete-dialog', - backdropDismiss: false, - componentProps: { - header: 'Delete Expense', - body: 'Once deleted, the action cannot be undone', - ctaText: 'Exclude and Delete', - disableDelete: false, - deleteMethod: jasmine.any(Function), - }, - }); - })); - - it('should show message using matSnackbar if data is successfully deleted and selectedElements are greater than 1', fakeAsync(() => { - const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); - deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); - popoverController.create.and.resolveTo(deletePopOverSpy); - - component.openDeleteExpensesPopover(); - tick(100); - - expect(matSnackBar.openFromComponent).toHaveBeenCalledOnceWith(ToastMessageComponent, { - ...snackbarPropertiesRes3, - panelClass: ['msb-success-with-camera-icon'], - }); - expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('success', { - message: '2 expenses have been deleted', - }); - expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ - ToastContent: '2 expenses have been deleted', - }); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toEqual(HeaderState.base); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - - it('should show message using matSnackbar if data is successfully deleted and selectedElements is 1', fakeAsync(() => { - const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); - deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); - popoverController.create.and.resolveTo(deletePopOverSpy); - component.expensesToBeDeleted = cloneDeep(apiExpenses1); - component.selectedElements = cloneDeep([apiExpenses1[0]]); - - component.openDeleteExpensesPopover(); - tick(100); - - expect(matSnackBar.openFromComponent).toHaveBeenCalledOnceWith(ToastMessageComponent, { - ...snackbarPropertiesRes3, - panelClass: ['msb-success-with-camera-icon'], - }); - expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('success', { - message: '1 expense has been deleted', - }); - expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ ToastContent: '1 expense has been deleted' }); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toEqual(HeaderState.base); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - - it('should show message using matSnackbar if data cannot be deleted', fakeAsync(() => { - snackbarProperties.setSnackbarProperties.and.returnValue(snackbarPropertiesRes4); - const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); - deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'failure' } }); - popoverController.create.and.resolveTo(deletePopOverSpy); - component.expensesToBeDeleted = cloneDeep(apiExpenses1); - component.selectedElements = cloneDeep([apiExpenses1[0]]); - - component.openDeleteExpensesPopover(); - tick(100); - - expect(matSnackBar.openFromComponent).toHaveBeenCalledOnceWith(ToastMessageComponent, { - ...snackbarPropertiesRes4, - panelClass: ['msb-failure-with-camera-icon'], - }); - expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('failure', { - message: 'We could not delete the expenses. Please try again', - }); - expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ - ToastContent: 'We could not delete the expenses. Please try again', - }); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectionMode).toBeFalse(); - expect(component.headerState).toEqual(HeaderState.base); - expect(component.doRefresh).toHaveBeenCalledTimes(1); - })); - }); - - describe('onSelectAll(): ', () => { - beforeEach(() => { - expensesService.getAllExpenses.and.returnValue(of(cloneDeep(apiExpenses1))); - sharedExpenseService.excludeCCCExpenses.and.returnValue(apiExpenses1); - sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); - spyOn(component, 'setExpenseStatsOnSelect'); - component.loadExpenses$ = new BehaviorSubject({ pageNumber: 1, searchString: 'Bus' }); - }); - - it('should set selectedElement to empty array if checked is false', () => { - component.selectedElements = cloneDeep(apiExpenses1); - component.isReportableExpensesSelected = false; - component.onSelectAll(false); - expect(component.selectedElements).toEqual([]); - expect(sharedExpenseService.getReportableExpenses).toHaveBeenCalledOnceWith([]); - expect(component.isReportableExpensesSelected).toBeTrue(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - }); - - it('should select all pending transactions and update stats', () => { - component.pendingTransactions = expenseList4; - transactionService.getReportableExpenses.and.returnValue(expenseList4); - spyOn(component, 'setOutboxExpenseStatsOnSelect'); - - component.onSelectAll(true); - - expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith(expenseList4); - expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(component.isReportableExpensesSelected).toBeTrue(); - }); - - it('should update selectedElements, allExpensesCount and call apiV2Service if checked is true', () => { - expensesService.getAllExpenses.and.returnValue(of(cloneDeep(apiExpenses1))); - component.outboxExpensesToBeDeleted = apiExpenseRes; - component.pendingTransactions = cloneDeep([]); - component.onSelectAll(true); - expect(component.isReportableExpensesSelected).toBeTrue(); - - expect(expensesService.getAllExpenses).toHaveBeenCalledOnceWith({ - queryParams: { report_id: 'is.null', state: 'in.(COMPLETE,DRAFT)', q: 'Bus:*' }, - }); - expect(sharedExpenseService.excludeCCCExpenses).toHaveBeenCalledOnceWith(apiExpenses1); - expect(component.cccExpenses).toBe(0); - expect(component.selectedElements).toEqual([...apiExpenses1]); - expect(component.allExpensesCount).toBe(2); - expect(component.isReportableExpensesSelected).toBeTrue(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - }); - }); - - it('onSimpleSearchCancel(): should set headerState to base and call clearText', () => { - component.headerState = HeaderState.simpleSearch; - spyOn(component, 'clearText'); - - component.onSimpleSearchCancel(); - - expect(component.headerState).toEqual(HeaderState.base); - expect(component.clearText).toHaveBeenCalledOnceWith('onSimpleSearchCancel'); - }); - - it('onFilterPillsClearAll(): should call clearFilters', () => { - spyOn(component, 'clearFilters'); - component.onFilterPillsClearAll(); - expect(component.clearFilters).toHaveBeenCalledTimes(1); - }); - - describe('onFilterClick(): ', () => { - beforeEach(() => { - spyOn(component, 'openFilters'); - }); - filterTypeMappings.forEach((filterTypeMapping) => { - it('should call openFilters with Type if argument is state', fakeAsync(() => { - component.onFilterClick(filterTypeMapping.type); - tick(100); - - expect(component.openFilters).toHaveBeenCalledOnceWith(filterTypeMapping.label); - })); - }); - }); - - describe('onFilterClose(): ', () => { - beforeEach(() => { - component.loadExpenses$ = new BehaviorSubject({}); - component.filters = { - sortDir: 'asc', - sortParam: 'tx_org_category', - }; - component.currentPageNumber = 2; - spyOn(component, 'addNewFiltersToParams').and.returnValue({ - pageNumber: 3, - }); - spyOn(component, 'generateFilterPills').and.returnValue(creditTxnFilterPill); - }); - - it('should remove sortDir and sortParam if filterType is sort', () => { - component.onFilterClose('sort'); - - expect(component.filters.sortDir).toBeUndefined(); - expect(component.filters.sortParam).toBeUndefined(); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((data) => { - expect(data).toEqual({ pageNumber: 3 }); - }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - }); - - it('should remove property from filter if filterType is other than sort', () => { - component.onFilterClose('sortDir'); - expect(component.filters).toEqual({ - sortParam: 'tx_org_category', - }); - expect(component.currentPageNumber).toBe(1); - expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadExpenses$.subscribe((data) => { - expect(data).toEqual({ pageNumber: 3 }); - }); - expect(component.filterPills).toEqual(creditTxnFilterPill); - }); - }); - - it('onHomeClicked(): should navigate to my_dashboard and call trackingService', () => { - component.onHomeClicked(); - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_dashboard'], { - queryParams: { state: 'home' }, - }); - expect(trackingService.footerHomeTabClicked).toHaveBeenCalledOnceWith({ - page: 'Expenses', - }); - }); - - it('onCameraClicked(): should navigate to camera_overlay', () => { - component.onCameraClicked(); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'camera_overlay', - { - navigate_back: true, - }, - ]); - }); - - it('onTaskClicked(): should navigate to my_dashboard and call trackingService', () => { - component.onTaskClicked(); - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_dashboard'], { - queryParams: { state: 'tasks', tasksFilters: 'expenses' }, - }); - expect(trackingService.tasksPageOpened).toHaveBeenCalledOnceWith({ - Asset: 'Mobile', - from: 'My Expenses', - }); - }); - - it('searchClick(): should set headerState and call focus method on input', fakeAsync(() => { - component.simpleSearchInput = fixture.debugElement.query(By.css('.my-expenses--simple-search-input')); - inputElement = component.simpleSearchInput.nativeElement; - const mockFocus = spyOn(inputElement, 'focus'); - - component.searchClick(); - expect(component.headerState).toEqual(HeaderState.simpleSearch); - tick(300); - expect(mockFocus).toHaveBeenCalledTimes(1); - })); - - it('mergeExpense(): should navigate to merge_expenses with payload data', () => { - component.selectedElements = apiExpenses1; - component.mergeExpenses(); - expect(router.navigate).toHaveBeenCalledOnceWith([ - '/', - 'enterprise', - 'merge_expense', - { - expenseIDs: JSON.stringify(['txDDLtRaflUW', 'tx5WDG9lxBDT']), - from: 'MY_EXPENSES', - }, - ]); - }); - - describe('showCamera(): ', () => { - it('should set isCameraPreviewStarted to false if argument is false', () => { - component.isCameraPreviewStarted = true; - component.showCamera(false); - expect(component.isCameraPreviewStarted).toBeFalse(); - }); - - it('should set isCameraPreviewStarted to true if argument is true', () => { - component.isCameraPreviewStarted = false; - component.showCamera(true); - expect(component.isCameraPreviewStarted).toBeTrue(); - }); - }); - - it('setOutboxExpenseStatsOnSelect(): should update stats on selecting outbox expenses', (done) => { - component.selectedOutboxExpenses = expenseList4; - - component.setOutboxExpenseStatsOnSelect(); - - component.allExpensesStats$.subscribe((res) => { - expect(res).toEqual({ - count: 3, - amount: 49475.76, - }); - done(); - }); - }); - - describe('selectOutboxExpense(): ', () => { - beforeEach(() => { - transactionService.getReportableExpenses.and.returnValue(apiExpenseRes); - component.allExpensesCount = 1; - spyOn(component, 'setExpenseStatsOnSelect'); - spyOn(component, 'setOutboxExpenseStatsOnSelect'); - component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); - transactionService.isMergeAllowed.and.returnValue(true); - transactionService.excludeCCCExpenses.and.returnValue(apiExpenseRes); - }); - - it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses', () => { - transactionService.getReportableExpenses.and.returnValue([]); - const expense = apiExpenseRes[0]; - component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); - - component.selectOutboxExpense(expense); - - expect(component.selectedOutboxExpenses).toEqual([]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeFalse(); - expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); - expect(component.isMergeAllowed).toBeTrue(); - }); - - it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses', () => { - transactionService.getReportableExpenses.and.returnValue([]); - component.allExpensesCount = 4; - const expense = apiExpenseRes[0]; - component.selectedOutboxExpenses = cloneDeep(cloneDeep(expenseList4)); - - component.selectOutboxExpense(expense); - - expect(component.selectedOutboxExpenses).toEqual([...expenseList4, expense]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeTrue(); - expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([...expenseList4, expense]); - expect(component.isMergeAllowed).toBeTrue(); - }); - - it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses and allExpenseCount is not equal to length of selectedOutboxExpenses', () => { - transactionService.getReportableExpenses.and.returnValue([]); - const expense = apiExpenseRes[0]; - component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); - - component.selectOutboxExpense(expense); - - expect(component.selectedOutboxExpenses).toEqual([]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeFalse(); - expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); - expect(component.isMergeAllowed).toBeTrue(); - }); - - it('should update expenseToBeDeleted if selectedOutboxExpenses is an array of atleast 1', () => { - component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); - component.selectOutboxExpense(expenseData2); - - const expectedSelectedElements = [...apiExpenseRes, expenseData2]; - expect(component.selectedOutboxExpenses).toEqual(expectedSelectedElements); - expect(component.outboxExpensesToBeDeleted).toEqual(apiExpenseRes); - expect(component.cccExpenses).toBe(1); - expect(component.selectAll).toBeFalse(); - }); - - it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses and tx_id is not present in expense', () => { - transactionService.getReportableExpenses.and.returnValue([]); - component.allExpensesCount = 0; - const expense = cloneDeep(apiExpenseRes[0]); - expense.tx_id = undefined; - component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); - component.selectedOutboxExpenses[0].tx_id = undefined; - - component.selectOutboxExpense(expense); - - expect(component.selectedOutboxExpenses).toEqual([]); - expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeTrue(); - expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); - expect(component.isMergeAllowed).toBeTrue(); - }); - }); - - describe('checkDeleteDisabled():', () => { - it('should check and enable the button for online mode', (done) => { - component.isConnected$ = of(true); - component.selectedElements = apiExpenses1; - component.expensesToBeDeleted = []; - - component.checkDeleteDisabled().subscribe(() => { - expect(component.isDisabled).toBeFalse(); - done(); - }); - }); - - it('should check and enable the button for offline mode', (done) => { - component.isConnected$ = of(false); - component.selectedOutboxExpenses = apiExpenseRes; - component.outboxExpensesToBeDeleted = []; - - component.checkDeleteDisabled().subscribe(() => { - expect(component.isDisabled).toBeFalse(); - done(); - }); - }); - }); -}); diff --git a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.ts b/src/app/fyle/my-expenses-v2/my-expenses-v2.page.ts deleted file mode 100644 index 1f83dc1b72..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses-v2.page.ts +++ /dev/null @@ -1,1650 +0,0 @@ -import { getCurrencySymbol } from '@angular/common'; -import { Component, ElementRef, EventEmitter, OnInit, ViewChild } from '@angular/core'; -import { MatBottomSheet } from '@angular/material/bottom-sheet'; -import { MatSnackBar } from '@angular/material/snack-bar'; -import { ActivatedRoute, Params, Router } from '@angular/router'; -import { ActionSheetController, ModalController, NavController, PopoverController } from '@ionic/angular'; -import { cloneDeep, isEqual } from 'lodash'; -import { - BehaviorSubject, - Observable, - Subject, - Subscription, - concat, - forkJoin, - from, - fromEvent, - iif, - noop, - of, -} from 'rxjs'; -import { - debounceTime, - distinctUntilChanged, - filter, - finalize, - map, - shareReplay, - switchMap, - take, - takeUntil, -} from 'rxjs/operators'; -import { BackButtonActionPriority } from 'src/app/core/models/back-button-action-priority.enum'; -import { CardAggregateStats } from 'src/app/core/models/card-aggregate-stats.model'; -import { Expense } from 'src/app/core/models/expense.model'; -import { OrgSettings } from 'src/app/core/models/org-settings.model'; -import { ExpenseFilters } from 'src/app/core/models/platform/expense-filters.model'; -import { PlatformCategory } from 'src/app/core/models/platform/platform-category.model'; -import { Expense as PlatformExpense } from 'src/app/core/models/platform/v1/expense.model'; -import { GetExpenseQueryParam } from 'src/app/core/models/platform/v1/get-expenses-query.model'; -import { ReportV1 } from 'src/app/core/models/report-v1.model'; -import { ExtendedReport } from 'src/app/core/models/report.model'; -import { UniqueCardStats } from 'src/app/core/models/unique-cards-stats.model'; -import { UniqueCards } from 'src/app/core/models/unique-cards.model'; -import { Transaction } from 'src/app/core/models/v1/transaction.model'; -import { ApiV2Service } from 'src/app/core/services/api-v2.service'; -import { CategoriesService } from 'src/app/core/services/categories.service'; -import { CorporateCreditCardExpenseService } from 'src/app/core/services/corporate-credit-card-expense.service'; -import { CurrencyService } from 'src/app/core/services/currency.service'; -import { LoaderService } from 'src/app/core/services/loader.service'; -import { ModalPropertiesService } from 'src/app/core/services/modal-properties.service'; -import { NetworkService } from 'src/app/core/services/network.service'; -import { OrgSettingsService } from 'src/app/core/services/org-settings.service'; -import { OrgUserSettingsService } from 'src/app/core/services/org-user-settings.service'; -import { PlatformHandlerService } from 'src/app/core/services/platform-handler.service'; -import { ExpensesService as SharedExpenseService } from 'src/app/core/services/platform/v1/shared/expenses.service'; -import { ExpensesService } from 'src/app/core/services/platform/v1/spender/expenses.service'; -import { PopupService } from 'src/app/core/services/popup.service'; -import { ReportService } from 'src/app/core/services/report.service'; -import { TasksService } from 'src/app/core/services/tasks.service'; -import { TokenService } from 'src/app/core/services/token.service'; -import { TransactionService } from 'src/app/core/services/transaction.service'; -import { TransactionsOutboxService } from 'src/app/core/services/transactions-outbox.service'; -import { CreateNewReportComponent } from 'src/app/shared/components/create-new-report-v2/create-new-report.component'; -import { SelectedFilters } from 'src/app/shared/components/fy-filters/selected-filters.interface'; -import { PopupAlertComponent } from 'src/app/shared/components/popup-alert/popup-alert.component'; -import { ToastMessageComponent } from 'src/app/shared/components/toast-message/toast-message.component'; -import { MaskNumber } from 'src/app/shared/pipes/mask-number.pipe'; -import { environment } from 'src/environments/environment'; -import { SnackbarPropertiesService } from '../../core/services/snackbar-properties.service'; -import { StorageService } from '../../core/services/storage.service'; -import { TrackingService } from '../../core/services/tracking.service'; -import { FyDeleteDialogComponent } from '../../shared/components/fy-delete-dialog/fy-delete-dialog.component'; -import { FilterPill } from '../../shared/components/fy-filter-pills/filter-pill.interface'; -import { FilterOptionType } from '../../shared/components/fy-filters/filter-option-type.enum'; -import { FilterOptions } from '../../shared/components/fy-filters/filter-options.interface'; -import { FyFiltersComponent } from '../../shared/components/fy-filters/fy-filters.component'; -import { HeaderState } from '../../shared/components/fy-header/header-state.enum'; -import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog/add-txn-to-report-dialog.component'; -import { MyExpensesService } from './my-expenses.service'; - -@Component({ - selector: 'app-my-expenses', - templateUrl: './my-expenses-v2.page.html', - styleUrls: ['./my-expenses-v2.page.scss'], -}) -export class MyExpensesV2Page implements OnInit { - @ViewChild('simpleSearchInput') simpleSearchInput: ElementRef; - - isConnected$: Observable; - - myExpenses$: Observable; - - count$: Observable; - - isInfiniteScrollRequired$: Observable; - - loadExpenses$: BehaviorSubject>; - - currentPageNumber = 1; - - acc: PlatformExpense[] = []; - - filters: Partial; - - allExpensesStats$: Observable<{ count: number; amount: number }>; - - draftExpensesCount$: Observable; - - homeCurrency$: Observable; - - isInstaFyleEnabled$: Observable; - - isBulkFyleEnabled$: Observable; - - isMileageEnabled$: Observable; - - isPerDiemEnabled$: Observable; - - orgSettings$: Observable; - - specialCategories$: Observable; - - pendingTransactions: Partial[] = []; - - selectionMode = false; - - selectedElements: PlatformExpense[]; - - selectedOutboxExpenses: Partial[] = []; - - syncing = false; - - simpleSearchText = ''; - - allExpenseCountHeader$: Observable; - - navigateBack = false; - - openAddExpenseListLoader = false; - - clusterDomain: string; - - isNewUser$: Observable; - - isLoading = false; - - headerState: HeaderState = HeaderState.base; - - actionSheetButtons: { text: string; icon: string; cssClass: string; handler: () => void }[] = []; - - selectAll = false; - - filterPills = []; - - reviewMode = false; - - ROUTER_API_ENDPOINT: string; - - isReportableExpensesSelected = false; - - isSearchBarFocused = false; - - openReports$: Observable; - - homeCurrencySymbol: string; - - isLoadingDataInInfiniteScroll: boolean; - - allExpensesCount: number; - - onPageExit$ = new Subject(); - - expensesTaskCount = 0; - - isCameraPreviewStarted = false; - - cardNumbers: { label: string; value: string }[] = []; - - maskNumber = new MaskNumber(); - - expensesToBeDeleted: PlatformExpense[]; - - outboxExpensesToBeDeleted: Partial[] = []; - - cccExpenses: number; - - isMergeAllowed: boolean; - - hardwareBackButton: Subscription; - - isNewReportsFlowEnabled = false; - - isDisabled = false; - - restrictPendingTransactionsEnabled = false; - - constructor( - private networkService: NetworkService, - private loaderService: LoaderService, - private modalController: ModalController, - private transactionService: TransactionService, - private popoverController: PopoverController, - private router: Router, - private transactionOutboxService: TransactionsOutboxService, - private activatedRoute: ActivatedRoute, - private popupService: PopupService, - private trackingService: TrackingService, - private storageService: StorageService, - private tokenService: TokenService, - private apiV2Service: ApiV2Service, - private modalProperties: ModalPropertiesService, - private reportService: ReportService, - private matBottomSheet: MatBottomSheet, - private matSnackBar: MatSnackBar, - private actionSheetController: ActionSheetController, - private snackbarProperties: SnackbarPropertiesService, - private tasksService: TasksService, - private corporateCreditCardService: CorporateCreditCardExpenseService, - private myExpensesService: MyExpensesService, - private orgSettingsService: OrgSettingsService, - private currencyService: CurrencyService, - private orgUserSettingsService: OrgUserSettingsService, - private platformHandlerService: PlatformHandlerService, - private categoriesService: CategoriesService, - private navController: NavController, - private expenseService: ExpensesService, - private sharedExpenseService: SharedExpenseService - ) {} - - get HeaderState(): typeof HeaderState { - return HeaderState; - } - - clearText(isFromCancel: string): void { - this.simpleSearchText = ''; - const searchInput = this.simpleSearchInput.nativeElement; - searchInput.value = ''; - searchInput.dispatchEvent(new Event('keyup')); - if (isFromCancel === 'onSimpleSearchCancel') { - this.isSearchBarFocused = !this.isSearchBarFocused; - } else { - this.isSearchBarFocused = !!this.isSearchBarFocused; - } - } - - onSearchBarFocus(): void { - this.isSearchBarFocused = true; - } - - ngOnInit(): void { - this.setupNetworkWatcher(); - } - - formatTransactions(transactions: Partial[]): Partial[] { - return transactions.map((transaction) => { - const formattedTxn = >{}; - Object.keys(transaction).forEach((key: keyof Partial) => { - formattedTxn['tx_' + key] = transaction[key]; - }); - return formattedTxn; - }); - } - - switchSelectionMode(expense?: PlatformExpense): void { - this.selectionMode = !this.selectionMode; - if (!this.selectionMode) { - if (this.loadExpenses$.getValue().searchString) { - this.headerState = HeaderState.simpleSearch; - } else { - this.headerState = HeaderState.base; - } - - this.selectedElements = []; - this.setAllExpensesCountAndAmount(); - } else { - this.headerState = HeaderState.multiselect; - // setting Expense amount & count stats to zero on select init - this.allExpensesStats$ = of({ - count: 0, - amount: 0, - }); - } - - if (expense) { - this.selectExpense(expense); - } - } - - switchOutboxSelectionMode(expense?: Expense): void { - this.selectionMode = !this.selectionMode; - if (!this.selectionMode) { - if (this.loadExpenses$.getValue().searchString) { - this.headerState = HeaderState.simpleSearch; - } else { - this.headerState = HeaderState.base; - } - - this.selectedOutboxExpenses = []; - this.setOutboxExpenseStatsOnSelect(); - } else { - this.headerState = HeaderState.multiselect; - // setting Expense amount & count stats to zero on select init - this.allExpensesStats$ = of({ - count: 0, - amount: 0, - }); - } - - if (expense) { - this.selectOutboxExpense(expense); - } - } - - async sendFirstExpenseCreatedEvent(): Promise { - // checking if the expense is first expense - const isFirstExpenseCreated = await this.storageService.get('isFirstExpenseCreated'); - - // for first expense etxnc size will be 0 - if (!isFirstExpenseCreated) { - this.allExpensesStats$.subscribe(async (res) => { - if (res.count === 0) { - this.trackingService.createFirstExpense(); - await this.storageService.set('isFirstExpenseCreated', true); - } - }); - } - } - - setAllExpensesCountAndAmount(): void { - this.allExpensesStats$ = this.loadExpenses$.pipe( - switchMap((params) => { - const queryParams = cloneDeep(params.queryParams) || {}; - - queryParams.report_id = (queryParams.report_id || 'is.null') as string; - queryParams.state = 'in.(COMPLETE,DRAFT)'; - - if (queryParams['matched_corporate_card_transactions->0->corporate_card_number']) { - const cardParamsCopy = queryParams['matched_corporate_card_transactions->0->corporate_card_number'] as string; - - queryParams.or = (queryParams.or || []) as string[]; - queryParams.or.push('(matched_corporate_card_transactions->0->corporate_card_number.' + cardParamsCopy + ')'); - delete queryParams['matched_corporate_card_transactions->0->corporate_card_number']; - } - - return this.expenseService.getExpenseStats(queryParams).pipe( - map((stats) => ({ - count: stats.data.count, - amount: stats.data.total_amount, - })) - ); - }) - ); - } - - actionSheetButtonsHandler(action: string, route: string) { - return (): void => { - this.trackingService.myExpensesActionSheetAction({ - Action: action, - }); - this.router.navigate([ - '/', - 'enterprise', - route, - { - navigate_back: true, - }, - ]); - }; - } - - setupActionSheet(orgSettings: OrgSettings, allowedExpenseTypes: Record): void { - const that = this; - const mileageEnabled = orgSettings.mileage.enabled && allowedExpenseTypes.mileage; - const isPerDiemEnabled = orgSettings.per_diem.enabled && allowedExpenseTypes.perDiem; - that.actionSheetButtons = [ - { - text: 'Capture Receipt', - icon: 'assets/svg/camera.svg', - cssClass: 'capture-receipt', - handler: this.actionSheetButtonsHandler('capture receipts', 'camera_overlay'), - }, - { - text: 'Add Manually', - icon: 'assets/svg/list.svg', - cssClass: 'capture-receipt', - handler: this.actionSheetButtonsHandler('Add Expense', 'add_edit_expense'), - }, - ]; - - if (mileageEnabled) { - that.actionSheetButtons.push({ - text: 'Add Mileage', - icon: 'assets/svg/mileage.svg', - cssClass: 'capture-receipt', - handler: this.actionSheetButtonsHandler('Add Mileage', 'add_edit_mileage'), - }); - } - - if (isPerDiemEnabled) { - that.actionSheetButtons.push({ - text: 'Add Per Diem', - icon: 'assets/svg/calendar.svg', - cssClass: 'capture-receipt', - handler: this.actionSheetButtonsHandler('Add Per Diem', 'add_edit_per_diem'), - }); - } - } - - getCardDetail(statsResponses: CardAggregateStats[]): UniqueCardStats[] { - const cardNames: { cardNumber: string; cardName: string }[] = []; - statsResponses.forEach((response) => { - const cardDetail = { - cardNumber: response.key[1].column_value, - cardName: response.key[0].column_value, - }; - cardNames.push(cardDetail); - }); - const uniqueCards = JSON.parse(JSON.stringify(cardNames)) as UniqueCards[]; - - return this.corporateCreditCardService.getExpenseDetailsInCards(uniqueCards, statsResponses); - } - - ionViewWillLeave(): void { - this.hardwareBackButton.unsubscribe(); - this.onPageExit$.next(null); - } - - backButtonAction(): void { - if (this.headerState === HeaderState.multiselect) { - this.switchSelectionMode(); - } else if (this.headerState === HeaderState.simpleSearch) { - this.onSimpleSearchCancel(); - } else { - this.navController.back(); - } - } - - ionViewWillEnter(): void { - this.isNewReportsFlowEnabled = false; - this.hardwareBackButton = this.platformHandlerService.registerBackButtonAction( - BackButtonActionPriority.MEDIUM, - this.backButtonAction - ); - - this.tasksService.getExpensesTaskCount().subscribe((expensesTaskCount) => { - this.expensesTaskCount = expensesTaskCount; - }); - - const getOrgUserSettingsService$ = this.orgUserSettingsService.get().pipe(shareReplay(1)); - - this.isInstaFyleEnabled$ = getOrgUserSettingsService$.pipe( - map( - (orgUserSettings) => - orgUserSettings?.insta_fyle_settings?.allowed && orgUserSettings.insta_fyle_settings.enabled - ) - ); - - this.isBulkFyleEnabled$ = getOrgUserSettingsService$.pipe( - map((orgUserSettings) => orgUserSettings?.bulk_fyle_settings?.enabled) - ); - - this.orgSettings$ = this.orgSettingsService.get().pipe(shareReplay(1)); - this.specialCategories$ = this.categoriesService.getMileageOrPerDiemCategories().pipe(shareReplay(1)); - - this.isMileageEnabled$ = this.orgSettings$.pipe(map((orgSettings) => orgSettings?.mileage?.enabled)); - this.isPerDiemEnabled$ = this.orgSettings$.pipe(map((orgSettings) => orgSettings?.per_diem?.enabled)); - - this.orgSettings$.subscribe((orgSettings) => { - this.isNewReportsFlowEnabled = orgSettings?.simplified_report_closure_settings?.enabled || false; - this.restrictPendingTransactionsEnabled = - (orgSettings?.corporate_credit_card_settings?.enabled && - orgSettings?.pending_cct_expense_restriction?.enabled) || - false; - }); - - forkJoin({ - orgSettings: this.orgSettings$, - specialCategories: this.specialCategories$, - }).subscribe(({ orgSettings, specialCategories }) => { - const allowedExpenseTypes = { - mileage: specialCategories.some((category) => category.system_category === 'Mileage'), - perDiem: specialCategories.some((category) => category.system_category === 'Per Diem'), - }; - this.setupActionSheet(orgSettings, allowedExpenseTypes); - }); - - forkJoin({ - isConnected: this.isConnected$.pipe(take(1)), - }) - .pipe( - filter(({ isConnected }) => isConnected), - switchMap(() => this.corporateCreditCardService.getAssignedCards()) - ) - .subscribe((allCards) => { - const cards = this.getCardDetail(allCards.cardDetails); - cards.forEach((card) => { - this.cardNumbers.push({ label: this.maskNumber.transform(card.cardNumber), value: card.cardNumber }); - }); - }); - - this.headerState = HeaderState.base; - - this.isLoading = true; - this.reviewMode = false; - - from(this.tokenService.getClusterDomain()).subscribe((clusterDomain) => { - this.clusterDomain = clusterDomain; - }); - - this.ROUTER_API_ENDPOINT = environment.ROUTER_API_ENDPOINT; - - this.navigateBack = !!this.activatedRoute.snapshot.params.navigateBack; - this.acc = []; - this.simpleSearchText = ''; - - this.currentPageNumber = 1; - - this.loadExpenses$ = new BehaviorSubject({ - pageNumber: 1, - }); - - this.selectionMode = false; - this.selectedElements = []; - - this.syncOutboxExpenses(); - - this.isConnected$.pipe(takeUntil(this.onPageExit$.asObservable())).subscribe((connected) => { - if (connected) { - this.syncOutboxExpenses(); - } - }); - - const getHomeCurrency$ = this.currencyService.getHomeCurrency().pipe(shareReplay(1)); - - this.homeCurrency$ = getHomeCurrency$; - - getHomeCurrency$.subscribe((homeCurrency) => { - this.homeCurrencySymbol = getCurrencySymbol(homeCurrency, 'wide'); - }); - - this.simpleSearchInput.nativeElement.value = ''; - fromEvent<{ srcElement: { value: string } }>(this.simpleSearchInput.nativeElement, 'keyup') - .pipe( - map((event) => event.srcElement.value), - distinctUntilChanged(), - debounceTime(400) - ) - .subscribe((searchString) => { - const currentParams = this.loadExpenses$.getValue(); - currentParams.searchString = searchString; - this.currentPageNumber = 1; - currentParams.pageNumber = this.currentPageNumber; - - this.loadExpenses$.next(currentParams); - }); - - const paginatedPipe = this.loadExpenses$.pipe( - switchMap((params) => { - const queryParams = params.queryParams || {}; - - queryParams.report_id = queryParams.report_id || 'is.null'; - queryParams.state = 'in.(COMPLETE,DRAFT)'; - - if (params.searchString) { - queryParams.q = params.searchString; - queryParams.q = queryParams.q + ':*'; - } else if (params.searchString === '') { - delete queryParams.q; - } - const orderByParams = - params.sortParam && params.sortDir - ? `${params.sortParam}.${params.sortDir}` - : 'spent_at.desc,created_at.desc,id.desc'; - this.isLoadingDataInInfiniteScroll = true; - - return this.expenseService.getExpensesCount(queryParams).pipe( - switchMap((count) => { - if (count > (params.pageNumber - 1) * 10) { - return this.expenseService.getExpenses({ - offset: (params.pageNumber - 1) * 10, - limit: 10, - ...queryParams, - order: orderByParams, - }); - } else { - return of([]); - } - }), - map((res) => { - this.isLoadingDataInInfiniteScroll = false; - if (this.currentPageNumber === 1) { - this.acc = []; - } - this.acc = this.acc.concat(res as PlatformExpense[]); - return this.acc; - }) - ); - }) - ); - - this.myExpenses$ = paginatedPipe.pipe(shareReplay(1)); - - this.count$ = this.loadExpenses$.pipe( - switchMap((params) => { - const queryParams = params.queryParams || {}; - - queryParams.report_id = queryParams.report_id || 'is.null'; - queryParams.state = 'in.(COMPLETE,DRAFT)'; - return this.expenseService.getExpensesCount(queryParams); - }), - shareReplay(1) - ); - - this.isNewUser$ = this.expenseService.getExpensesCount({ offset: 0, limit: 200 }).pipe(map((res) => res === 0)); - - const paginatedScroll$ = this.myExpenses$.pipe( - switchMap((etxns) => this.count$.pipe(map((count) => count > etxns.length))) - ); - - this.isInfiniteScrollRequired$ = this.loadExpenses$.pipe(switchMap(() => paginatedScroll$)); - - this.setAllExpensesCountAndAmount(); - - this.allExpenseCountHeader$ = this.loadExpenses$.pipe( - switchMap(() => - this.expenseService.getExpenseStats({ - state: 'in.(COMPLETE,DRAFT)', - report_id: 'is.null', - }) - ), - map((stats) => stats.data.count) - ); - - this.draftExpensesCount$ = this.loadExpenses$.pipe( - switchMap(() => - this.expenseService.getExpenseStats({ - state: 'in.(DRAFT)', - report_id: 'is.null', - }) - ), - map((stats) => stats.data.count) - ); - - this.loadExpenses$.subscribe(() => { - const queryParams: Params = { filters: JSON.stringify(this.filters) }; - this.router.navigate([], { - relativeTo: this.activatedRoute, - queryParams, - replaceUrl: true, - }); - }); - - this.myExpenses$.subscribe(noop); - this.count$.subscribe(noop); - this.isInfiniteScrollRequired$.subscribe(noop); - if (this.activatedRoute.snapshot.queryParams.filters) { - this.filters = Object.assign( - {}, - this.filters, - JSON.parse(this.activatedRoute.snapshot.queryParams.filters as string) as Partial - ); - this.currentPageNumber = 1; - const params = this.addNewFiltersToParams(); - - this.loadExpenses$.next(params); - this.filterPills = this.generateFilterPills(this.filters); - } else if (this.activatedRoute.snapshot.params.state) { - let filters = {}; - if ((this.activatedRoute.snapshot.params.state as string).toLowerCase() === 'needsreceipt') { - filters = { is_receipt_mandotary: 'eq.true', state: 'NEEDS_RECEIPT' }; - } else if ((this.activatedRoute.snapshot.params.state as string).toLowerCase() === 'policyviolated') { - filters = { - is_policy_flagged: 'eq.true', - or: '(policy_amount.is.null,policy_amount.gt.0.0001)', - state: 'POLICY_VIOLATED', - }; - } else if ((this.activatedRoute.snapshot.params.state as string).toLowerCase() === 'cannotreport') { - filters = { policy_amount: 'lt.0.0001', state: 'CANNOT_REPORT' }; - } - this.filters = Object.assign({}, this.filters, filters); - this.currentPageNumber = 1; - const params = this.addNewFiltersToParams(); - - this.loadExpenses$.next(params); - this.filterPills = this.generateFilterPills(this.filters); - } else { - this.clearFilters(); - } - - setTimeout(() => { - this.isLoading = false; - }, 500); - - const queryParams = { rp_state: 'in.(DRAFT,APPROVER_PENDING,APPROVER_INQUIRY)' }; - - this.openReports$ = this.reportService.getAllExtendedReports({ queryParams }).pipe( - map((openReports) => - openReports.filter( - (openReport) => - // JSON.stringify(openReport.report_approvals).indexOf('APPROVAL_DONE') -> Filter report if any approver approved this report. - // Converting this object to string and checking If `APPROVAL_DONE` is present in the string, removing the report from the list - !openReport.report_approvals || - (openReport.report_approvals && - !(JSON.stringify(openReport.report_approvals).indexOf('APPROVAL_DONE') > -1)) - ) - ) - ); - this.doRefresh(); - - this.checkDeleteDisabled(); - } - - setupNetworkWatcher(): void { - const networkWatcherEmitter = new EventEmitter(); - this.networkService.connectivityWatcher(networkWatcherEmitter); - this.isConnected$ = concat(this.networkService.isOnline(), networkWatcherEmitter.asObservable()); - } - - loadData(event: { target?: { complete?: () => void } }): void { - this.currentPageNumber = this.currentPageNumber + 1; - - const params = this.loadExpenses$.getValue(); - params.pageNumber = this.currentPageNumber; - this.loadExpenses$.next(params); - - setTimeout(() => { - event?.target?.complete(); - }, 1000); - } - - syncOutboxExpenses(): void { - this.pendingTransactions = this.formatTransactions(this.transactionOutboxService.getPendingTransactions()); - if (this.pendingTransactions.length > 0) { - this.syncing = true; - from(this.pendingTransactions) - .pipe( - switchMap(() => from(this.transactionOutboxService.sync())), - finalize(() => { - this.syncing = false; - const pendingTransactions = this.formatTransactions(this.transactionOutboxService.getPendingTransactions()); - if (pendingTransactions.length === 0) { - this.doRefresh(); - } - }) - ) - .subscribe(noop); - } - } - - doRefresh(event?: { target?: { complete?: () => void } }): void { - this.currentPageNumber = 1; - this.selectedElements = []; - if (this.selectionMode) { - this.setExpenseStatsOnSelect(); - } - const params = this.loadExpenses$.getValue(); - params.pageNumber = this.currentPageNumber; - this.transactionService.clearCache().subscribe(() => { - this.loadExpenses$.next(params); - if (event) { - setTimeout(() => { - event.target?.complete(); - }, 1000); - } - }); - } - - generateFilterPills(filter: Partial): FilterPill[] { - const filterPills: FilterPill[] = []; - - if (filter.state?.length > 0) { - this.myExpensesService.generateStateFilterPills(filterPills, filter); - } - - if (filter.receiptsAttached) { - this.myExpensesService.generateReceiptsAttachedFilterPills(filterPills, filter); - } - - if (filter.date) { - this.myExpensesService.generateDateFilterPills(filter, filterPills); - } - - if (filter.type?.length > 0) { - this.myExpensesService.generateTypeFilterPills(filter, filterPills); - } - - if (filter.sortParam && filter.sortDir) { - this.myExpensesService.generateSortFilterPills(filter, filterPills); - } - - if (filter.cardNumbers?.length > 0) { - this.myExpensesService.generateCardFilterPills(filterPills, filter); - } - - if (filter.splitExpense) { - this.myExpensesService.generateSplitExpenseFilterPills(filterPills, filter); - } - - return filterPills; - } - - addNewFiltersToParams(): Partial { - let currentParams = this.loadExpenses$.getValue(); - currentParams.pageNumber = 1; - let newQueryParams: Record = { - or: [], - }; - - newQueryParams = this.sharedExpenseService.generateCardNumberParams(newQueryParams, this.filters); - - newQueryParams = this.sharedExpenseService.generateDateParams(newQueryParams, this.filters); - - newQueryParams = this.sharedExpenseService.generateReceiptAttachedParams(newQueryParams, this.filters); - - newQueryParams = this.sharedExpenseService.generateStateFilters(newQueryParams, this.filters); - - newQueryParams = this.sharedExpenseService.generateTypeFilters(newQueryParams, this.filters); - - currentParams = this.sharedExpenseService.setSortParams(currentParams, this.filters); - - newQueryParams = this.sharedExpenseService.generateSplitExpenseParams(newQueryParams, this.filters); - - currentParams.queryParams = newQueryParams; - - const onlyDraftStateFilterApplied = - this.filters.state && this.filters.state.length === 1 && this.filters.state.includes('DRAFT'); - const onlyCriticalPolicyFilterApplied = - this.filters.state?.length === 1 && this.filters.state.includes('CANNOT_REPORT'); - const draftAndCriticalPolicyFilterApplied = - this.filters.state?.length === 2 && - this.filters.state.includes('DRAFT') && - this.filters.state.includes('CANNOT_REPORT'); - - this.reviewMode = false; - if (onlyDraftStateFilterApplied || onlyCriticalPolicyFilterApplied || draftAndCriticalPolicyFilterApplied) { - this.reviewMode = true; - } - - return currentParams; - } - - async openFilters(activeFilterInitialName?: string): Promise { - const filterMain = this.myExpensesService.getFilters(); - if (this.cardNumbers?.length > 0) { - filterMain.push({ - name: 'Cards', - optionType: FilterOptionType.multiselect, - options: this.cardNumbers, - } as FilterOptions); - } - - const selectedFilters = this.myExpensesService.generateSelectedFilters(this.filters); - - const filterPopover = await this.modalController.create({ - component: FyFiltersComponent, - componentProps: { - filterOptions: filterMain, - selectedFilterValues: selectedFilters, - activeFilterInitialName, - }, - cssClass: 'dialog-popover', - }); - - await filterPopover.present(); - - const { data } = (await filterPopover.onWillDismiss()) as { data: SelectedFilters[] }; - - if (data) { - const filters1 = this.myExpensesService.convertSelectedOptionsToExpenseFilters(data); - this.filters = filters1; - - this.currentPageNumber = 1; - const params = this.addNewFiltersToParams(); - - this.loadExpenses$.next(params); - - this.filterPills = this.generateFilterPills(this.filters); - this.trackingService.myExpensesFilterApplied({ - ...this.filters, - }); - } - } - - clearFilters(): void { - this.filters = {}; - this.currentPageNumber = 1; - const params = this.addNewFiltersToParams(); - this.loadExpenses$.next(params); - this.filterPills = this.generateFilterPills(this.filters); - } - - async setState(): Promise { - this.isLoading = true; - this.currentPageNumber = 1; - const params = this.addNewFiltersToParams(); - this.loadExpenses$.next(params); - setTimeout(() => { - this.isLoading = false; - }, 500); - } - - setExpenseStatsOnSelect(): void { - this.allExpensesStats$ = of({ - count: this.selectedElements.length, - amount: this.selectedElements.reduce((acc, txnObj) => acc + txnObj.amount, 0), - }); - } - - setOutboxExpenseStatsOnSelect(): void { - this.allExpensesStats$ = of({ - count: this.selectedOutboxExpenses.length, - amount: this.selectedOutboxExpenses.reduce((acc, txnObj) => acc + txnObj.tx_amount, 0), - }); - } - - selectOutboxExpense(expense: Expense): void { - let isSelectedElementsIncludesExpense = false; - if (expense.tx_id) { - isSelectedElementsIncludesExpense = this.selectedOutboxExpenses.some((txn) => expense.tx_id === txn.tx_id); - } else { - isSelectedElementsIncludesExpense = this.selectedOutboxExpenses.some((txn) => isEqual(txn, expense)); - } - - if (isSelectedElementsIncludesExpense) { - if (expense.tx_id) { - this.selectedOutboxExpenses = this.selectedOutboxExpenses.filter((txn) => txn.tx_id !== expense.tx_id); - } else { - this.selectedOutboxExpenses = this.selectedOutboxExpenses.filter((txn) => !isEqual(txn, expense)); - } - } else { - this.selectedOutboxExpenses.push(expense); - } - this.isReportableExpensesSelected = - this.transactionService.getReportableExpenses(this.selectedOutboxExpenses).length > 0; - - if (this.selectedOutboxExpenses.length > 0) { - this.outboxExpensesToBeDeleted = this.transactionService.excludeCCCExpenses(this.selectedOutboxExpenses); - - this.cccExpenses = this.selectedOutboxExpenses.length - this.outboxExpensesToBeDeleted.length; - } - - // setting Expenses count and amount stats on select - if (this.allExpensesCount === this.selectedOutboxExpenses.length) { - this.selectAll = true; - } else { - this.selectAll = false; - } - this.setOutboxExpenseStatsOnSelect(); - this.isMergeAllowed = this.transactionService.isMergeAllowed(this.selectedOutboxExpenses); - } - - selectExpense(expense: PlatformExpense): void { - let isSelectedElementsIncludesExpense = false; - if (expense.id) { - isSelectedElementsIncludesExpense = this.selectedElements.some((txn) => expense.id === txn.id); - } else { - isSelectedElementsIncludesExpense = this.selectedElements.some((txn) => isEqual(txn, expense)); - } - - if (isSelectedElementsIncludesExpense) { - if (expense.id) { - this.selectedElements = this.selectedElements.filter((txn) => txn.id !== expense.id); - } else { - this.selectedElements = this.selectedElements.filter((txn) => !isEqual(txn, expense)); - } - } else { - this.selectedElements.push(expense); - } - this.isReportableExpensesSelected = - this.sharedExpenseService.getReportableExpenses(this.selectedElements, this.restrictPendingTransactionsEnabled) - .length > 0; - - if (this.selectedElements.length > 0) { - this.expensesToBeDeleted = this.sharedExpenseService.excludeCCCExpenses(this.selectedElements); - - this.cccExpenses = this.selectedElements.length - this.expensesToBeDeleted.length; - } - - // setting Expenses count and amount stats on select - if (this.allExpensesCount === this.selectedElements.length) { - this.selectAll = true; - } else { - this.selectAll = false; - } - this.setExpenseStatsOnSelect(); - this.isMergeAllowed = this.sharedExpenseService.isMergeAllowed(this.selectedElements); - } - - goToTransaction(event: { expense: PlatformExpense; expenseIndex: number }): void { - let category: string; - - if (event.expense?.category?.name) { - category = event.expense.category.name.toLowerCase(); - } - - if (category === 'mileage') { - this.router.navigate(['/', 'enterprise', 'add_edit_mileage', { id: event.expense.id, persist_filters: true }]); - } else if (category === 'per diem') { - this.router.navigate(['/', 'enterprise', 'add_edit_per_diem', { id: event.expense.id, persist_filters: true }]); - } else { - this.router.navigate(['/', 'enterprise', 'add_edit_expense', { id: event.expense.id, persist_filters: true }]); - } - } - - async openCriticalPolicyViolationPopOver(config: { - title: string; - message: string; - reportType: string; - }): Promise { - const criticalPolicyViolationPopOver = await this.popoverController.create({ - component: PopupAlertComponent, - componentProps: { - title: config.title, - message: config.message, - primaryCta: { - text: 'Exclude and Continue', - action: 'continue', - }, - secondaryCta: { - text: 'Cancel', - action: 'cancel', - }, - }, - cssClass: 'pop-up-in-center', - }); - - await criticalPolicyViolationPopOver.present(); - - const { data } = (await criticalPolicyViolationPopOver.onWillDismiss()) as { data: { action: string } }; - - if (data && data.action) { - if (data.action === 'continue') { - if (config.reportType === 'oldReport') { - this.showOldReportsMatBottomSheet(); - } else { - this.showNewReportModal(); - } - } - } - } - - showNonReportableExpenseSelectedToast(message: string): void { - this.matSnackBar.openFromComponent(ToastMessageComponent, { - ...this.snackbarProperties.setSnackbarProperties('failure', { message }), - panelClass: ['msb-failure-with-report-btn'], - }); - this.trackingService.showToastMessage({ ToastContent: message }); - } - - isSelectionContainsException( - policyViolationsCount: number, - draftCount: number, - pendingTransactionsCount: number - ): boolean { - return ( - policyViolationsCount > 0 || - draftCount > 0 || - (this.restrictPendingTransactionsEnabled && pendingTransactionsCount > 0) - ); - } - - unreportableExpenseExceptionHandler( - draftCount: number, - policyViolationsCount: number, - pendingTransactionsCount: number - ): void { - // This Map contains different messages based on different conditions , the first character in map key is draft, second is policy violation, third is pending transactions - // draft, policy, pending - const toastMessage = new Map([ - ['111', "You can't add draft expenses and expenses with critical policy violation & pending transactions."], - ['110', "You can't add draft expenses & expenses with critical policy violations to a report."], - ['101', "You can't add draft expenses & expenses with pending transactions to a report."], - ['011', "You can't add expenses with critical policy violation & pending transactions to a report."], - ['100', "You can't add draft expenses to a report."], - ['010', "You can't add expenses with critical policy violations to a report."], - ['001', "You can't add expenses with pending transactions to a report."], - ]); - const messageConfig = `${draftCount > 0 ? 1 : 0}${policyViolationsCount > 0 ? 1 : 0}${ - pendingTransactionsCount > 0 ? 1 : 0 - }`; - - if (toastMessage.has(messageConfig)) { - this.showNonReportableExpenseSelectedToast(toastMessage.get(messageConfig)); - } - if (pendingTransactionsCount > 0) { - this.trackingService.spenderSelectedPendingTxnFromMyExpenses(); - } - } - - reportableExpenseDialogHandler( - draftCount: number, - policyViolationsCount: number, - pendingTransactionsCount: number, - reportType: 'oldReport' | 'newReport' - ): void { - const title = "Can't add these expenses..."; - let message = ''; - - if (draftCount > 0) { - message += `${draftCount} ${draftCount > 1 ? 'expenses are' : 'expense is'} in draft state.`; - } - if (pendingTransactionsCount > 0) { - message += `${message.length ? '

' : ''}${pendingTransactionsCount} ${ - pendingTransactionsCount > 1 ? 'expenses' : 'expense' - } with pending transactions.`; - } - if (policyViolationsCount > 0) { - message += `${message.length ? '

' : ''}${policyViolationsCount} ${ - policyViolationsCount > 1 ? 'expenses' : 'expense' - } with Critical Policy Violations.`; - } - - this.openCriticalPolicyViolationPopOver({ title, message, reportType }); - } - - async openCreateReportWithSelectedIds(reportType: 'oldReport' | 'newReport'): Promise { - let selectedElements = cloneDeep(this.selectedElements); - // Removing offline expenses from the list - selectedElements = selectedElements.filter((expense) => expense.id); - if (!selectedElements.length) { - this.showNonReportableExpenseSelectedToast('Please select one or more expenses to be reported'); - return; - } - const expensesWithCriticalPolicyViolations = selectedElements.filter((expense) => - this.sharedExpenseService.isCriticalPolicyViolatedExpense(expense) - ); - const expensesInDraftState = selectedElements.filter((expense) => - this.sharedExpenseService.isExpenseInDraft(expense) - ); - let expensesWithPendingTransactions = []; - //only handle pending txns if it is enabled from settings - if (this.restrictPendingTransactionsEnabled) { - expensesWithPendingTransactions = selectedElements.filter((expense) => - this.sharedExpenseService.doesExpenseHavePendingCardTransaction(expense) - ); - } - - const noOfExpensesWithCriticalPolicyViolations = expensesWithCriticalPolicyViolations.length; - const noOfExpensesInDraftState = expensesInDraftState.length; - const noOfExpensesWithPendingTransactions = expensesWithPendingTransactions.length; - - if (!this.isReportableExpensesSelected) { - this.unreportableExpenseExceptionHandler( - noOfExpensesInDraftState, - noOfExpensesWithCriticalPolicyViolations, - noOfExpensesWithPendingTransactions - ); - } else { - this.trackingService.addToReport(); - const totalUnreportableCount = - noOfExpensesInDraftState + noOfExpensesWithCriticalPolicyViolations + noOfExpensesWithPendingTransactions; - - if (totalUnreportableCount > 0) { - this.reportableExpenseDialogHandler( - noOfExpensesInDraftState, - noOfExpensesWithCriticalPolicyViolations, - noOfExpensesWithPendingTransactions, - reportType - ); - } else { - if (reportType === 'oldReport') { - this.showOldReportsMatBottomSheet(); - } else { - this.showNewReportModal(); - } - } - } - } - - async showNewReportModal(): Promise { - const reportAbleExpenses = this.sharedExpenseService.getReportableExpenses( - this.selectedElements, - this.restrictPendingTransactionsEnabled - ); - const addExpenseToNewReportModal = await this.modalController.create({ - component: CreateNewReportComponent, - componentProps: { - selectedExpensesToReport: reportAbleExpenses, - }, - mode: 'ios', - ...this.modalProperties.getModalDefaultProperties(), - }); - await addExpenseToNewReportModal.present(); - - const { data } = (await addExpenseToNewReportModal.onDidDismiss()) as { - data: { report: ExtendedReport; message: string }; - }; - - if (data && data.report) { - this.showAddToReportSuccessToast({ report: data.report, message: data.message }); - } - } - - openCreateReport(): void { - this.trackingService.clickCreateReport(); - this.router.navigate(['/', 'enterprise', 'my_create_report']); - } - - openReviewExpenses(): void { - const allDataPipe$ = this.loadExpenses$.pipe( - take(1), - switchMap((params) => { - const queryParams = params.queryParams || {}; - - queryParams.report_id = queryParams.report_id || 'is.null'; - - queryParams.state = 'in.(COMPLETE,DRAFT)'; - - const orderByParams = - params.sortParam && params.sortDir - ? `${params.sortParam}.${params.sortDir}` - : 'spent_at.desc,created_at.desc,id.desc'; - - return this.expenseService - .getAllExpenses({ - queryParams, - order: orderByParams, - }) - .pipe( - map((expenses) => - expenses.filter((expense) => { - if (params.searchString) { - return this.filterExpensesBySearchString(expense, params.searchString); - } else { - return true; - } - }) - ) - ); - }), - map((etxns) => etxns.map((etxn) => etxn.id)) - ); - from(this.loaderService.showLoader()) - .pipe( - switchMap(() => { - const txnIds = this.selectedElements.map((expense) => expense.id); - return iif(() => this.selectedElements.length === 0, allDataPipe$, of(txnIds)); - }), - switchMap((selectedIds) => { - const initial = selectedIds[0]; - const allIds = selectedIds; - - return this.expenseService.getExpenseById(initial).pipe( - map((expense) => ({ - inital: expense, - allIds, - })) - ); - }), - finalize(() => from(this.loaderService.hideLoader())) - ) - .subscribe(({ inital, allIds }) => { - let category: string; - - if (inital.category.name) { - category = inital.category.name.toLowerCase(); - } - - if (category.includes('mileage')) { - this.router.navigate([ - '/', - 'enterprise', - 'add_edit_mileage', - { - id: inital.id, - txnIds: JSON.stringify(allIds), - activeIndex: 0, - }, - ]); - } else if (category.includes('per diem')) { - this.router.navigate([ - '/', - 'enterprise', - 'add_edit_per_diem', - { - id: inital.id, - txnIds: JSON.stringify(allIds), - activeIndex: 0, - }, - ]); - } else { - this.router.navigate([ - '/', - 'enterprise', - 'add_edit_expense', - { - id: inital.id, - txnIds: JSON.stringify(allIds), - activeIndex: 0, - }, - ]); - } - }); - } - - filterExpensesBySearchString(expense: PlatformExpense, searchString: string): boolean { - return Object.values(expense) - .map((value: keyof PlatformExpense) => value && value.toString().toLowerCase()) - .filter((value) => !!value) - .some((value) => value.toLowerCase().includes(searchString.toLowerCase())); - } - - async onAddTransactionToReport(event: { tx_id: string }): Promise { - const addExpenseToReportModal = await this.modalController.create({ - component: AddTxnToReportDialogComponent, - componentProps: { - txId: event.tx_id, - }, - mode: 'ios', - ...this.modalProperties.getModalDefaultProperties(), - }); - await addExpenseToReportModal.present(); - - const { data } = (await addExpenseToReportModal.onDidDismiss()) as { data: { reload: boolean } }; - if (data && data.reload) { - this.doRefresh(); - } - } - - showAddToReportSuccessToast(config: { message: string; report: ExtendedReport | ReportV1 }): void { - const toastMessageData = { - message: config.message, - redirectionText: 'View Report', - }; - const expensesAddedToReportSnackBar = this.matSnackBar.openFromComponent(ToastMessageComponent, { - ...this.snackbarProperties.setSnackbarProperties('success', toastMessageData), - panelClass: ['msb-success-with-camera-icon'], - }); - this.trackingService.showToastMessage({ ToastContent: config.message }); - - this.isReportableExpensesSelected = false; - this.selectionMode = false; - this.headerState = HeaderState.base; - this.doRefresh(); - - expensesAddedToReportSnackBar.onAction().subscribe(() => { - // Mixed data type as CREATE report and GET report API returns different responses - const reportId = (config.report as ExtendedReport).rp_id || (config.report as ReportV1).id; - this.router.navigate(['/', 'enterprise', 'my_view_report', { id: reportId, navigateBack: true }]); - }); - } - - addTransactionsToReport(report: ExtendedReport, selectedExpensesId: string[]): Observable { - return from(this.loaderService.showLoader('Adding transaction to report')).pipe( - switchMap(() => this.reportService.addTransactions(report.rp_id, selectedExpensesId).pipe(map(() => report))), - finalize(() => this.loaderService.hideLoader()) - ); - } - - showOldReportsMatBottomSheet(): void { - const reportAbleExpenses = this.sharedExpenseService.getReportableExpenses( - this.selectedElements, - this.restrictPendingTransactionsEnabled - ); - const selectedExpensesId = reportAbleExpenses.map((expenses) => expenses.id); - - this.openReports$ - .pipe( - switchMap((openReports) => { - const addTxnToReportDialog = this.matBottomSheet.open(AddTxnToReportDialogComponent, { - data: { openReports, isNewReportsFlowEnabled: this.isNewReportsFlowEnabled }, - panelClass: ['mat-bottom-sheet-1'], - }); - return addTxnToReportDialog.afterDismissed() as Observable<{ report: ExtendedReport }>; - }), - switchMap((data) => { - if (data && data.report) { - return this.addTransactionsToReport(data.report, selectedExpensesId); - } else { - return of(null); - } - }) - ) - .subscribe((report: ExtendedReport) => { - if (report) { - let message = ''; - if (report.rp_state.toLowerCase() === 'draft') { - message = 'Expenses added to an existing draft report'; - } else { - message = 'Expenses added to report successfully'; - } - this.showAddToReportSuccessToast({ message, report }); - } - }); - } - - async openActionSheet(): Promise { - const that = this; - const actionSheet = await this.actionSheetController.create({ - header: 'ADD EXPENSE', - mode: 'md', - cssClass: 'fy-action-sheet', - buttons: that.actionSheetButtons, - }); - await actionSheet.present(); - } - - deleteSelectedExpenses(offlineExpenses: Partial[]): Observable { - if (offlineExpenses?.length > 0) { - this.transactionOutboxService.deleteBulkOfflineExpenses(this.pendingTransactions, offlineExpenses); - return of(null); - } else { - this.selectedElements = this.expensesToBeDeleted.filter((expense) => expense.id); - if (this.selectedElements.length > 0) { - return this.transactionService.deleteBulk(this.selectedElements.map((selectedExpense) => selectedExpense.id)); - } else { - return of(null); - } - } - } - - async openDeleteExpensesPopover(): Promise { - const offlineExpenses = this.outboxExpensesToBeDeleted.filter((expense) => !expense.tx_id); - - let expenseDeletionMessage: string; - let cccExpensesMessage: string; - let totalDeleteLength = 0; - - if (offlineExpenses.length > 0) { - expenseDeletionMessage = this.transactionService.getExpenseDeletionMessage(offlineExpenses); - cccExpensesMessage = this.transactionService.getCCCExpenseMessage(offlineExpenses, this.cccExpenses); - totalDeleteLength = this.outboxExpensesToBeDeleted.length; - } else { - expenseDeletionMessage = this.sharedExpenseService.getExpenseDeletionMessage(this.expensesToBeDeleted); - cccExpensesMessage = this.sharedExpenseService.getCCCExpenseMessage(this.expensesToBeDeleted, this.cccExpenses); - totalDeleteLength = this.expensesToBeDeleted?.length; - } - - const deletePopover = await this.popoverController.create({ - component: FyDeleteDialogComponent, - cssClass: 'delete-dialog', - backdropDismiss: false, - componentProps: { - header: 'Delete Expense', - body: this.sharedExpenseService.getDeleteDialogBody( - totalDeleteLength, - this.cccExpenses, - expenseDeletionMessage, - cccExpensesMessage - ), - ctaText: totalDeleteLength > 0 && this.cccExpenses > 0 ? 'Exclude and Delete' : 'Delete', - disableDelete: totalDeleteLength === 0 ? true : false, - deleteMethod: () => this.deleteSelectedExpenses(offlineExpenses), - }, - }); - - await deletePopover.present(); - - const { data } = (await deletePopover.onDidDismiss()) as { data: { status: string } }; - - if (data) { - this.trackingService.myExpensesBulkDeleteExpenses({ - count: this.selectedElements.length, - }); - - if (data.status === 'success') { - let totalNoOfSelectedExpenses = 0; - if (offlineExpenses?.length > 0) { - totalNoOfSelectedExpenses = offlineExpenses.length + this.selectedElements.length; - } else { - totalNoOfSelectedExpenses = this.selectedElements.length; - } - - const message = - totalNoOfSelectedExpenses === 1 - ? '1 expense has been deleted' - : `${totalNoOfSelectedExpenses} expenses have been deleted`; - this.matSnackBar.openFromComponent(ToastMessageComponent, { - ...this.snackbarProperties.setSnackbarProperties('success', { message }), - panelClass: ['msb-success-with-camera-icon'], - }); - this.trackingService.showToastMessage({ ToastContent: message }); - } else { - const message = 'We could not delete the expenses. Please try again'; - this.matSnackBar.openFromComponent(ToastMessageComponent, { - ...this.snackbarProperties.setSnackbarProperties('failure', { message }), - panelClass: ['msb-failure-with-camera-icon'], - }); - this.trackingService.showToastMessage({ ToastContent: message }); - } - - this.isReportableExpensesSelected = false; - this.selectionMode = false; - this.headerState = HeaderState.base; - - this.doRefresh(); - } - } - - onSelectAll(checked: boolean): void { - if (checked) { - this.selectedElements = []; - if (this.pendingTransactions.length > 0) { - this.selectedOutboxExpenses = this.pendingTransactions; - this.allExpensesCount = this.pendingTransactions.length; - this.isReportableExpensesSelected = - this.transactionService.getReportableExpenses(this.selectedOutboxExpenses).length > 0; - this.outboxExpensesToBeDeleted = this.selectedOutboxExpenses; - this.setOutboxExpenseStatsOnSelect(); - } else { - this.loadExpenses$ - .pipe( - take(1), - map((params) => { - const queryParams = params.queryParams || {}; - - queryParams.report_id = queryParams.report_id || 'is.null'; - queryParams.state = 'in.(COMPLETE,DRAFT)'; - if (params.searchString) { - queryParams.q = params?.searchString + ':*'; - } - - return queryParams; - }), - switchMap((queryParams) => this.expenseService.getAllExpenses({ queryParams })) - ) - .subscribe((allExpenses) => { - this.selectedElements = this.selectedElements.concat(allExpenses); - if (this.selectedElements.length > 0) { - this.expensesToBeDeleted = this.sharedExpenseService.excludeCCCExpenses(this.selectedElements); - - this.cccExpenses = this.selectedElements.length - this.expensesToBeDeleted.length; - } - this.allExpensesCount = this.selectedElements.length; - this.isReportableExpensesSelected = - this.sharedExpenseService.getReportableExpenses(this.selectedElements).length > 0; - this.setExpenseStatsOnSelect(); - }); - } - } else { - this.selectedElements = []; - this.selectedOutboxExpenses = []; - this.outboxExpensesToBeDeleted = []; - this.isReportableExpensesSelected = - this.sharedExpenseService.getReportableExpenses(this.selectedElements).length > 0; - this.setExpenseStatsOnSelect(); - } - } - - onSimpleSearchCancel(): void { - this.headerState = HeaderState.base; - this.clearText('onSimpleSearchCancel'); - } - - onFilterPillsClearAll(): void { - this.clearFilters(); - } - - async onFilterClick(filterType: string): Promise { - if (filterType === 'state') { - await this.openFilters('Type'); - } else if (filterType === 'receiptsAttached') { - await this.openFilters('Receipts Attached'); - } else if (filterType === 'type') { - await this.openFilters('Expense Type'); - } else if (filterType === 'date') { - await this.openFilters('Date'); - } else if (filterType === 'sort') { - await this.openFilters('Sort By'); - } else if (filterType === 'splitExpense') { - await this.openFilters('Split Expense'); - } - } - - onFilterClose(filterType: string): void { - if (filterType === 'sort') { - delete this.filters.sortDir; - delete this.filters.sortParam; - } else { - delete this.filters[filterType]; - } - this.currentPageNumber = 1; - const params = this.addNewFiltersToParams(); - this.loadExpenses$.next(params); - this.filterPills = this.generateFilterPills(this.filters); - } - - onHomeClicked(): void { - const queryParams: Params = { state: 'home' }; - this.router.navigate(['/', 'enterprise', 'my_dashboard'], { - queryParams, - }); - - this.trackingService.footerHomeTabClicked({ - page: 'Expenses', - }); - } - - onTaskClicked(): void { - const queryParams: Params = { state: 'tasks', tasksFilters: 'expenses' }; - this.router.navigate(['/', 'enterprise', 'my_dashboard'], { - queryParams, - }); - this.trackingService.tasksPageOpened({ - Asset: 'Mobile', - from: 'My Expenses', - }); - } - - onCameraClicked(): void { - this.router.navigate([ - '/', - 'enterprise', - 'camera_overlay', - { - navigate_back: true, - }, - ]); - } - - searchClick(): void { - this.headerState = HeaderState.simpleSearch; - const searchInput = this.simpleSearchInput.nativeElement; - setTimeout(() => { - searchInput.focus(); - }, 300); - } - - mergeExpenses(): void { - const expenseIDs = this.selectedElements.map((expense) => expense.id); - this.router.navigate([ - '/', - 'enterprise', - 'merge_expense', - { - expenseIDs: JSON.stringify(expenseIDs), - from: 'MY_EXPENSES', - }, - ]); - } - - showCamera(isCameraPreviewStarted: boolean): void { - this.isCameraPreviewStarted = isCameraPreviewStarted; - } - - checkDeleteDisabled(): Observable { - return this.isConnected$.pipe( - map((isConnected) => { - if (isConnected) { - this.isDisabled = - this.selectedElements?.length === 0 || - !this.expensesToBeDeleted || - (this.expensesToBeDeleted?.length === 0 && this.cccExpenses > 0); - } else if (!isConnected) { - this.isDisabled = this.selectedOutboxExpenses.length === 0 || !this.outboxExpensesToBeDeleted; - } - }) - ); - } -} diff --git a/src/app/fyle/my-expenses-v2/my-expenses.service.spec.ts b/src/app/fyle/my-expenses-v2/my-expenses.service.spec.ts deleted file mode 100644 index 522b6e28de..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses.service.spec.ts +++ /dev/null @@ -1,521 +0,0 @@ -import { TestBed } from '@angular/core/testing'; -import { MyExpensesService } from './my-expenses.service'; -import { - expenseFiltersData1, - expenseFiltersData3, - expenseFiltersData4, - expenseFiltersData5, - expenseFiltersData6, -} from 'src/app/core/mock-data/expense-filters.data'; -import { - cardFilterPill, - creditTxnFilterPill, - expectedDateFilterPill, - expectedFilterPill2, - receiptsAttachedFilterPill, - sortByAscFilterPill, - sortByDateAscFilterPill, - sortByDateDescFilterPill, - sortByDescFilterPill, - sortFilterPill, - splitExpenseFilterPill, - stateFilterPill2, -} from 'src/app/core/mock-data/filter-pills.data'; -import { selectedFilters7, selectedFilters8, selectedFilters9 } from 'src/app/core/mock-data/selected-filters.data'; -import { FilterPill } from 'src/app/shared/components/fy-filter-pills/filter-pill.interface'; -import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; -import { - expectedFilterPill3, - expectedFilterPill4, - expectedFilterPill5, - expectedFilterPill6, - expectedFilterPill7, - expectedFilterPill8, - expectedFilterPill9, -} from 'src/app/core/mock-data/my-reports-filterpills.data'; -import { filter1, filter2 } from 'src/app/core/mock-data/my-reports-filters.data'; -import { filterOptions2 } from 'src/app/core/mock-data/filter-options.data'; -import { ExpenseType } from 'src/app/core/enums/expense-type.enum'; - -describe('MyExpensesService', () => { - let myExpensesService: MyExpensesService; - - beforeEach(() => { - TestBed.configureTestingModule({ - providers: [MyExpensesService], - }); - - myExpensesService = TestBed.inject(MyExpensesService); - }); - - it('should be created', () => { - expect(myExpensesService).toBeTruthy(); - }); - - it('generateSortFilterPills(): should call generateSortTxnDatePills, generateSortAmountPills and generateSortCategoryPills once', () => { - spyOn(myExpensesService, 'generateSortTxnDatePills'); - spyOn(myExpensesService, 'generateSortAmountPills'); - //@ts-ignore - spyOn(myExpensesService, 'generateSortCategoryPills'); - - myExpensesService.generateSortFilterPills(expenseFiltersData1, creditTxnFilterPill); - - expect(myExpensesService.generateSortTxnDatePills).toHaveBeenCalledTimes(1); - expect(myExpensesService.generateSortAmountPills).toHaveBeenCalledTimes(1); - //@ts-ignore - expect(myExpensesService.generateSortCategoryPills).toHaveBeenCalledTimes(1); - }); - - describe('covertFilters():', () => { - it('should modify the selected filters and return the generated filter', () => { - spyOn(myExpensesService, 'convertSelectedSortFitlersToFilters'); - const sortBy = { name: 'Sort By', value: 'dateNewToOld' }; - - const convertedFilters = myExpensesService.convertSelectedOptionsToExpenseFilters(selectedFilters7); - - expect(myExpensesService.convertSelectedSortFitlersToFilters).toHaveBeenCalledOnceWith( - sortBy, - expenseFiltersData3 - ); - - expect(convertedFilters).toEqual(expenseFiltersData3); - }); - - it('should set customDateStart and customDateEnd as undefined if associated data is undefined', () => { - spyOn(myExpensesService, 'convertSelectedSortFitlersToFilters'); - const sortBy = { name: 'Sort By', value: 'dateNewToOld' }; - - const convertedFilters = myExpensesService.convertSelectedOptionsToExpenseFilters(selectedFilters8); - - expect(myExpensesService.convertSelectedSortFitlersToFilters).toHaveBeenCalledOnceWith( - sortBy, - expenseFiltersData4 - ); - - expect(convertedFilters).toEqual(expenseFiltersData4); - }); - }); - - describe('generateSortAmountPills():', () => { - it('should add amount - high to low as sort params if sort direction is decreasing', () => { - const filterPill = []; - myExpensesService.generateSortAmountPills(expenseFiltersData5, filterPill); - expect(filterPill).toEqual(sortByDescFilterPill); - }); - - it('should add amount - low to high as sort params if sort direction is ascending', () => { - const filterPill = []; - myExpensesService.generateSortAmountPills({ ...expenseFiltersData5, sortDir: 'asc' }, filterPill); - expect(filterPill).toEqual(sortByAscFilterPill); - }); - }); - - describe('generateSortTxnDatePills():', () => { - it('should add date - old to new as sort params if sort direction is ascending', () => { - const filterPill = []; - myExpensesService.generateSortTxnDatePills(expenseFiltersData6, filterPill); - expect(filterPill).toEqual(sortByDateAscFilterPill); - }); - - it('should add date - new to old as sort params if sort direction is descending', () => { - const filterPill = []; - myExpensesService.generateSortTxnDatePills({ ...expenseFiltersData6, sortDir: 'desc' }, filterPill); - expect(filterPill).toEqual(sortByDateDescFilterPill); - }); - }); - - it('generateTypeFilterPills(): should add combined expense types value in filter pills', () => { - const filterPill = []; - myExpensesService.generateTypeFilterPills( - { ...expenseFiltersData1, type: [ExpenseType.EXPENSE, ExpenseType.PER_DIEM, ExpenseType.MILEAGE, 'custom'] }, - filterPill - ); - expect(filterPill).toEqual([ - { label: 'Expense Type', type: 'type', value: 'Regular Expenses, Per Diem, Mileage, custom' }, - ]); - }); - - describe('generateDateFilterPills(): ', () => { - it('should generate filter pill for "this Week"', () => { - const filter = { - date: DateFilters.thisWeek, - }; - - const res = myExpensesService.generateDateFilterPills(filter, []); - - expect(res).toEqual(expectedFilterPill5); - }); - - it('should generate filter pill for "this Month"', () => { - const filter = { - date: DateFilters.thisMonth, - }; - - const res = myExpensesService.generateDateFilterPills(filter, []); - - expect(res).toEqual(expectedFilterPill6); - }); - - it('should generate filter pill for "All"', () => { - const filter = { - date: DateFilters.all, - }; - - const res = myExpensesService.generateDateFilterPills(filter, []); - - expect(res).toEqual(expectedFilterPill7); - }); - - it('should generate filter pill for "Last Month"', () => { - const filter = { - date: DateFilters.lastMonth, - }; - - const res = myExpensesService.generateDateFilterPills(filter, []); - - expect(res).toEqual(expectedFilterPill8); - }); - - it('should generate custom date filter pill', () => { - const filter = filter2; - spyOn(myExpensesService, 'generateCustomDatePill').and.returnValue(expectedFilterPill9); - - const res = myExpensesService.generateDateFilterPills(filter, []); - - expect(myExpensesService.generateCustomDatePill).toHaveBeenCalledOnceWith(filter, []); - - expect(res).toEqual(expectedFilterPill9); - }); - }); - - describe('generateCustomDatePill(): ', () => { - it('should generate custom date filter pill with start and end date', () => { - const filter = { - customDateStart: new Date('2023-01-21'), - customDateEnd: new Date('2023-01-31'), - }; - - const res = myExpensesService.generateCustomDatePill(filter, []); - - expect(res).toEqual(expectedDateFilterPill); - }); - - it('should generate custom date filter pill with only start date', () => { - const filter = { - customDateStart: new Date('2023-01-21'), - customDateEnd: null, - }; - - const res = myExpensesService.generateCustomDatePill(filter, []); - - expect(res).toEqual(expectedFilterPill3); - }); - - it('should generate custom date filter pill with only end date', () => { - const filter = { - customDateStart: null, - customDateEnd: new Date('2023-01-31'), - }; - - const res = myExpensesService.generateCustomDatePill(filter, []); - - expect(res).toEqual(expectedFilterPill4); - }); - - it('should not generate custom date filter pill if start and end date are null', () => { - const filter = { - customDateStart: null, - customDateEnd: null, - }; - - const res = myExpensesService.generateCustomDatePill(filter, []); - - expect(res).toEqual([]); - }); - }); - - it('generateReceiptsAttachedFilterPills(): should add receipt attached filter pill', () => { - const filterPill = []; - myExpensesService.generateReceiptsAttachedFilterPills(filterPill, expenseFiltersData1); - expect(filterPill).toEqual([receiptsAttachedFilterPill]); - }); - - it('generateSplitExpenseFilterPills(): should add split expense filter pill', () => { - const filterPill = []; - myExpensesService.generateSplitExpenseFilterPills(filterPill, expenseFiltersData1); - expect(filterPill).toEqual([splitExpenseFilterPill]); - }); - - it('generateCardFilterPills(): should add card filter pill', () => { - const filterPill = []; - myExpensesService.generateCardFilterPills(filterPill, expenseFiltersData1); - expect(filterPill).toEqual([cardFilterPill]); - }); - - it('generateStateFilterPills(): should add state filter pill', () => { - const state = ['DRAFT', 'READY_TO_REPORT', 'APPROVED']; - const filterPill = []; - myExpensesService.generateStateFilterPills(filterPill, { ...expenseFiltersData1, state }); - expect(filterPill).toEqual([stateFilterPill2]); - }); - - describe('convertSelectedSortFiltersToFilters(): ', () => { - it('should convert selected sort filter to corresponding sortParam and sortDir', () => { - const sortBy = { - name: 'Sort By', - value: 'dateNewToOld', - }; - const generatedFilters = {}; - - myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - expect(generatedFilters).toEqual({ - sortParam: 'spent_at', - sortDir: 'desc', - }); - }); - - it('should convert selected sort filter to corresponding sortParam and sortDir (dateOldToNew)', () => { - const sortBy = { - name: 'Sort By', - value: 'dateOldToNew', - }; - const generatedFilters = {}; - - myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - expect(generatedFilters).toEqual({ - sortParam: 'spent_at', - sortDir: 'asc', - }); - }); - - it('should convert selected sort filter to corresponding sortParam and sortDir (amountHighToLow)', () => { - const sortBy = { - name: 'Sort By', - value: 'amountHighToLow', - }; - const generatedFilters = {}; - - myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - expect(generatedFilters).toEqual({ - sortParam: 'amount', - sortDir: 'desc', - }); - }); - - it('should convert selected sort filter to corresponding sortParam and sortDir (amountLowToHigh)', () => { - const sortBy = { - name: 'Sort By', - value: 'amountLowToHigh', - }; - const generatedFilters = {}; - - myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - expect(generatedFilters).toEqual({ - sortParam: 'amount', - sortDir: 'asc', - }); - }); - - it('should convert selected sort filter to corresponding sortParam and sortDir (nameAToZ)', () => { - const sortBy = { - name: 'Sort By', - value: 'categoryAToZ', - }; - const generatedFilters = {}; - - myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - expect(generatedFilters).toEqual({ - sortParam: 'category->name', - sortDir: 'asc', - }); - }); - - it('should convert selected sort filter to corresponding sortParam and sortDir (nameZToA)', () => { - const sortBy = { - name: 'Sort By', - value: 'categoryZToA', - }; - const generatedFilters = {}; - - myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - expect(generatedFilters).toEqual({ - sortParam: 'category->name', - sortDir: 'desc', - }); - }); - }); - - it('getFilters(): should return all the filters', () => { - const filters = myExpensesService.getFilters(); - - expect(filters).toEqual(filterOptions2); - }); - - it('generateSelectedFilters(): should generate selected filters', () => { - spyOn(myExpensesService, 'addSortToGeneratedFilters'); - - const filters = myExpensesService.generateSelectedFilters(expenseFiltersData1); - - expect(myExpensesService.addSortToGeneratedFilters).toHaveBeenCalledOnceWith(expenseFiltersData1, filters); - - expect(filters).toEqual(selectedFilters9); - }); - - it('addSortToGeneratedFilters(): should call convertTxnDtSortToSelectedFilters, convertAmountSortToSelectedFilters and convertCategorySortToSelectedFilters once', () => { - spyOn(myExpensesService, 'convertTxnDtSortToSelectedFilters'); - spyOn(myExpensesService, 'convertAmountSortToSelectedFilters'); - spyOn(myExpensesService, 'convertCategorySortToSelectedFilters'); - - myExpensesService.addSortToGeneratedFilters(expenseFiltersData1, selectedFilters9); - - expect(myExpensesService.convertTxnDtSortToSelectedFilters).toHaveBeenCalledOnceWith( - expenseFiltersData1, - selectedFilters9 - ); - expect(myExpensesService.convertAmountSortToSelectedFilters).toHaveBeenCalledOnceWith( - expenseFiltersData1, - selectedFilters9 - ); - expect(myExpensesService.convertCategorySortToSelectedFilters).toHaveBeenCalledOnceWith( - expenseFiltersData1, - selectedFilters9 - ); - }); - - describe('convertCategorySortToSelectedFilters():', () => { - it('should add categoryAToZ sort params if sort direction is ascending', () => { - const generatedFilters = []; - - myExpensesService.convertCategorySortToSelectedFilters(expenseFiltersData1, generatedFilters); - - expect(generatedFilters).toEqual([ - { - name: 'Sort By', - value: 'categoryAToZ', - }, - ]); - }); - - it('should add categoryZToA sort params if sort direction is descending', () => { - const generatedFilters = []; - - myExpensesService.convertCategorySortToSelectedFilters( - { ...expenseFiltersData1, sortDir: 'desc' }, - generatedFilters - ); - - expect(generatedFilters).toEqual([ - { - name: 'Sort By', - value: 'categoryZToA', - }, - ]); - }); - }); - - describe('convertAmountSortToSelectedFilters(): ', () => { - it('should convert amount sort to selected filters for descending sort', () => { - const filter = { - sortParam: 'amount', - sortDir: 'desc', - }; - const generatedFilters = []; - - myExpensesService.convertAmountSortToSelectedFilters(filter, generatedFilters); - - expect(generatedFilters).toEqual([ - { - name: 'Sort By', - value: 'amountHighToLow', - }, - ]); - }); - - it('should convert amount sort to selected filters for ascending sort', () => { - const filter = { - sortParam: 'amount', - sortDir: 'asc', - }; - const generatedFilters = []; - - myExpensesService.convertAmountSortToSelectedFilters(filter, generatedFilters); - - expect(generatedFilters).toEqual([ - { - name: 'Sort By', - value: 'amountLowToHigh', - }, - ]); - }); - }); - - describe('convertTxnDtSortToSelectedFilters():', () => { - it('should covert txn date sort to selected filters for descending sort', () => { - const filter = { - sortParam: 'spent_at', - sortDir: 'desc', - }; - const generatedFilters = []; - - myExpensesService.convertTxnDtSortToSelectedFilters(filter, generatedFilters); - - expect(generatedFilters).toEqual([ - { - name: 'Sort By', - value: 'dateNewToOld', - }, - ]); - }); - - it('should covert txn date sort to selected filters for ascending sort', () => { - const filter = { - sortParam: 'spent_at', - sortDir: 'asc', - }; - const generatedFilters = []; - - myExpensesService.convertTxnDtSortToSelectedFilters(filter, generatedFilters); - - expect(generatedFilters).toEqual([ - { - name: 'Sort By', - value: 'dateOldToNew', - }, - ]); - }); - }); - - describe('generateSortCategoryPills():', () => { - it('should add category - a to z as sort params if sort direction is ascending', () => { - const filter = { - sortParam: 'category->name', - sortDir: 'asc', - }; - const filterPill = []; - - //@ts-ignore - myExpensesService.generateSortCategoryPills(filter, filterPill); - - expect(filterPill).toEqual([sortFilterPill]); - }); - - it('should add category - z to a as sort params if sort direction is descending', () => { - const filter = { - sortParam: 'category->name', - sortDir: 'desc', - }; - const filterPill = []; - - //@ts-ignore - myExpensesService.generateSortCategoryPills(filter, filterPill); - - expect(filterPill).toEqual([{ ...sortFilterPill, value: 'category - z to a' }]); - }); - }); -}); diff --git a/src/app/fyle/my-expenses-v2/my-expenses.service.ts b/src/app/fyle/my-expenses-v2/my-expenses.service.ts deleted file mode 100644 index c9a07c6be7..0000000000 --- a/src/app/fyle/my-expenses-v2/my-expenses.service.ts +++ /dev/null @@ -1,530 +0,0 @@ -import { Injectable } from '@angular/core'; -import * as dayjs from 'dayjs'; -import { FilterPill } from 'src/app/shared/components/fy-filter-pills/filter-pill.interface'; -import { DateFilters } from 'src/app/shared/components/fy-filters/date-filters.enum'; -import { FilterOptionType } from 'src/app/shared/components/fy-filters/filter-option-type.enum'; -import { FilterOptions } from 'src/app/shared/components/fy-filters/filter-options.interface'; -import { SelectedFilters } from 'src/app/shared/components/fy-filters/selected-filters.interface'; -import { MaskNumber } from 'src/app/shared/pipes/mask-number.pipe'; -import { ExpenseType } from 'src/app/core/enums/expense-type.enum'; -import { ExpenseFilters } from 'src/app/core/models/platform/expense-filters.model'; - -@Injectable({ - providedIn: 'root', -}) -export class MyExpensesService { - maskNumber = new MaskNumber(); - - generateSortFilterPills(filter: Partial, filterPills: FilterPill[]): void { - this.generateSortTxnDatePills(filter, filterPills); - - this.generateSortAmountPills(filter, filterPills); - - this.generateSortCategoryPills(filter, filterPills); - } - - convertSelectedOptionsToExpenseFilters( - selectedFilters: SelectedFilters[] - ): Partial { - const generatedFilters: Partial = {}; - - const typeFilter = selectedFilters.find((filter) => filter.name === 'Type'); - if (typeFilter) { - generatedFilters.state = typeFilter.value; - } - - const dateFilter = selectedFilters.find((filter) => filter.name === 'Date'); - if (dateFilter) { - generatedFilters.date = dateFilter.value; - generatedFilters.customDateStart = dateFilter.associatedData?.startDate; - generatedFilters.customDateEnd = dateFilter.associatedData?.endDate; - } - - const receiptAttachedFilter = selectedFilters.find((filter) => filter.name === 'Receipts Attached'); - - if (receiptAttachedFilter) { - generatedFilters.receiptsAttached = receiptAttachedFilter.value; - } - - const expenseTypeFilter = selectedFilters.find((filter) => filter.name === 'Expense Type'); - - if (expenseTypeFilter) { - generatedFilters.type = expenseTypeFilter.value; - } - - const cardsFilter = selectedFilters.find((filter) => filter.name === 'Cards'); - - if (cardsFilter) { - generatedFilters.cardNumbers = cardsFilter.value; - } - - const sortBy = selectedFilters.find((filter) => filter.name === 'Sort By'); - - this.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); - - const splitExpenseFilter = selectedFilters.find((filter) => filter.name === 'Split Expense'); - - if (splitExpenseFilter) { - generatedFilters.splitExpense = splitExpenseFilter.value; - } - - return generatedFilters; - } - - generateSortAmountPills(filter: Partial, filterPills: FilterPill[]): void { - if (filter.sortParam === 'amount' && filter.sortDir === 'desc') { - filterPills.push({ - label: 'Sort By', - type: 'sort', - value: 'amount - high to low', - }); - } else if (filter.sortParam === 'amount' && filter.sortDir === 'asc') { - filterPills.push({ - label: 'Sort By', - type: 'sort', - value: 'amount - low to high', - }); - } - } - - generateSortTxnDatePills(filter: Partial, filterPills: FilterPill[]): void { - if (filter.sortParam === 'spent_at' && filter.sortDir === 'asc') { - filterPills.push({ - label: 'Sort By', - type: 'sort', - value: 'date - old to new', - }); - } else if (filter.sortParam === 'spent_at' && filter.sortDir === 'desc') { - filterPills.push({ - label: 'Sort By', - type: 'sort', - value: 'date - new to old', - }); - } - } - - generateTypeFilterPills(filter: Partial, filterPills: FilterPill[]): void { - const combinedValue = filter.type - .map((type) => { - if (type === 'EXPENSE') { - return 'Regular Expenses'; - } else if (type === 'PER_DIEM') { - return 'Per Diem'; - } else if (type === 'MILEAGE') { - return 'Mileage'; - } else { - return type; - } - }) - .reduce((type1, type2) => `${type1}, ${type2}`); - - filterPills.push({ - label: 'Expense Type', - type: 'type', - value: combinedValue, - }); - } - - generateDateFilterPills(filter: Partial, filterPills: FilterPill[]): FilterPill[] { - if (filter.date === DateFilters.thisWeek) { - filterPills.push({ - label: 'Date', - type: 'date', - value: 'this Week', - }); - } - - if (filter.date === DateFilters.thisMonth) { - filterPills.push({ - label: 'Date', - type: 'date', - value: 'this Month', - }); - } - - if (filter.date === DateFilters.all) { - filterPills.push({ - label: 'Date', - type: 'date', - value: 'All', - }); - } - - if (filter.date === DateFilters.lastMonth) { - filterPills.push({ - label: 'Date', - type: 'date', - value: 'Last Month', - }); - } - - if (filter.date === DateFilters.custom) { - filterPills = this.generateCustomDatePill(filter, filterPills); - } - - return filterPills; - } - - generateCustomDatePill(filter: Partial, filterPills: FilterPill[]): FilterPill[] { - const startDate = filter.customDateStart && dayjs(filter.customDateStart).format('YYYY-MM-D'); - const endDate = filter.customDateEnd && dayjs(filter.customDateEnd).format('YYYY-MM-D'); - - if (startDate && endDate) { - filterPills.push({ - label: 'Date', - type: 'date', - value: `${startDate} to ${endDate}`, - }); - } else if (startDate) { - filterPills.push({ - label: 'Date', - type: 'date', - value: `>= ${startDate}`, - }); - } else if (endDate) { - filterPills.push({ - label: 'Date', - type: 'date', - value: `<= ${endDate}`, - }); - } - - return filterPills; - } - - generateReceiptsAttachedFilterPills(filterPills: FilterPill[], filter: Partial): void { - filterPills.push({ - label: 'Receipts Attached', - type: 'receiptsAttached', - value: filter.receiptsAttached.toLowerCase(), - }); - } - - generateSplitExpenseFilterPills(filterPills: FilterPill[], filter: Partial): void { - filterPills.push({ - label: 'Split Expense', - type: 'splitExpense', - value: filter.splitExpense.toLowerCase(), - }); - } - - generateCardFilterPills(filterPills: FilterPill[], filter: Partial): void { - filterPills.push({ - label: 'Cards', - type: 'cardNumbers', - value: filter.cardNumbers - .map((cardNumber) => this.maskNumber.transform(cardNumber)) - .reduce((state1, state2) => `${state1}, ${state2}`), - }); - } - - generateStateFilterPills(filterPills: FilterPill[], filter: Partial): void { - const filterState = filter.state as string[]; - - filterPills.push({ - label: 'Type', - type: 'state', - value: filterState - .map((state) => { - if (state === 'DRAFT') { - return 'Incomplete'; - } else if (state === 'READY_TO_REPORT') { - return 'Complete'; - } else { - return state.replace(/_/g, ' ').toLowerCase(); - } - }) - .reduce((state1, state2) => `${state1}, ${state2}`), - }); - } - - convertSelectedSortFitlersToFilters( - sortBy: SelectedFilters, - generatedFilters: Partial - ): void { - if (sortBy) { - if (sortBy.value === 'dateNewToOld') { - generatedFilters.sortParam = 'spent_at'; - generatedFilters.sortDir = 'desc'; - } else if (sortBy.value === 'dateOldToNew') { - generatedFilters.sortParam = 'spent_at'; - generatedFilters.sortDir = 'asc'; - } else if (sortBy.value === 'amountHighToLow') { - generatedFilters.sortParam = 'amount'; - generatedFilters.sortDir = 'desc'; - } else if (sortBy.value === 'amountLowToHigh') { - generatedFilters.sortParam = 'amount'; - generatedFilters.sortDir = 'asc'; - } else if (sortBy.value === 'categoryAToZ') { - generatedFilters.sortParam = 'category->name'; - generatedFilters.sortDir = 'asc'; - } else if (sortBy.value === 'categoryZToA') { - generatedFilters.sortParam = 'category->name'; - generatedFilters.sortDir = 'desc'; - } - } - } - - getFilters(): FilterOptions[] { - return [ - { - name: 'Type', - optionType: FilterOptionType.multiselect, - options: [ - { - label: 'Complete', - value: 'READY_TO_REPORT', - }, - { - label: 'Policy Violated', - value: 'POLICY_VIOLATED', - }, - { - label: 'Cannot Report', - value: 'CANNOT_REPORT', - }, - { - label: 'Incomplete', - value: 'DRAFT', - }, - ], - } as FilterOptions, - { - name: 'Date', - optionType: FilterOptionType.date, - options: [ - { - label: 'All', - value: DateFilters.all, - }, - { - label: 'This Week', - value: DateFilters.thisWeek, - }, - { - label: 'This Month', - value: DateFilters.thisMonth, - }, - { - label: 'Last Month', - value: DateFilters.lastMonth, - }, - { - label: 'Custom', - value: DateFilters.custom, - }, - ], - } as FilterOptions, - { - name: 'Receipts Attached', - optionType: FilterOptionType.singleselect, - options: [ - { - label: 'Yes', - value: 'YES', - }, - { - label: 'No', - value: 'NO', - }, - ], - } as FilterOptions, - { - name: 'Expense Type', - optionType: FilterOptionType.multiselect, - options: [ - { - label: 'Mileage', - value: ExpenseType.MILEAGE, - }, - { - label: 'Per Diem', - value: ExpenseType.PER_DIEM, - }, - { - label: 'Regular Expenses', - value: ExpenseType.EXPENSE, - }, - ], - } as FilterOptions, - { - name: 'Sort By', - optionType: FilterOptionType.singleselect, - options: [ - { - label: 'Date - New to Old', - value: 'dateNewToOld', - }, - { - label: 'Date - Old to New', - value: 'dateOldToNew', - }, - { - label: 'Amount - High to Low', - value: 'amountHighToLow', - }, - { - label: 'Amount - Low to High', - value: 'amountLowToHigh', - }, - { - label: 'Category - A to Z', - value: 'categoryAToZ', - }, - { - label: 'Category - Z to A', - value: 'categoryZToA', - }, - ], - } as FilterOptions, - { - name: 'Split Expense', - optionType: FilterOptionType.singleselect, - options: [ - { - label: 'Yes', - value: 'YES', - }, - { - label: 'No', - value: 'NO', - }, - ], - } as FilterOptions, - ]; - } - - generateSelectedFilters(filter: Partial): SelectedFilters[] { - const generatedFilters: SelectedFilters[] = []; - - if (filter.state) { - generatedFilters.push({ - name: 'Type', - value: filter.state, - }); - } - - if (filter.receiptsAttached) { - generatedFilters.push({ - name: 'Receipts Attached', - value: filter.receiptsAttached, - }); - } - - if (filter.date) { - generatedFilters.push({ - name: 'Date', - value: filter.date, - associatedData: { - startDate: filter.customDateStart, - endDate: filter.customDateEnd, - }, - }); - } - - if (filter.type) { - generatedFilters.push({ - name: 'Expense Type', - value: filter.type, - }); - } - - if (filter.cardNumbers) { - generatedFilters.push({ - name: 'Cards', - value: filter.cardNumbers, - }); - } - - if (filter.sortParam && filter.sortDir) { - this.addSortToGeneratedFilters(filter, generatedFilters); - } - - if (filter.splitExpense) { - generatedFilters.push({ - name: 'Split Expense', - value: filter.splitExpense, - }); - } - - return generatedFilters; - } - - addSortToGeneratedFilters( - filter: Partial, - generatedFilters: SelectedFilters[] - ): void { - this.convertTxnDtSortToSelectedFilters(filter, generatedFilters); - - this.convertAmountSortToSelectedFilters(filter, generatedFilters); - - this.convertCategorySortToSelectedFilters(filter, generatedFilters); - } - - convertCategorySortToSelectedFilters( - filter: Partial, - generatedFilters: SelectedFilters[] - ): void { - if (filter.sortParam === 'category->name' && filter.sortDir === 'asc') { - generatedFilters.push({ - name: 'Sort By', - value: 'categoryAToZ', - }); - } else if (filter.sortParam === 'category->name' && filter.sortDir === 'desc') { - generatedFilters.push({ - name: 'Sort By', - value: 'categoryZToA', - }); - } - } - - convertAmountSortToSelectedFilters( - filter: Partial, - generatedFilters: SelectedFilters[] - ): void { - if (filter.sortParam === 'amount' && filter.sortDir === 'desc') { - generatedFilters.push({ - name: 'Sort By', - value: 'amountHighToLow', - }); - } else if (filter.sortParam === 'amount' && filter.sortDir === 'asc') { - generatedFilters.push({ - name: 'Sort By', - value: 'amountLowToHigh', - }); - } - } - - convertTxnDtSortToSelectedFilters( - filter: Partial, - generatedFilters: SelectedFilters[] - ): void { - if (filter.sortParam === 'spent_at' && filter.sortDir === 'asc') { - generatedFilters.push({ - name: 'Sort By', - value: 'dateOldToNew', - }); - } else if (filter.sortParam === 'spent_at' && filter.sortDir === 'desc') { - generatedFilters.push({ - name: 'Sort By', - value: 'dateNewToOld', - }); - } - } - - private generateSortCategoryPills(filter: Partial, filterPills: FilterPill[]): void { - if (filter.sortParam === 'category->name' && filter.sortDir === 'asc') { - filterPills.push({ - label: 'Sort By', - type: 'sort', - value: 'category - a to z', - }); - } else if (filter.sortParam === 'category->name' && filter.sortDir === 'desc') { - filterPills.push({ - label: 'Sort By', - type: 'sort', - value: 'category - z to a', - }); - } - } -} diff --git a/src/app/fyle/my-expenses-v2/add-expense-popover/add-expense-popover.component.html b/src/app/fyle/my-expenses/add-expense-popover/add-expense-popover.component.html similarity index 100% rename from src/app/fyle/my-expenses-v2/add-expense-popover/add-expense-popover.component.html rename to src/app/fyle/my-expenses/add-expense-popover/add-expense-popover.component.html diff --git a/src/app/fyle/my-expenses/my-expenses-filters.model.ts b/src/app/fyle/my-expenses/my-expenses-filters.model.ts index ddd0353bc5..6066994dc9 100644 --- a/src/app/fyle/my-expenses/my-expenses-filters.model.ts +++ b/src/app/fyle/my-expenses/my-expenses-filters.model.ts @@ -10,16 +10,3 @@ export type Filters = Partial<{ cardNumbers: string[]; splitExpense: string; }>; - -export type ReportFilters = Partial<{ - state: string | string[]; - date: string; - customDateStart: Date; - customDateEnd: Date; - receiptsAttached: string; - type: string[]; - sortParam: string; - sortDir: string; - cardNumbers: string[]; - splitExpense: string; -}>; diff --git a/src/app/fyle/my-expenses/my-expenses.page.html b/src/app/fyle/my-expenses/my-expenses.page.html index da40ddc2db..1e2a7754c4 100644 --- a/src/app/fyle/my-expenses/my-expenses.page.html +++ b/src/app/fyle/my-expenses/my-expenses.page.html @@ -66,7 +66,7 @@ - {{selectedElements?.length}} {{(selectedElements?.length > 1) ? 'expenses' : 'expense'}} worth + {{allExpensesStats?.count}} {{(allExpensesStats?.count > 1) ? 'expenses' : 'expense'}} worth {{(allExpensesStats.amount || 0) | humanizeCurrency: homeCurrency }} @@ -75,10 +75,7 @@ - + @@ -141,7 +138,7 @@ -
+
No Expenses Found
No expenses were found for the selected filter.
@@ -178,10 +175,10 @@ [previousExpenseTxnDate]="list[i-1]?.tx_txn_dt" [previousExpenseCreatedAt]="list[i-1]?.tx_created_at" [isSelectionModeEnabled]="selectionMode" - [selectedElements]="selectedElements" + [selectedElements]="selectedOutboxExpenses" [isOutboxExpense]="true" - (setMultiselectMode)="switchSelectionMode($event)" - (cardClickedForSelection)="selectExpense($event)" + (setMultiselectMode)="switchOutboxSelectionMode($event)" + (cardClickedForSelection)="selectOutboxExpense($event)" >
@@ -189,10 +186,10 @@
- - +
diff --git a/src/app/fyle/my-expenses/my-expenses.page.spec.ts b/src/app/fyle/my-expenses/my-expenses.page.spec.ts index d96cba4f05..9386478b54 100644 --- a/src/app/fyle/my-expenses/my-expenses.page.spec.ts +++ b/src/app/fyle/my-expenses/my-expenses.page.spec.ts @@ -20,17 +20,19 @@ import { allowedExpenseTypes } from 'src/app/core/mock-data/allowed-expense-type import { apiAuthRes } from 'src/app/core/mock-data/auth-reponse.data'; import { cardAggregateStatParam } from 'src/app/core/mock-data/card-aggregate-stats.data'; import { expectedAssignedCCCStats } from 'src/app/core/mock-data/ccc-expense.details.data'; -import { expectedCriticalPolicyViolationPopoverParams3 } from 'src/app/core/mock-data/critical-policy-violation-popover.data'; +import { + expectedCriticalPolicyViolationPopoverParams, + expectedCriticalPolicyViolationPopoverParams2, + expectedCriticalPolicyViolationPopoverParams3, +} from 'src/app/core/mock-data/critical-policy-violation-popover.data'; import { expenseFiltersData1, expenseFiltersData2 } from 'src/app/core/mock-data/expense-filters.data'; import { apiExpenseRes, expectedFormattedTransaction, expenseData1, expenseData2, - expenseData3, expenseList4, - mileageExpenseWithoutDistance, - perDiemExpenseSingleNumDays, + expenseListwithoutID, } from 'src/app/core/mock-data/expense.data'; import { cardFilterPill, @@ -46,12 +48,18 @@ import { typeFilterPill, } from 'src/app/core/mock-data/filter-pills.data'; import { filterOptions1 } from 'src/app/core/mock-data/filter.data'; -import { expectedCurrentParams } from 'src/app/core/mock-data/get-expenses-query-params-with-filters.data'; import { - addExpenseToReportModalParams, + expectedCurrentParamsCannotReportState, + expectedCurrentParamsDraftState, + expectedCurrentParamsWithDraftCannotReportState, + expectedCurrentParamsWoFilterState, +} from 'src/app/core/mock-data/get-expenses-query-params-with-filters.data'; +import { + addExpenseToReportModalParams2, modalControllerParams, modalControllerParams2, newReportModalParams, + newReportModalParams2, openFromComponentConfig, popoverControllerParams, } from 'src/app/core/mock-data/modal-controller.data'; @@ -59,6 +67,13 @@ import { fyModalProperties } from 'src/app/core/mock-data/model-properties.data' import { mileagePerDiemPlatformCategoryData } from 'src/app/core/mock-data/org-category.data'; import { orgSettingsParamsWithSimplifiedReport, orgSettingsRes } from 'src/app/core/mock-data/org-settings.data'; import { orgUserSettingsData } from 'src/app/core/mock-data/org-user-settings.data'; +import { + apiExpenses1, + apiExpenses2, + expenseData, + mileageExpenseWithDistance, + perDiemExpenseWithSingleNumDays, +} from 'src/app/core/mock-data/platform/v1/expense.data'; import { reportUnflattenedData } from 'src/app/core/mock-data/report-v1.data'; import { apiExtendedReportRes, expectedReportSingleResponse } from 'src/app/core/mock-data/report.data'; import { selectedFilters1, selectedFilters2 } from 'src/app/core/mock-data/selected-filters.data'; @@ -88,6 +103,8 @@ import { NetworkService } from 'src/app/core/services/network.service'; import { OrgSettingsService } from 'src/app/core/services/org-settings.service'; import { OrgUserSettingsService } from 'src/app/core/services/org-user-settings.service'; import { PlatformHandlerService } from 'src/app/core/services/platform-handler.service'; +import { ExpensesService as SharedExpenseService } from 'src/app/core/services/platform/v1/shared/expenses.service'; +import { ExpensesService } from 'src/app/core/services/platform/v1/spender/expenses.service'; import { PopupService } from 'src/app/core/services/popup.service'; import { ReportService } from 'src/app/core/services/report.service'; import { SnackbarPropertiesService } from 'src/app/core/services/snackbar-properties.service'; @@ -106,8 +123,9 @@ import { environment } from 'src/environments/environment'; import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog/add-txn-to-report-dialog.component'; import { MyExpensesPage } from './my-expenses.page'; import { MyExpensesService } from './my-expenses.service'; +import { completeStats, incompleteStats } from 'src/app/core/mock-data/platform/v1/expenses-stats.data'; -describe('MyExpensesPage', () => { +describe('MyExpensesV2Page', () => { let component: MyExpensesPage; let fixture: ComponentFixture; let tasksService: jasmine.SpyObj; @@ -139,6 +157,8 @@ describe('MyExpensesPage', () => { let popoverController: jasmine.SpyObj; let snackbarProperties: jasmine.SpyObj; let inputElement: HTMLInputElement; + let sharedExpenseService: jasmine.SpyObj; + let expensesService: jasmine.SpyObj; beforeEach(waitForAsync(() => { const tasksServiceSpy = jasmine.createSpyObj('TasksService', ['getReportsTaskCount', 'getExpensesTaskCount']); @@ -168,8 +188,8 @@ describe('MyExpensesPage', () => { 'getReportableExpenses', 'isMergeAllowed', 'excludeCCCExpenses', - 'getIsCriticalPolicyViolated', - 'getIsDraft', + 'isCriticalPolicyViolatedExpense', + 'isExpenseInDraft', 'getETxnUnflattened', 'getAllExpenses', 'getDeleteDialogBody', @@ -207,6 +227,7 @@ describe('MyExpensesPage', () => { 'convertFilters', 'generateSelectedFilters', 'getFilters', + 'convertSelectedOptionsToExpenseFilters', ]); const tokenServiceSpy = jasmine.createSpyObj('TokenService', ['getClusterDomain']); const actionSheetControllerSpy = jasmine.createSpyObj('ActionSheetController', ['create']); @@ -230,12 +251,39 @@ describe('MyExpensesPage', () => { 'addToReport', 'clickCreateReport', 'myExpensesBulkDeleteExpenses', + 'spenderSelectedPendingTxnFromMyExpenses', ]); const modalControllerSpy = jasmine.createSpyObj('ModalController', ['create']); const loaderServiceSpy = jasmine.createSpyObj('LoaderService', ['showLoader', 'hideLoader']); const popupServiceSpy = jasmine.createSpyObj('PopupService', ['showPopup']); const popoverControllerSpy = jasmine.createSpyObj('PopoverController', ['create']); const snackbarPropertiesSpy = jasmine.createSpyObj('SnackbarPropertiesService', ['setSnackbarProperties']); + const expensesServiceSpy = jasmine.createSpyObj('ExpensesService', [ + 'getExpensesCount', + 'getExpenses', + 'getAllExpenses', + 'getExpenseById', + 'getExpenseStats', + ]); + const sharedExpenseServiceSpy = jasmine.createSpyObj('SharedExpenseService', [ + 'generateCardNumberParams', + 'generateDateParams', + 'generateReceiptAttachedParams', + 'generateStateFilters', + 'generateTypeFilters', + 'setSortParams', + 'generateSplitExpenseParams', + 'getReportableExpenses', + 'excludeCCCExpenses', + 'isMergeAllowed', + 'isCriticalPolicyViolatedExpense', + 'isExpenseInDraft', + 'getExpenseDeletionMessage', + 'getCCCExpenseMessage', + 'getDeleteDialogBody', + 'restrictPendingTransactionsEnabled', + 'doesExpenseHavePendingCardTransaction', + ]); TestBed.configureTestingModule({ declarations: [MyExpensesPage, ReportState, MaskNumber], @@ -333,6 +381,14 @@ describe('MyExpensesPage', () => { provide: CategoriesService, useValue: categoriesServiceSpy, }, + { + provide: ExpensesService, + useValue: expensesServiceSpy, + }, + { + provide: SharedExpenseService, + useValue: sharedExpenseServiceSpy, + }, ReportState, MaskNumber, ], @@ -375,7 +431,10 @@ describe('MyExpensesPage', () => { popupService = TestBed.inject(PopupService) as jasmine.SpyObj; popoverController = TestBed.inject(PopoverController) as jasmine.SpyObj; snackbarProperties = TestBed.inject(SnackbarPropertiesService) as jasmine.SpyObj; - component.loadData$ = new BehaviorSubject({}); + expensesService = TestBed.inject(ExpensesService) as jasmine.SpyObj; + sharedExpenseService = TestBed.inject(SharedExpenseService) as jasmine.SpyObj; + + component.loadExpenses$ = new BehaviorSubject({}); })); it('should create', () => { @@ -407,20 +466,14 @@ describe('MyExpensesPage', () => { spyOn(component, 'setupActionSheet'); tokenService.getClusterDomain.and.resolveTo(apiAuthRes.cluster_domain); currencyService.getHomeCurrency.and.returnValue(of('USD')); - apiV2Service.extendQueryParamsForTextSearch.and.returnValue({ - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }); - transactionService.getMyExpensesCount.and.returnValue(of(10)); - transactionService.getTransactionStats.and.returnValue(of(transactionDatum1)); - transactionService.getMyExpenses.and.returnValue( - of({ count: 2, limit: 10, offset: 0, data: apiExpenseRes, url: '' }) - ); - transactionService.getPaginatedETxncCount.and.returnValue(of({ count: 10 })); + expensesService.getExpenseStats.and.returnValue(of(completeStats)); + expensesService.getExpensesCount.and.returnValue(of(10)); + expensesService.getExpenses.and.returnValue(of(apiExpenses1)); + reportService.getAllExtendedReports.and.returnValue(of(apiExtendedReportRes)); spyOn(component, 'doRefresh'); spyOn(component, 'backButtonAction'); - transactionOutboxService.getPendingTransactions.and.returnValue(txnList); + spyOn(component, 'formatTransactions').and.returnValue(apiExpenseRes); spyOn(component, 'addNewFiltersToParams').and.returnValue({ pageNumber: 1, sortDir: 'desc' }); spyOn(component, 'generateFilterPills').and.returnValue(creditTxnFilterPill); @@ -635,7 +688,7 @@ describe('MyExpensesPage', () => { inputElement.dispatchEvent(new Event('keyup')); tick(500); - component.loadData$.subscribe((loadData) => { + component.loadExpenses$.subscribe((loadData) => { expect(loadData).toEqual({ pageNumber: 1, searchString: 'example' }); }); })); @@ -646,37 +699,22 @@ describe('MyExpensesPage', () => { inputElement.value = 'example'; inputElement.dispatchEvent(new Event('keyup')); tick(500); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledTimes(4); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledWith( - { - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }, - undefined - ); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledWith( - { - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }, - 'example' - ); - expect(transactionService.getMyExpensesCount).toHaveBeenCalledTimes(4); - expect(transactionService.getMyExpensesCount).toHaveBeenCalledWith({ - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', + + expect(expensesService.getExpensesCount).toHaveBeenCalledTimes(5); + expect(expensesService.getExpensesCount).toHaveBeenCalledWith({ + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', }); - expect(transactionService.getMyExpenses).toHaveBeenCalledTimes(2); - expect(transactionService.getMyExpenses).toHaveBeenCalledWith({ + expect(expensesService.getExpenses).toHaveBeenCalledTimes(2); + expect(expensesService.getExpenses).toHaveBeenCalledWith({ offset: 0, limit: 10, - queryParams: { - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }, - order: null, + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', + order: 'spent_at.desc,created_at.desc,id.desc', }); - expect(component.acc).toEqual(apiExpenseRes); + + expect(component.acc).toEqual(apiExpenses1); })); it('should not call getMyExpenses if count is less than (params.pageNumber - 1) * 10', fakeAsync(() => { @@ -687,69 +725,41 @@ describe('MyExpensesPage', () => { inputElement.dispatchEvent(new Event('keyup')); tick(500); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledTimes(4); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledWith( - { - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }, - undefined - ); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledWith( - { - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }, - 'example' - ); - expect(transactionService.getMyExpensesCount).toHaveBeenCalledTimes(4); - expect(transactionService.getMyExpensesCount).toHaveBeenCalledWith({ - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', + expect(expensesService.getExpensesCount).toHaveBeenCalledTimes(5); + expect(expensesService.getExpensesCount).toHaveBeenCalledWith({ + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', }); expect(component.clusterDomain).toEqual(apiAuthRes.cluster_domain); expect(transactionService.getMyExpenses).not.toHaveBeenCalled(); - expect(component.acc).toEqual([]); + expect(component.acc).toEqual(apiExpenses1); })); it('should call getMyExpenseCount with order if sortDir and sortParam are defined', fakeAsync(() => { component.ionViewWillEnter(); - component.loadData$.next({ + component.loadExpenses$.next({ pageNumber: 1, sortDir: 'asc', sortParam: 'approvalDate', }); tick(500); - expect(transactionService.getMyExpenses).toHaveBeenCalledTimes(2); - expect(transactionService.getMyExpenses).toHaveBeenCalledWith({ + expect(expensesService.getExpenses).toHaveBeenCalledTimes(2); + expect(expensesService.getExpenses).toHaveBeenCalledWith({ offset: 0, limit: 10, - queryParams: { - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }, - order: 'approvalDate.asc', + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', + order: 'spent_at.desc,created_at.desc,id.desc', }); })); - it('should set pendingTransactions by calling transactionOutboxService', fakeAsync(() => { - component.ionViewWillEnter(); - tick(500); - - expect(component.isLoadingDataInInfiniteScroll).toBeFalse(); - expect(component.acc).toEqual(apiExpenseRes); - expect(transactionOutboxService.getPendingTransactions).toHaveBeenCalledTimes(1); - expect(component.pendingTransactions).toEqual(apiExpenseRes); - expect(component.formatTransactions).toHaveBeenCalledTimes(1); - })); - it('should set myExpenses$, count$, isNewUser$ and isInfiniteScrollRequired', fakeAsync(() => { component.ionViewWillEnter(); tick(500); component.myExpenses$.subscribe((myExpenses) => { - expect(myExpenses).toEqual(apiExpenseRes); + expect(myExpenses).toEqual(apiExpenses1); }); component.count$.subscribe((count) => { expect(count).toBe(10); @@ -774,22 +784,20 @@ describe('MyExpensesPage', () => { tick(500); component.allExpenseCountHeader$.subscribe((allExpenseCountHeader) => { - expect(transactionService.getTransactionStats).toHaveBeenCalledWith('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_state: 'in.(COMPLETE,DRAFT)', - tx_report_id: 'is.null', + expect(expensesService.getExpenseStats).toHaveBeenCalledWith({ + state: 'in.(COMPLETE,DRAFT)', + report_id: 'is.null', }); - expect(allExpenseCountHeader).toBe(4); + expect(allExpenseCountHeader).toBe(3); }); component.draftExpensesCount$.subscribe((draftExpensesCount) => { - expect(transactionService.getTransactionStats).toHaveBeenCalledWith('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_report_id: 'is.null', - tx_state: 'in.(DRAFT)', + expect(expensesService.getExpenseStats).toHaveBeenCalledWith({ + report_id: 'is.null', + state: 'in.(DRAFT)', }); - expect(draftExpensesCount).toBe(4); + expect(draftExpensesCount).toBe(3); }); - expect(transactionService.getTransactionStats).toHaveBeenCalledTimes(2); + expect(expensesService.getExpenseStats).toHaveBeenCalledTimes(2); })); it('should navigate relative to activatedRoute and call clearFilters if snapshot.params and queryParams are undefined', fakeAsync(() => { @@ -821,7 +829,7 @@ describe('MyExpensesPage', () => { }; expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((loadData) => { + component.loadExpenses$.subscribe((loadData) => { expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); }); expect(component.filterPills).toEqual(creditTxnFilterPill); @@ -835,19 +843,13 @@ describe('MyExpensesPage', () => { component.ionViewWillEnter(); tick(500); expect(component.clearFilters).not.toHaveBeenCalled(); - expect(component.filters).toEqual({ - tx_receipt_required: 'eq.true', - state: 'NEEDS_RECEIPT', - }); + expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((loadData) => { + component.loadExpenses$.subscribe((loadData) => { expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); }); - expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ - state: 'NEEDS_RECEIPT', - tx_receipt_required: 'eq.true', - }); + expect(component.generateFilterPills).toHaveBeenCalledTimes(1); expect(component.filterPills).toEqual(creditTxnFilterPill); })); @@ -861,19 +863,19 @@ describe('MyExpensesPage', () => { expect(component.clearFilters).not.toHaveBeenCalled(); expect(component.filters).toEqual({ - tx_policy_flag: 'eq.true', - or: '(tx_policy_amount.is.null,tx_policy_amount.gt.0.0001)', + is_policy_flagged: 'eq.true', + or: '(policy_amount.is.null,policy_amount.gt.0.0001)', state: 'POLICY_VIOLATED', }); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((loadData) => { + component.loadExpenses$.subscribe((loadData) => { expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); }); expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ - tx_policy_flag: 'eq.true', - or: '(tx_policy_amount.is.null,tx_policy_amount.gt.0.0001)', state: 'POLICY_VIOLATED', + is_policy_flagged: 'eq.true', + or: '(policy_amount.is.null,policy_amount.gt.0.0001)', }); expect(component.filterPills).toEqual(creditTxnFilterPill); })); @@ -888,16 +890,16 @@ describe('MyExpensesPage', () => { expect(component.clearFilters).not.toHaveBeenCalled(); expect(component.filters).toEqual({ - tx_policy_amount: 'lt.0.0001', + policy_amount: 'lt.0.0001', state: 'CANNOT_REPORT', }); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((loadData) => { + component.loadExpenses$.subscribe((loadData) => { expect(loadData).toEqual({ pageNumber: 1, sortDir: 'desc' }); }); expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ - tx_policy_amount: 'lt.0.0001', + policy_amount: 'lt.0.0001', state: 'CANNOT_REPORT', }); expect(component.filterPills).toEqual(creditTxnFilterPill); @@ -934,6 +936,24 @@ describe('MyExpensesPage', () => { }); expect(component.doRefresh).toHaveBeenCalledTimes(1); })); + + it('should return an empty array if no expenses are found and search is empty', fakeAsync(() => { + expensesService.getExpensesCount.and.returnValue(of(0)); + component.ionViewWillEnter(); + component.loadExpenses$.next({ + pageNumber: 1, + searchString: '', + sortParam: 'category->name', + sortDir: 'asc', + }); + tick(500); + + component.myExpenses$.subscribe((res) => { + expect(res).toEqual([]); + }); + + expect(expensesService.getExpenses).not.toHaveBeenCalled(); + })); }); it('HeaderState(): should return the headerState', () => { @@ -986,7 +1006,7 @@ describe('MyExpensesPage', () => { describe('switchSelectionMode(): ', () => { beforeEach(() => { component.selectionMode = true; - component.loadData$ = new BehaviorSubject({ + component.loadExpenses$ = new BehaviorSubject({ searchString: 'example', }); component.headerState = HeaderState.simpleSearch; @@ -1005,16 +1025,15 @@ describe('MyExpensesPage', () => { }); it('should set headerState to base if searchString is defined in loadData and if expense is selected', () => { - component.loadData$ = new BehaviorSubject({}); - const expense = apiExpenseRes[0]; + component.loadExpenses$ = new BehaviorSubject({}); - component.switchSelectionMode(expense); + component.switchSelectionMode(apiExpenses1[0]); expect(component.selectionMode).toBeFalse(); expect(component.headerState).toBe(HeaderState.base); expect(component.selectedElements).toEqual([]); expect(component.setAllExpensesCountAndAmount).toHaveBeenCalledTimes(1); - expect(component.selectExpense).toHaveBeenCalledOnceWith(expense); + expect(component.selectExpense).toHaveBeenCalledOnceWith(apiExpenses1[0]); }); it('should update allExpensesStats$ and headerState if selectionMode is false', () => { @@ -1033,6 +1052,56 @@ describe('MyExpensesPage', () => { }); }); + describe('switchOutboxSelectionMode(): ', () => { + beforeEach(() => { + component.selectionMode = true; + component.loadExpenses$ = new BehaviorSubject({ + searchString: 'example', + }); + component.headerState = HeaderState.simpleSearch; + component.allExpensesStats$ = of({ count: 10, amount: 1000 }); + spyOn(component, 'selectExpense'); + spyOn(component, 'setAllExpensesCountAndAmount'); + spyOn(component, 'setOutboxExpenseStatsOnSelect'); + }); + it('should set headerState to simpleSearch if searchString is defined in loadData', () => { + component.switchOutboxSelectionMode(); + + expect(component.selectionMode).toBeFalse(); + expect(component.headerState).toBe(HeaderState.simpleSearch); + expect(component.selectedOutboxExpenses).toEqual([]); + expect(component.selectExpense).not.toHaveBeenCalled(); + }); + + it('should set headerState to base if searchString is defined in loadData and if expense is selected', () => { + component.loadExpenses$ = new BehaviorSubject({}); + transactionService.getReportableExpenses.and.returnValue([]); + transactionService.excludeCCCExpenses.and.returnValue([]); + + component.switchOutboxSelectionMode(apiExpenseRes[0]); + + expect(component.selectionMode).toBeFalse(); + expect(component.headerState).toBe(HeaderState.base); + expect(component.selectedOutboxExpenses.length).toEqual(1); + expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(2); + }); + + it('should update allExpensesStats$ and headerState if selectionMode is false', () => { + component.selectionMode = false; + + component.switchOutboxSelectionMode(); + + expect(component.selectionMode).toBeTrue(); + expect(component.headerState).toBe(HeaderState.multiselect); + expect(component.setOutboxExpenseStatsOnSelect).not.toHaveBeenCalled(); + expect(component.selectExpense).not.toHaveBeenCalled(); + component.allExpensesStats$.subscribe((stats) => { + expect(stats.count).toBe(0); + expect(stats.amount).toBe(0); + }); + }); + }); + it('sendFirstExpenseCreatedEvent(): should store the first expense created event', fakeAsync(() => { component.allExpensesStats$ = of({ count: 0, @@ -1047,65 +1116,63 @@ describe('MyExpensesPage', () => { })); describe('setAllExpensesCountAndAmount(): ', () => { - it('should call transactionService.getTransactionStats if loadData contains queryParams', () => { - component.loadData$ = new BehaviorSubject({ + it('should call expensesService.getExpenseStats if loadExpenses contains queryParams', () => { + component.loadExpenses$ = new BehaviorSubject({ queryParams: { - corporate_credit_card_account_number: '8698', + 'matched_corporate_card_transactions->0->corporate_card_number': '8698', }, }); - transactionService.getTransactionStats.and.returnValue(of(transactionDatum1)); + + expensesService.getExpenseStats.and.returnValue(of(completeStats)); component.setAllExpensesCountAndAmount(); component.allExpensesStats$.subscribe((allExpenseStats) => { - expect(transactionService.getTransactionStats).toHaveBeenCalledOnceWith('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - or: ['(corporate_credit_card_account_number.8698)'], + expect(expensesService.getExpenseStats).toHaveBeenCalledOnceWith({ + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', + or: ['(matched_corporate_card_transactions->0->corporate_card_number.8698)'], }); expect(allExpenseStats).toEqual({ - count: 4, - amount: 3494, + count: 3, + amount: 30, }); }); }); - it('should call transactionService.getTransactionStats and initialize queryParams to empty object if loadData.queryParams is falsy', () => { - component.loadData$ = new BehaviorSubject({ + it('should call expensesService.getExpenseStats and initialize queryParams to empty object if loadData.queryParams is falsy', () => { + component.loadExpenses$ = new BehaviorSubject({ queryParams: null, }); - transactionService.getTransactionStats.and.returnValue(of(transactionDatum3)); + expensesService.getExpenseStats.and.returnValue(of(incompleteStats)); component.setAllExpensesCountAndAmount(); component.allExpensesStats$.subscribe((allExpenseStats) => { - expect(transactionService.getTransactionStats).toHaveBeenCalledOnceWith('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', + expect(expensesService.getExpenseStats).toHaveBeenCalledOnceWith({ + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', }); expect(allExpenseStats).toEqual({ - count: 4, - amount: 0, + count: incompleteStats.data.count, + amount: incompleteStats.data.total_amount, }); }); }); - it('should handle error in getTransactionStats and complete the observable', () => { - component.loadData$ = new BehaviorSubject({ + it('should handle error in getExpenseStats and complete the observable', () => { + component.loadExpenses$ = new BehaviorSubject({ queryParams: { - corporate_credit_card_account_number: '8698', + 'matched_corporate_card_transactions->0->corporate_card_number': '8698', }, }); - transactionService.getTransactionStats.and.returnValue(throwError(() => new Error('error message'))); + expensesService.getExpenseStats.and.returnValue(throwError(() => new Error('error message'))); component.setAllExpensesCountAndAmount(); component.allExpensesStats$.subscribe({ - complete: () => { - expect().nothing(); + error: (err) => { + expect(err.message).toEqual('error message'); }, }); - expect(transactionService.getTransactionStats).toHaveBeenCalledOnceWith('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - or: ['(corporate_credit_card_account_number.8698)'], + expect(expensesService.getExpenseStats).toHaveBeenCalledOnceWith({ + report_id: 'is.null', + state: 'in.(COMPLETE,DRAFT)', + or: ['(matched_corporate_card_transactions->0->corporate_card_number.8698)'], }); }); }); @@ -1265,7 +1332,7 @@ describe('MyExpensesPage', () => { describe('loadData(): ', () => { beforeEach(() => { component.currentPageNumber = 2; - component.loadData$ = new BehaviorSubject({ + component.loadExpenses$ = new BehaviorSubject({ pageNumber: 2, }); }); @@ -1275,7 +1342,7 @@ describe('MyExpensesPage', () => { component.loadData(mockEvent); expect(component.currentPageNumber).toBe(3); - expect(component.loadData$.getValue().pageNumber).toBe(3); + expect(component.loadExpenses$.getValue().pageNumber).toBe(3); tick(1000); expect(mockEvent.target.complete).toHaveBeenCalledTimes(1); })); @@ -1286,7 +1353,7 @@ describe('MyExpensesPage', () => { component.loadData(mockEvent); expect(component.currentPageNumber).toBe(3); - expect(component.loadData$.getValue().pageNumber).toBe(3); + expect(component.loadExpenses$.getValue().pageNumber).toBe(3); }); it('should increment currentPageNumber and emit updated params if event if undefined', () => { @@ -1295,7 +1362,7 @@ describe('MyExpensesPage', () => { component.loadData(mockEvent); expect(component.currentPageNumber).toBe(3); - expect(component.loadData$.getValue().pageNumber).toBe(3); + expect(component.loadExpenses$.getValue().pageNumber).toBe(3); }); }); @@ -1303,7 +1370,7 @@ describe('MyExpensesPage', () => { beforeEach(() => { transactionService.clearCache.and.returnValue(of(null)); component.currentPageNumber = 2; - component.loadData$ = new BehaviorSubject({ + component.loadExpenses$ = new BehaviorSubject({ pageNumber: 2, }); spyOn(component, 'setExpenseStatsOnSelect'); @@ -1315,7 +1382,7 @@ describe('MyExpensesPage', () => { expect(component.selectedElements).toEqual([]); expect(transactionService.clearCache).toHaveBeenCalledTimes(1); expect(component.currentPageNumber).toBe(1); - expect(component.loadData$.getValue().pageNumber).toBe(1); + expect(component.loadExpenses$.getValue().pageNumber).toBe(1); })); it('should refresh data and call complete if ionRefresher event if present and selectionMode is true', fakeAsync(() => { @@ -1329,7 +1396,7 @@ describe('MyExpensesPage', () => { expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); expect(transactionService.clearCache).toHaveBeenCalledTimes(1); expect(component.currentPageNumber).toBe(1); - expect(component.loadData$.getValue().pageNumber).toBe(1); + expect(component.loadExpenses$.getValue().pageNumber).toBe(1); expect(mockEvent.target.complete).toHaveBeenCalledTimes(1); })); @@ -1342,7 +1409,7 @@ describe('MyExpensesPage', () => { expect(component.selectedElements).toEqual([]); expect(transactionService.clearCache).toHaveBeenCalledTimes(1); expect(component.currentPageNumber).toBe(1); - expect(component.loadData$.getValue().pageNumber).toBe(1); + expect(component.loadExpenses$.getValue().pageNumber).toBe(1); })); }); @@ -1398,38 +1465,35 @@ describe('MyExpensesPage', () => { describe('addNewFiltersToParams(): ', () => { beforeEach(() => { - component.loadData$ = new BehaviorSubject({ - pageNumber: 2, - }); - transactionService.generateCardNumberParams.and.returnValue({ - corporate_credit_card_account_number: 'in.(789)', + sharedExpenseService.generateCardNumberParams.and.returnValue({ + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [], }); - transactionService.generateDateParams.and.returnValue({ - corporate_credit_card_account_number: 'in.(789)', - and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', + sharedExpenseService.generateDateParams.and.returnValue({ + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', or: [], }); - transactionService.generateReceiptAttachedParams.and.returnValue({ - corporate_credit_card_account_number: 'in.(789)', - and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', + sharedExpenseService.generateReceiptAttachedParams.and.returnValue({ + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', or: [], }); - transactionService.generateStateFilters.and.returnValue({ - corporate_credit_card_account_number: 'in.(789)', - and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', + sharedExpenseService.generateStateFilters.and.returnValue({ + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', or: [], }); - transactionService.generateTypeFilters.and.returnValue({ - corporate_credit_card_account_number: 'in.(789)', - and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', + sharedExpenseService.generateTypeFilters.and.returnValue({ + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', or: [], }); - transactionService.setSortParams.and.returnValue({ sortDir: 'asc' }); - transactionService.generateSplitExpenseParams.and.returnValue({ - or: ['(tx_is_split_expense.eq.true)'], - corporate_credit_card_account_number: 'in.(789)', - and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', + sharedExpenseService.setSortParams.and.returnValue({ sortDir: 'asc' }); + sharedExpenseService.generateSplitExpenseParams.and.returnValue({ + or: ['(is_split.eq.true)'], + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', }); }); @@ -1438,30 +1502,46 @@ describe('MyExpensesPage', () => { const currentParams = component.addNewFiltersToParams(); - expect(transactionService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(transactionService.generateDateParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', or: [] }, + expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); + expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( + { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, component.filters ); - expect(transactionService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateStateFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateTypeFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(transactionService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); + expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(currentParams).toEqual(expectedCurrentParams); + expect(currentParams).toEqual(expectedCurrentParamsWoFilterState); expect(component.reviewMode).toBeFalse(); }); @@ -1472,30 +1552,46 @@ describe('MyExpensesPage', () => { const currentParams = component.addNewFiltersToParams(); - expect(transactionService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(transactionService.generateDateParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', or: [] }, + expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); + expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( + { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, component.filters ); - expect(transactionService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateStateFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateTypeFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(transactionService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); + expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(currentParams).toEqual(expectedCurrentParams); + expect(currentParams).toEqual(expectedCurrentParamsDraftState); expect(component.reviewMode).toBeTrue(); }); @@ -1506,30 +1602,46 @@ describe('MyExpensesPage', () => { const currentParams = component.addNewFiltersToParams(); - expect(transactionService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(transactionService.generateDateParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', or: [] }, + expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); + expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( + { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, component.filters ); - expect(transactionService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateStateFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateTypeFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(transactionService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); + expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(currentParams).toEqual(expectedCurrentParams); + expect(currentParams).toEqual(expectedCurrentParamsCannotReportState); expect(component.reviewMode).toBeTrue(); }); @@ -1540,30 +1652,46 @@ describe('MyExpensesPage', () => { const currentParams = component.addNewFiltersToParams(); - expect(transactionService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); - expect(transactionService.generateDateParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', or: [] }, + expect(sharedExpenseService.generateCardNumberParams).toHaveBeenCalledOnceWith({ or: [] }, component.filters); + expect(sharedExpenseService.generateDateParams).toHaveBeenCalledOnceWith( + { 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', or: [] }, component.filters ); - expect(transactionService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateReceiptAttachedParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateStateFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateStateFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.generateTypeFilters).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.generateTypeFilters).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(transactionService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); - expect(transactionService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( - { corporate_credit_card_account_number: 'in.(789)', and: '(tx_txn_dt.gte.March,tx_txn_dt.lt.April)', or: [] }, + expect(sharedExpenseService.setSortParams).toHaveBeenCalledOnceWith({ pageNumber: 1 }, component.filters); + expect(sharedExpenseService.generateSplitExpenseParams).toHaveBeenCalledOnceWith( + { + 'matched_corporate_card_transactions->0->corporate_card_number': 'in.(789)', + and: '(spent_at.gte.March,spent_at.lt.April)', + or: [], + }, component.filters ); - expect(currentParams).toEqual(expectedCurrentParams); + expect(currentParams).toEqual(expectedCurrentParamsWithDraftCannotReportState); expect(component.reviewMode).toBeTrue(); }); @@ -1587,11 +1715,11 @@ describe('MyExpensesPage', () => { state: [], }; myExpenseService.generateSelectedFilters.and.returnValue(selectedFilters1); - component.loadData$ = new BehaviorSubject({ + component.loadExpenses$ = new BehaviorSubject({ pageNumber: 1, }); component.currentPageNumber = 2; - myExpenseService.convertFilters.and.returnValue({ sortDir: 'asc', splitExpense: 'YES' }); + myExpenseService.convertSelectedOptionsToExpenseFilters.and.returnValue({ sortDir: 'asc', splitExpense: 'YES' }); spyOn(component, 'addNewFiltersToParams').and.returnValue({ searchString: 'example' }); spyOn(component, 'generateFilterPills').and.returnValue([ { @@ -1614,11 +1742,11 @@ describe('MyExpensesPage', () => { tick(200); expect(modalController.create).toHaveBeenCalledOnceWith(modalControllerParams); - expect(myExpenseService.convertFilters).toHaveBeenCalledOnceWith(selectedFilters2); + expect(myExpenseService.convertSelectedOptionsToExpenseFilters).toHaveBeenCalledOnceWith(selectedFilters2); expect(component.filters).toEqual({ sortDir: 'asc', splitExpense: 'YES' }); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((loadData) => { + component.loadExpenses$.subscribe((loadData) => { expect(loadData).toEqual({ searchString: 'example' }); }); @@ -1635,12 +1763,12 @@ describe('MyExpensesPage', () => { expect(modalController.create).toHaveBeenCalledOnceWith(modalControllerParams2); - expect(myExpenseService.convertFilters).toHaveBeenCalledOnceWith(selectedFilters2); + expect(myExpenseService.convertSelectedOptionsToExpenseFilters).toHaveBeenCalledOnceWith(selectedFilters2); expect(component.filters).toEqual({ sortDir: 'asc', splitExpense: 'YES' }); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((loadData) => { - expect(loadData).toEqual({ searchString: 'example' }); + component.loadExpenses$.subscribe((loadExpenses) => { + expect(loadExpenses).toEqual({ searchString: 'example' }); }); expect(component.generateFilterPills).toHaveBeenCalledOnceWith({ sortDir: 'asc', splitExpense: 'YES' }); expect(component.filterPills).toEqual(creditTxnFilterPill); @@ -1651,7 +1779,7 @@ describe('MyExpensesPage', () => { it('clearFilters(): should clear the filters and call generateFilterPills', () => { component.filters = { sortDir: 'asc', - sortParam: 'tx_org_category', + sortParam: 'category->name', }; component.currentPageNumber = 3; spyOn(component, 'addNewFiltersToParams').and.returnValue({ @@ -1666,7 +1794,7 @@ describe('MyExpensesPage', () => { expect(component.filters).toEqual({}); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((data) => { + component.loadExpenses$.subscribe((data) => { expect(data).toEqual({ pageNumber: 1, searchString: 'example', @@ -1681,7 +1809,7 @@ describe('MyExpensesPage', () => { pageNumber: 1, searchString: 'example', }); - component.loadData$ = new BehaviorSubject({ + component.loadExpenses$ = new BehaviorSubject({ pageNumber: 1, }); @@ -1689,7 +1817,7 @@ describe('MyExpensesPage', () => { expect(component.isLoading).toBeTrue(); expect(component.currentPageNumber).toBe(1); - component.loadData$.subscribe((data) => { + component.loadExpenses$.subscribe((data) => { expect(data).toEqual({ pageNumber: 1, searchString: 'example', @@ -1701,129 +1829,145 @@ describe('MyExpensesPage', () => { describe('selectExpense(): ', () => { beforeEach(() => { - transactionService.getReportableExpenses.and.returnValue(apiExpenseRes); - component.allExpensesCount = 1; + sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); + component.allExpensesCount = 2; spyOn(component, 'setExpenseStatsOnSelect'); - component.selectedElements = cloneDeep(apiExpenseRes); - transactionService.isMergeAllowed.and.returnValue(true); - transactionService.excludeCCCExpenses.and.returnValue(apiExpenseRes); + component.selectedElements = cloneDeep(apiExpenses1); + sharedExpenseService.isMergeAllowed.and.returnValue(true); + sharedExpenseService.excludeCCCExpenses.and.returnValue(apiExpenses1); }); it('should remove an expense from selectedElements if it is present in selectedElements', () => { - transactionService.getReportableExpenses.and.returnValue([]); - const expense = apiExpenseRes[0]; - component.selectedElements = cloneDeep(apiExpenseRes); + sharedExpenseService.getReportableExpenses.and.returnValue([]); + const expense = apiExpenses1[0]; + component.selectedElements = cloneDeep(apiExpenses1); component.selectExpense(expense); - expect(component.selectedElements).toEqual([]); + expect(component.selectedElements).toEqual([apiExpenses1[1]]); expect(component.isReportableExpensesSelected).toBeFalse(); expect(component.selectAll).toBeFalse(); expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); + expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([apiExpenses1[1]]); expect(component.isMergeAllowed).toBeTrue(); }); it('should remove an expense from selectedElements if it is present in selectedElements', () => { - transactionService.getReportableExpenses.and.returnValue([]); - component.allExpensesCount = 4; - const expense = apiExpenseRes[0]; - component.selectedElements = cloneDeep(cloneDeep(expenseList4)); + sharedExpenseService.getReportableExpenses.and.returnValue([]); + component.allExpensesCount = 3; - component.selectExpense(expense); + component.selectedElements = cloneDeep(cloneDeep(apiExpenses1)); + + component.selectExpense(expenseData); - expect(component.selectedElements).toEqual([...expenseList4, expense]); + expect(component.selectedElements).toEqual([...apiExpenses1, expenseData]); expect(component.isReportableExpensesSelected).toBeFalse(); expect(component.selectAll).toBeTrue(); expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([...expenseList4, expense]); + expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([...apiExpenses1, expenseData]); expect(component.isMergeAllowed).toBeTrue(); }); it('should remove an expense from selectedElements if it is present in selectedElements and allExpenseCount is not equal to length of selectedElements', () => { - transactionService.getReportableExpenses.and.returnValue([]); - const expense = apiExpenseRes[0]; - component.selectedElements = cloneDeep(apiExpenseRes); + sharedExpenseService.getReportableExpenses.and.returnValue([]); - component.selectExpense(expense); + component.selectedElements = cloneDeep(apiExpenses1); - expect(component.selectedElements).toEqual([]); + component.selectExpense(apiExpenses1[0]); + + expect(component.selectedElements).toEqual([apiExpenses1[1]]); expect(component.isReportableExpensesSelected).toBeFalse(); expect(component.selectAll).toBeFalse(); expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); + expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([apiExpenses1[1]]); expect(component.isMergeAllowed).toBeTrue(); }); it('should update expenseToBeDeleted if selectedElements is an array of atleast 1', () => { - component.selectedElements = cloneDeep(apiExpenseRes); - component.selectExpense(expenseData2); + sharedExpenseService.excludeCCCExpenses.and.returnValue([apiExpenses1[1]]); + component.selectedElements = cloneDeep(apiExpenses1); + component.selectExpense(apiExpenses1[0]); - const expectedSelectedElements = [...apiExpenseRes, expenseData2]; - expect(component.selectedElements).toEqual(expectedSelectedElements); - expect(component.expensesToBeDeleted).toEqual(apiExpenseRes); - expect(component.cccExpenses).toBe(1); + expect(component.selectedElements).toEqual([apiExpenses1[1]]); + expect(component.expensesToBeDeleted).toEqual([apiExpenses1[1]]); + expect(component.cccExpenses).toBe(0); expect(component.selectAll).toBeFalse(); }); it('should remove an expense from selectedElements if it is present in selectedElements and tx_id is not present in expense', () => { - transactionService.getReportableExpenses.and.returnValue([]); + sharedExpenseService.getReportableExpenses.and.returnValue([]); component.allExpensesCount = 0; - const expense = cloneDeep(apiExpenseRes[0]); - expense.tx_id = undefined; - component.selectedElements = cloneDeep(apiExpenseRes); - component.selectedElements[0].tx_id = undefined; + const expense = cloneDeep(apiExpenses1[0]); + expense.id = undefined; + component.selectedElements = cloneDeep(apiExpenses1); + component.selectedElements[0].id = undefined; component.selectExpense(expense); - expect(component.selectedElements).toEqual([]); + expect(component.selectedElements).toEqual([apiExpenses1[1]]); expect(component.isReportableExpensesSelected).toBeFalse(); - expect(component.selectAll).toBeTrue(); + expect(component.selectAll).toBeFalse(); expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); - expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); + expect(sharedExpenseService.isMergeAllowed).toHaveBeenCalledOnceWith([apiExpenses1[1]]); expect(component.isMergeAllowed).toBeTrue(); }); }); it('setExpenseStatsOnSelect(): should update allExpenseStats$', () => { - component.selectedElements = expenseList4; + component.selectedElements = apiExpenses1; component.setExpenseStatsOnSelect(); component.allExpensesStats$.subscribe((expenseStats) => { expect(expenseStats).toEqual({ - count: 3, - amount: 49475.76, + count: 2, + amount: 25, }); }); }); describe('goToTransaction():', () => { it('should navigate to add_edit_mileage page if category is mileage', () => { - component.goToTransaction({ etxn: mileageExpenseWithoutDistance, etxnIndex: 1 }); + component.goToTransaction({ + expense: { ...expenseData, category: { ...expenseData.category, name: 'mileage' } }, + expenseIndex: 1, + }); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'add_edit_mileage', - { id: 'txEpXa1cd6oq', persist_filters: true }, + { id: expenseData.id, persist_filters: true }, ]); }); it('should navigate to add_edit_per_diem if category is per diem', () => { - component.goToTransaction({ etxn: perDiemExpenseSingleNumDays, etxnIndex: 1 }); + component.goToTransaction({ + expense: { ...expenseData, category: { ...expenseData.category, name: 'per diem' } }, + expenseIndex: 1, + }); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'add_edit_per_diem', - { id: 'txWDbbZhNwdA', persist_filters: true }, + { id: expenseData.id, persist_filters: true }, ]); }); it('should navigate to add_edit_expense if category is something else', () => { - component.goToTransaction({ etxn: expenseData3, etxnIndex: 1 }); + component.goToTransaction({ expense: apiExpenses1[0], expenseIndex: 1 }); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'add_edit_expense', - { id: 'tx3qHxFNgRcZ', persist_filters: true }, + { id: apiExpenses1[0].id, persist_filters: true }, + ]); + }); + + it('should not navigate to any other page, if category is not present', () => { + component.goToTransaction({ expense: { ...apiExpenses1[0], category: null }, expenseIndex: 1 }); + expect(router.navigate).toHaveBeenCalledOnceWith([ + '/', + 'enterprise', + 'add_edit_expense', + { id: apiExpenses1[0].id, persist_filters: true }, ]); }); }); @@ -1888,219 +2032,182 @@ describe('MyExpensesPage', () => { spyOn(component, 'openCriticalPolicyViolationPopOver'); spyOn(component, 'showOldReportsMatBottomSheet'); spyOn(component, 'showNewReportModal'); + spyOn(component, 'unreportableExpenseExceptionHandler'); + spyOn(component, 'reportableExpenseDialogHandler'); }); - it('should call showNonReportableExpenseSelectedToast and return if selectedElement length is zero', fakeAsync(() => { - component.selectedElements = cloneDeep(apiExpenseRes); - component.selectedElements[0].tx_id = undefined; - - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - - expect(trackingService.addToReport).not.toHaveBeenCalled(); - - expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - 'Please select one or more expenses to be reported' - ); - expect(component.openCriticalPolicyViolationPopOver).not.toHaveBeenCalled(); - expect(component.showOldReportsMatBottomSheet).not.toHaveBeenCalled(); - expect(component.showNewReportModal).not.toHaveBeenCalled(); - })); + describe('when restrictPendingTransactionsEnabled is false', () => { + beforeEach(() => { + // sharedExpenseService.restrictPendingTransactionsEnabled.and.returnValues(false); + }); + it('should call showNonReportableExpenseSelectedToast and return if selectedElement length is zero', fakeAsync(() => { + const expenses = cloneDeep(apiExpenses1); + component.selectedElements = expenses.map((expense) => { + return { ...expense, id: null }; + }); + component.openCreateReportWithSelectedIds('oldReport'); + tick(100); + expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( + 'Please select one or more expenses to be reported' + ); + expect(component.openCriticalPolicyViolationPopOver).not.toHaveBeenCalled(); + expect(component.showOldReportsMatBottomSheet).not.toHaveBeenCalled(); + expect(component.showNewReportModal).not.toHaveBeenCalled(); + })); + it('should call unreportableExpenseExceptionHandler if none of the reportable expenses are selected', fakeAsync(() => { + component.selectedElements = cloneDeep(apiExpenses1); + sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(true, true); + sharedExpenseService.isExpenseInDraft.and.returnValues(false, true); + component.openCreateReportWithSelectedIds('oldReport'); + tick(100); + expect(sharedExpenseService.isCriticalPolicyViolatedExpense).toHaveBeenCalledTimes(2); + expect(sharedExpenseService.isCriticalPolicyViolatedExpense).toHaveBeenCalledWith(apiExpenses1[0]); + expect(sharedExpenseService.isCriticalPolicyViolatedExpense).toHaveBeenCalledWith(apiExpenses1[1]); - it('should call showNonReportableExpenseSelectedToast if policyViolationExpenses length is equal to selectedElements length', fakeAsync(() => { - component.selectedElements = expenseList4; - transactionService.getIsCriticalPolicyViolated.and.returnValues(true, true, true); - transactionService.getIsDraft.and.returnValues(false, false, true); + expect(sharedExpenseService.isExpenseInDraft).toHaveBeenCalledTimes(2); + expect(sharedExpenseService.isExpenseInDraft).toHaveBeenCalledWith(apiExpenses1[0]); + expect(sharedExpenseService.isExpenseInDraft).toHaveBeenCalledWith(apiExpenses1[1]); - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); + component.isReportableExpensesSelected = false; - expect(trackingService.addToReport).not.toHaveBeenCalled(); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[2]); + expect(component.unreportableExpenseExceptionHandler).toHaveBeenCalledOnceWith(1, 2, 0); + })); - expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - 'You cannot add critical policy violated expenses to a report' - ); - })); + it('should call showOldReportsMatBottomSheet if reportType is oldReport', fakeAsync(() => { + component.selectedElements = cloneDeep(apiExpenses1); + component.isReportableExpensesSelected = true; + sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(false, false); + sharedExpenseService.isExpenseInDraft.and.returnValues(false, false); + component.openCreateReportWithSelectedIds('oldReport'); + tick(100); + expect(trackingService.addToReport).toHaveBeenCalled(); + expect(component.showOldReportsMatBottomSheet).toHaveBeenCalledOnceWith(); + })); - it('should call showNonReportableExpenseSelectedToast if expensesInDraftState length is equal to selectedElements length', fakeAsync(() => { - component.selectedElements = expenseList4; - transactionService.getIsCriticalPolicyViolated.and.returnValues(false, false, true); - transactionService.getIsDraft.and.returnValues(true, true, true); + it('should call showOldReportsMatBottomSheet if reportType is newReport', fakeAsync(() => { + component.selectedElements = cloneDeep(apiExpenses1); + component.isReportableExpensesSelected = true; + sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(false, false); + sharedExpenseService.isExpenseInDraft.and.returnValues(false, false); + component.openCreateReportWithSelectedIds('newReport'); + tick(100); + expect(trackingService.addToReport).toHaveBeenCalled(); + expect(component.showNewReportModal).toHaveBeenCalledOnceWith(); + })); - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); + it('should call reportableExpenseDialogHandler if totalUnreportableCount greater than 0', fakeAsync(() => { + component.selectedElements = cloneDeep(apiExpenses1); + component.isReportableExpensesSelected = true; + sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(false, false); + sharedExpenseService.isExpenseInDraft.and.returnValues(false, true); + component.openCreateReportWithSelectedIds('newReport'); + tick(100); + expect(trackingService.addToReport).toHaveBeenCalled(); + expect(component.reportableExpenseDialogHandler).toHaveBeenCalledWith(1, 0, 0, 'newReport'); + })); + }); - expect(trackingService.addToReport).not.toHaveBeenCalled(); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[2]); + describe('when restrictPendingTransactionsEnabled is true', () => { + beforeEach(() => { + component.restrictPendingTransactionsEnabled = true; + }); + it('should call showNonReportableExpenseSelectedToast and return if selectedElement length is zero', fakeAsync(() => { + const expenses = cloneDeep(apiExpenses1); + component.selectedElements = expenses.map((expense) => { + return { ...expense, id: null }; + }); + component.openCreateReportWithSelectedIds('oldReport'); + tick(100); + expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( + 'Please select one or more expenses to be reported' + ); + expect(component.openCriticalPolicyViolationPopOver).not.toHaveBeenCalled(); + expect(component.showOldReportsMatBottomSheet).not.toHaveBeenCalled(); + expect(component.showNewReportModal).not.toHaveBeenCalled(); + })); + it('should call doesExpenseHavePendingCardTransaction', fakeAsync(() => { + component.selectedElements = cloneDeep(apiExpenses1); + sharedExpenseService.isCriticalPolicyViolatedExpense.and.returnValues(true, true); + sharedExpenseService.isExpenseInDraft.and.returnValues(false, true); + component.restrictPendingTransactionsEnabled = true; + component.openCreateReportWithSelectedIds('oldReport'); + tick(100); + expect(sharedExpenseService.doesExpenseHavePendingCardTransaction).toHaveBeenCalledTimes(2); + expect(sharedExpenseService.doesExpenseHavePendingCardTransaction).toHaveBeenCalledWith(apiExpenses1[0]); + expect(sharedExpenseService.doesExpenseHavePendingCardTransaction).toHaveBeenCalledWith(apiExpenses1[1]); + component.isReportableExpensesSelected = false; + expect(component.unreportableExpenseExceptionHandler).toHaveBeenCalledOnceWith(1, 2, 0); + })); + }); + }); + describe('unreportableExpenseExceptionHandler():', () => { + beforeEach(() => { + spyOn(component, 'showNonReportableExpenseSelectedToast'); + // sharedExpenseService.restrictPendingTransactionsEnabled.and.returnValues(true); + }); + it('should call showNonReportableExpenseSelectedToast when mix of expense types are selected', () => { + component.unreportableExpenseExceptionHandler(1, 1, 1); expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - 'You cannot add draft expenses to a report' + "You can't add draft expenses and expenses with critical policy violation & pending transactions." ); - })); - - it('should call showNonReportableExpenseSelectedToast if isReportableExpensesSelected is falsy', fakeAsync(() => { - component.isReportableExpensesSelected = false; - component.selectedElements = expenseList4; - transactionService.getIsCriticalPolicyViolated.and.returnValues(false, false, true); - transactionService.getIsDraft.and.returnValues(false, true, false); - - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - - expect(trackingService.addToReport).not.toHaveBeenCalled(); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[2]); + }); + it('should call showNonReportableExpenseSelectedToast when mix of draft and policy violation types are selected', () => { + component.unreportableExpenseExceptionHandler(1, 1, 0); expect(component.showNonReportableExpenseSelectedToast).toHaveBeenCalledOnceWith( - 'You cannot add draft expenses and critical policy violated expenses to a report' + "You can't add draft expenses & expenses with critical policy violations to a report." ); - })); - - it('should call trackingService and showOldReportsMatBottomSheet if report is oldReport and policyViolationExpenses and draftExpenses are zero', fakeAsync(() => { - component.isReportableExpensesSelected = true; - component.selectedElements = expenseList4; - transactionService.getIsCriticalPolicyViolated.and.returnValues(false, false, false); - transactionService.getIsDraft.and.returnValues(false, false, false); - - component.openCreateReportWithSelectedIds('oldReport'); - tick(100); - - expect(trackingService.addToReport).toHaveBeenCalledTimes(1); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[2]); - - expect(component.showOldReportsMatBottomSheet).toHaveBeenCalledOnceWith(); - })); - - it('should call trackingService and showNewReportModal if report is newReport and policyViolationExpenses and draftExpenses are zero', fakeAsync(() => { - component.isReportableExpensesSelected = true; - component.selectedElements = expenseList4; - transactionService.getIsCriticalPolicyViolated.and.returnValues(false, false, false); - transactionService.getIsDraft.and.returnValues(false, false, false); - - component.openCreateReportWithSelectedIds('newReport'); - tick(100); - - expect(trackingService.addToReport).toHaveBeenCalledTimes(1); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[2]); - - expect(component.showNewReportModal).toHaveBeenCalledOnceWith(); - })); - - it('should call trackingService and openCriticalPolicyViolationPopOver if policyViolationExpenses and draftExpenses are present', fakeAsync(() => { - component.isReportableExpensesSelected = true; - const mockExpenseList = cloneDeep(expenseList4); - mockExpenseList[1].tx_amount = undefined; - mockExpenseList[1].tx_admin_amount = 34; - component.selectedElements = mockExpenseList; - transactionService.getIsCriticalPolicyViolated.and.returnValues(true, true, false); - transactionService.getIsDraft.and.returnValues(false, false, true); - component.homeCurrency$ = of('USD'); - component.homeCurrencySymbol = '$'; - - component.openCreateReportWithSelectedIds('newReport'); - tick(100); - - expect(trackingService.addToReport).toHaveBeenCalledTimes(1); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(mockExpenseList[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(mockExpenseList[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(mockExpenseList[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(mockExpenseList[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(mockExpenseList[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(mockExpenseList[2]); - })); - - it('should call trackingService and openCriticalPolicyViolationPopOver if draftExpense is zero', fakeAsync(() => { - component.isReportableExpensesSelected = true; - const mockExpenseList = cloneDeep(expenseList4); - mockExpenseList[1].tx_amount = undefined; - mockExpenseList[1].tx_admin_amount = 34; - component.selectedElements = mockExpenseList; - transactionService.getIsCriticalPolicyViolated.and.returnValues(true, true, false); - transactionService.getIsDraft.and.returnValues(false, false, false); - component.homeCurrency$ = of('USD'); - component.homeCurrencySymbol = '$'; - - component.openCreateReportWithSelectedIds('newReport'); - tick(100); + }); + }); - expect(trackingService.addToReport).toHaveBeenCalledTimes(1); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(mockExpenseList[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(mockExpenseList[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(mockExpenseList[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(mockExpenseList[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(mockExpenseList[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(mockExpenseList[2]); - })); + describe('reportableExpenseDialogHandler():', () => { + beforeEach(() => { + spyOn(component, 'openCriticalPolicyViolationPopOver'); + // sharedExpenseService.restrictPendingTransactionsEnabled.and.returnValues(true); + }); + describe('reportableExpenseDialogHandler():', () => { + it('should set proper message when only draft count is greater than 0', () => { + component.reportableExpenseDialogHandler(1, 0, 0, 'newReport'); + expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ + title: "Can't add these expenses...", + message: '1 expense is in draft state.', + reportType: 'newReport', + }); + }); - it('should call trackingService and openCriticalPolicyViolationPopOver if policyViolationExpenses is zero', fakeAsync(() => { - component.isReportableExpensesSelected = true; - component.selectedElements = expenseList4; - transactionService.getIsCriticalPolicyViolated.and.returnValues(false, false, false); - transactionService.getIsDraft.and.returnValues(false, true, false); - component.homeCurrency$ = of('USD'); - component.homeCurrencySymbol = '$'; + it('should set proper message when only policy violation count is greater than 0', () => { + component.reportableExpenseDialogHandler(0, 1, 0, 'newReport'); + expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ + title: "Can't add these expenses...", + message: '1 expense with Critical Policy Violations.', + reportType: 'newReport', + }); + }); - component.openCreateReportWithSelectedIds('newReport'); - tick(100); + it('should set proper message when only pendingTransactionsCount count is greater than 0', () => { + component.reportableExpenseDialogHandler(0, 0, 1, 'newReport'); + expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ + title: "Can't add these expenses...", + message: '1 expense with pending transactions.', + reportType: 'newReport', + }); + }); - expect(trackingService.addToReport).toHaveBeenCalledTimes(1); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledTimes(3); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsCriticalPolicyViolated).toHaveBeenCalledWith(expenseList4[2]); - expect(transactionService.getIsDraft).toHaveBeenCalledTimes(3); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[0]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[1]); - expect(transactionService.getIsDraft).toHaveBeenCalledWith(expenseList4[2]); - - expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledOnceWith( - expectedCriticalPolicyViolationPopoverParams3 - ); - })); + it('should set proper message when policy violation and pendingTransactionsCount count is greater than 0', () => { + component.reportableExpenseDialogHandler(0, 1, 1, 'newReport'); + expect(component.openCriticalPolicyViolationPopOver).toHaveBeenCalledWith({ + title: "Can't add these expenses...", + message: '1 expense with pending transactions.

1 expense with Critical Policy Violations.', + reportType: 'newReport', + }); + }); + }); }); it('showNewReportModal(): should open modalController and call showAddToReportSuccessToast', fakeAsync(() => { - component.selectedElements = apiExpenseRes; - transactionService.getReportableExpenses.and.returnValue(apiExpenseRes); + component.selectedElements = apiExpenses1; + sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); const addExpenseToNewReportModalSpy = jasmine.createSpyObj('addExpenseToNewReportModal', [ 'present', 'onDidDismiss', @@ -2114,10 +2221,7 @@ describe('MyExpensesPage', () => { component.showNewReportModal(); tick(100); - - expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith(apiExpenseRes); - - expect(modalController.create).toHaveBeenCalledOnceWith(newReportModalParams); + expect(modalController.create).toHaveBeenCalledOnceWith(newReportModalParams2); expect(component.showAddToReportSuccessToast).toHaveBeenCalledOnceWith({ report: apiExtendedReportRes[0], message: 'new report is created', @@ -2133,16 +2237,17 @@ describe('MyExpensesPage', () => { }); describe('openReviewExpenses(): ', () => { - let mockExpense: Expense[]; beforeEach(() => { - component.loadData$ = new BehaviorSubject({ pageNumber: 1 }); - mockExpense = cloneDeep(apiExpenseRes); - component.selectedElements = mockExpense; - transactionService.getAllExpenses.and.returnValue(of(mockExpense)); + component.loadExpenses$ = new BehaviorSubject({ pageNumber: 1 }); + + component.selectedElements = apiExpenses1; + expensesService.getAllExpenses.and.returnValue(of(apiExpenses1)); spyOn(component, 'filterExpensesBySearchString').and.returnValue(true); + + expensesService.getExpenseById.withArgs(apiExpenses1[0].id).and.returnValue(of(apiExpenses1[0])); + expensesService.getExpenseById.withArgs(apiExpenses1[1].id).and.returnValue(of(apiExpenses1[1])); loaderService.showLoader.and.resolveTo(); loaderService.hideLoader.and.resolveTo(true); - transactionService.getETxnUnflattened.and.returnValue(of(unflattenedTxnData)); }); it('should call getAllExpenses if sortParams and sortDir is undefined in loadData$ and selectedElement length is zero', fakeAsync(() => { @@ -2150,105 +2255,96 @@ describe('MyExpensesPage', () => { component.openReviewExpenses(); tick(100); - expect(transactionService.getAllExpenses).toHaveBeenCalledOnceWith({ - queryParams: { tx_report_id: 'is.null', tx_state: 'in.(COMPLETE,DRAFT)' }, - order: null, + expect(expensesService.getAllExpenses).toHaveBeenCalledOnceWith({ + queryParams: Object({ report_id: 'is.null', state: 'in.(COMPLETE,DRAFT)' }), + order: 'spent_at.desc,created_at.desc,id.desc', }); expect(component.filterExpensesBySearchString).not.toHaveBeenCalled(); })); it('should call getAllExpenses and filterExpensesBySearchString if searchString, sortParams and sortDir are defined in loadData$ and selectedElement length is zero', fakeAsync(() => { - component.loadData$ = new BehaviorSubject({ + component.loadExpenses$ = new BehaviorSubject({ sortDir: 'asc', - sortParam: 'tx_org_category', + sortParam: 'category->name', searchString: 'example', }); component.selectedElements = []; component.openReviewExpenses(); - tick(100); - expect(transactionService.getAllExpenses).toHaveBeenCalledOnceWith({ - queryParams: { tx_report_id: 'is.null', tx_state: 'in.(COMPLETE,DRAFT)' }, - order: 'tx_org_category.asc', + expect(expensesService.getAllExpenses).toHaveBeenCalledOnceWith({ + queryParams: { report_id: 'is.null', state: 'in.(COMPLETE,DRAFT)' }, + order: 'category->name.asc', }); - expect(component.filterExpensesBySearchString).toHaveBeenCalledOnceWith(mockExpense[0], 'example'); + expect(component.filterExpensesBySearchString).toHaveBeenCalledTimes(2); + expect(component.filterExpensesBySearchString).toHaveBeenCalledWith(apiExpenses1[0], 'example'); })); it('should navigate to add_edit_mileage if org_category is mileage and selectedElement length is greater than zero', fakeAsync(() => { - const mockUnflattedData = cloneDeep(unflattenedTxnData); - mockUnflattedData.tx.org_category = 'Mileage'; - transactionService.getETxnUnflattened.and.returnValue(of(mockUnflattedData)); + component.selectedElements = [mileageExpenseWithDistance, apiExpenses1[1]]; + expensesService.getAllExpenses.and.returnValue(of([mileageExpenseWithDistance, apiExpenses1[1]])); + expensesService.getExpenseById.and.returnValue(of(mileageExpenseWithDistance)); component.openReviewExpenses(); tick(100); expect(loaderService.showLoader).toHaveBeenCalledTimes(1); - expect(transactionService.getETxnUnflattened).toHaveBeenCalledOnceWith('tx3nHShG60zq'); + expect(expensesService.getExpenseById).toHaveBeenCalledOnceWith(mileageExpenseWithDistance.id); expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'add_edit_mileage', - { - id: 'tx3qHxFNgRcZ', - txnIds: JSON.stringify(['tx3nHShG60zq']), - activeIndex: 0, - }, + { id: 'txcSFe6efB6R', txnIds: JSON.stringify(['txcSFe6efB6R', 'tx5WDG9lxBDT']), activeIndex: 0 }, ]); })); it('should navigate to add_edit_per_diem if org_category is Per Diem and selectedElement length is greater than zero', fakeAsync(() => { - const mockUnflattedData = cloneDeep(unflattenedTxnData); - mockUnflattedData.tx.org_category = 'Per Diem'; - transactionService.getETxnUnflattened.and.returnValue(of(mockUnflattedData)); + component.selectedElements = [perDiemExpenseWithSingleNumDays, apiExpenses1[1]]; + expensesService.getAllExpenses.and.returnValue(of([perDiemExpenseWithSingleNumDays, apiExpenses1[1]])); + expensesService.getExpenseById.and.returnValue(of(perDiemExpenseWithSingleNumDays)); + component.openReviewExpenses(); tick(100); expect(loaderService.showLoader).toHaveBeenCalledTimes(1); - expect(transactionService.getETxnUnflattened).toHaveBeenCalledOnceWith('tx3nHShG60zq'); + expect(expensesService.getExpenseById).toHaveBeenCalledOnceWith(perDiemExpenseWithSingleNumDays.id); expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'add_edit_per_diem', - { - id: 'tx3qHxFNgRcZ', - txnIds: JSON.stringify(['tx3nHShG60zq']), - activeIndex: 0, - }, + { id: 'txcSFe6efB6R', txnIds: JSON.stringify(['txcSFe6efB6R', 'tx5WDG9lxBDT']), activeIndex: 0 }, ]); })); it('should navigate to add_edit_expense if org_category is not amongst mileage and per diem and selectedElement length is greater than zero', fakeAsync(() => { - transactionService.getETxnUnflattened.and.returnValue(of(unflattenedTxnData)); + component.selectedElements = apiExpenses1; + expensesService.getAllExpenses.and.returnValue(of(apiExpenses1)); + expensesService.getExpenseById.and.returnValue(of(apiExpenses1[0])); component.openReviewExpenses(); tick(100); expect(loaderService.showLoader).toHaveBeenCalledTimes(1); - expect(transactionService.getETxnUnflattened).toHaveBeenCalledOnceWith('tx3nHShG60zq'); + expect(expensesService.getExpenseById).toHaveBeenCalledOnceWith(apiExpenses1[0].id); expect(loaderService.hideLoader).toHaveBeenCalledTimes(1); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'add_edit_expense', - { - id: 'tx3qHxFNgRcZ', - txnIds: JSON.stringify(['tx3nHShG60zq']), - activeIndex: 0, - }, + { id: 'txDDLtRaflUW', txnIds: '["txDDLtRaflUW","tx5WDG9lxBDT"]', activeIndex: 0 }, ]); })); }); describe('filterExpensesBySearchString(): ', () => { it('should return true if expense consist of searchString', () => { - const expectedFilteredExpenseRes = component.filterExpensesBySearchString(expenseData1, 'Groc'); + const expectedFilteredExpenseRes = component.filterExpensesBySearchString(expenseData, 'usvKA4X8Ugcr'); expect(expectedFilteredExpenseRes).toBeTrue(); }); it('should return false if expense does not consist of searchString', () => { - const expectedFilteredExpenseRes = component.filterExpensesBySearchString(expenseData1, 'Software'); + const expectedFilteredExpenseRes = component.filterExpensesBySearchString(expenseData, 'Software'); expect(expectedFilteredExpenseRes).toBeFalse(); }); @@ -2264,7 +2360,7 @@ describe('MyExpensesPage', () => { component.onAddTransactionToReport({ tx_id: '12345' }); tick(100); - expect(modalController.create).toHaveBeenCalledOnceWith(addExpenseToReportModalParams); + expect(modalController.create).toHaveBeenCalledOnceWith(addExpenseToReportModalParams2); expect(component.doRefresh).toHaveBeenCalledTimes(1); })); @@ -2362,15 +2458,16 @@ describe('MyExpensesPage', () => { describe('showOldReportsMatBottomSheet(): ', () => { beforeEach(() => { - component.selectedElements = apiExpenseRes; + component.selectedElements = apiExpenses1; component.isNewReportsFlowEnabled = true; component.openReports$ = of(apiExtendedReportRes); - transactionService.getReportableExpenses.and.returnValue(apiExpenseRes); + sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); spyOn(component, 'showAddToReportSuccessToast'); }); it('should call matBottomSheet.open and call showAddToReportSuccessToast if data.report is defined', () => { spyOn(component, 'addTransactionsToReport').and.returnValue(of(apiExtendedReportRes[0])); + matBottomsheet.open.and.returnValue({ afterDismissed: () => of({ @@ -2380,12 +2477,14 @@ describe('MyExpensesPage', () => { component.showOldReportsMatBottomSheet(); - expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith(apiExpenseRes); expect(matBottomsheet.open).toHaveBeenCalledOnceWith(AddTxnToReportDialogComponent, { data: { openReports: apiExtendedReportRes, isNewReportsFlowEnabled: true }, panelClass: ['mat-bottom-sheet-1'], }); - expect(component.addTransactionsToReport).toHaveBeenCalledOnceWith(apiExtendedReportRes[0], ['tx3nHShG60zq']); + expect(component.addTransactionsToReport).toHaveBeenCalledOnceWith(apiExtendedReportRes[0], [ + 'txDDLtRaflUW', + 'tx5WDG9lxBDT', + ]); expect(component.showAddToReportSuccessToast).toHaveBeenCalledOnceWith({ message: 'Expenses added to report successfully', report: apiExtendedReportRes[0], @@ -2405,13 +2504,15 @@ describe('MyExpensesPage', () => { } as MatBottomSheetRef); component.showOldReportsMatBottomSheet(); - expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith(apiExpenseRes); expect(matBottomsheet.open).toHaveBeenCalledOnceWith(AddTxnToReportDialogComponent, { data: { openReports: mockReportData, isNewReportsFlowEnabled: true }, panelClass: ['mat-bottom-sheet-1'], }); - expect(component.addTransactionsToReport).toHaveBeenCalledOnceWith(mockReportData[0], ['tx3nHShG60zq']); + expect(component.addTransactionsToReport).toHaveBeenCalledOnceWith(mockReportData[0], [ + 'txDDLtRaflUW', + 'tx5WDG9lxBDT', + ]); expect(component.showAddToReportSuccessToast).toHaveBeenCalledOnceWith({ message: 'Expenses added to an existing draft report', report: mockReportData[0], @@ -2428,7 +2529,6 @@ describe('MyExpensesPage', () => { } as MatBottomSheetRef); component.showOldReportsMatBottomSheet(); - expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith(apiExpenseRes); expect(matBottomsheet.open).toHaveBeenCalledOnceWith(AddTxnToReportDialogComponent, { data: { openReports: apiExtendedReportRes, isNewReportsFlowEnabled: true }, panelClass: ['mat-bottom-sheet-1'], @@ -2458,39 +2558,49 @@ describe('MyExpensesPage', () => { describe('deleteSelectedExpenses(): ', () => { beforeEach(() => { component.pendingTransactions = []; - component.expensesToBeDeleted = expenseList4; + component.expensesToBeDeleted = apiExpenses1; }); it('should update selectedElements and call deleteBulk method if expenseToBeDeleted is defined', () => { component.deleteSelectedExpenses([]); - expect(transactionOutboxService.deleteBulkOfflineExpenses).toHaveBeenCalledOnceWith([], []); - expect(component.selectedElements).toEqual(expenseList4); - expect(transactionService.deleteBulk).toHaveBeenCalledOnceWith(['txKFqMRPNLsa', 'txc5zbIpTGMU', 'txo3tuIb7em4']); + expect(transactionOutboxService.deleteBulkOfflineExpenses).not.toHaveBeenCalledOnceWith([], []); + expect(component.selectedElements).toEqual(apiExpenses1); + expect(transactionService.deleteBulk).toHaveBeenCalledOnceWith(['txDDLtRaflUW', 'tx5WDG9lxBDT']); }); + it('should not call deleteBulk method if tx_id is not present in expensesToBeDeleted', () => { - const mockExpensesWithoutId = cloneDeep(apiExpenseRes); - mockExpensesWithoutId[0].tx_id = undefined; + const mockExpensesWithoutId = cloneDeep([apiExpenses1[0]]); + mockExpensesWithoutId[0].id = undefined; component.expensesToBeDeleted = mockExpensesWithoutId; - component.deleteSelectedExpenses([]); - expect(transactionOutboxService.deleteBulkOfflineExpenses).toHaveBeenCalledOnceWith([], []); + component.deleteSelectedExpenses(null); + expect(transactionOutboxService.deleteBulkOfflineExpenses).not.toHaveBeenCalledOnceWith([], []); expect(component.selectedElements).toEqual([]); expect(transactionService.deleteBulk).not.toHaveBeenCalled(); }); + + it('should delete outbox expenses', () => { + component.deleteSelectedExpenses(expenseList4); + + expect(transactionOutboxService.deleteBulkOfflineExpenses).toHaveBeenCalledOnceWith( + component.pendingTransactions, + expenseList4 + ); + }); }); describe('openDeleteExpensesPopover(): ', () => { beforeEach(() => { - transactionService.getExpenseDeletionMessage.and.returnValue('You are about to delete this expense'); - transactionService.getCCCExpenseMessage.and.returnValue( + sharedExpenseService.getExpenseDeletionMessage.and.returnValue('You are about to delete this expense'); + sharedExpenseService.getCCCExpenseMessage.and.returnValue( 'There are 2 corporate credit cards which can be deleted' ); - transactionService.getDeleteDialogBody.and.returnValue('Once deleted, the action cannot be undone'); - component.expensesToBeDeleted = apiExpenseRes; + sharedExpenseService.getDeleteDialogBody.and.returnValue('Once deleted, the action cannot be undone'); + component.expensesToBeDeleted = apiExpenses1; component.cccExpenses = 1; transactionService.deleteBulk.and.returnValue(of(txnList)); snackbarProperties.setSnackbarProperties.and.returnValue(snackbarPropertiesRes3); spyOn(component, 'doRefresh'); - component.expensesToBeDeleted = cloneDeep(expenseList4); - component.selectedElements = cloneDeep(expenseList4); + component.expensesToBeDeleted = cloneDeep(apiExpenses1); + component.selectedElements = cloneDeep(apiExpenses1); }); it('should open a popover and get data of expenses on dismiss', fakeAsync(() => { @@ -2521,6 +2631,7 @@ describe('MyExpensesPage', () => { popoverController.create.and.resolveTo(deletePopOverSpy); component.cccExpenses = 0; component.expensesToBeDeleted = []; + spyOn(component, 'deleteSelectedExpenses').and.callThrough(); component.openDeleteExpensesPopover(); @@ -2540,6 +2651,29 @@ describe('MyExpensesPage', () => { }); })); + it('should open a popover and delete offline expenses', fakeAsync(() => { + component.outboxExpensesToBeDeleted = expenseListwithoutID; + const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); + deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); + popoverController.create.and.resolveTo(deletePopOverSpy); + + component.openDeleteExpensesPopover(); + tick(1000); + + expect(popoverController.create).toHaveBeenCalledOnceWith({ + component: FyDeleteDialogComponent, + cssClass: 'delete-dialog', + backdropDismiss: false, + componentProps: { + header: 'Delete Expense', + body: 'Once deleted, the action cannot be undone', + ctaText: 'Exclude and Delete', + disableDelete: false, + deleteMethod: jasmine.any(Function), + }, + }); + })); + it('should show message using matSnackbar if data is successfully deleted and selectedElements are greater than 1', fakeAsync(() => { const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); @@ -2553,10 +2687,10 @@ describe('MyExpensesPage', () => { panelClass: ['msb-success-with-camera-icon'], }); expect(snackbarProperties.setSnackbarProperties).toHaveBeenCalledOnceWith('success', { - message: '3 expenses have been deleted', + message: '2 expenses have been deleted', }); expect(trackingService.showToastMessage).toHaveBeenCalledOnceWith({ - ToastContent: '3 expenses have been deleted', + ToastContent: '2 expenses have been deleted', }); expect(component.isReportableExpensesSelected).toBeFalse(); expect(component.selectionMode).toBeFalse(); @@ -2568,9 +2702,8 @@ describe('MyExpensesPage', () => { const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'success' } }); popoverController.create.and.resolveTo(deletePopOverSpy); - const mockExpenseList = cloneDeep(expenseList4); - component.expensesToBeDeleted = cloneDeep(mockExpenseList); - component.selectedElements = cloneDeep([mockExpenseList[0]]); + component.expensesToBeDeleted = cloneDeep(apiExpenses1); + component.selectedElements = cloneDeep([apiExpenses1[0]]); component.openDeleteExpensesPopover(); tick(100); @@ -2594,9 +2727,8 @@ describe('MyExpensesPage', () => { const deletePopOverSpy = jasmine.createSpyObj('deletePopover', ['present', 'onDidDismiss']); deletePopOverSpy.onDidDismiss.and.resolveTo({ data: { status: 'failure' } }); popoverController.create.and.resolveTo(deletePopOverSpy); - const mockExpenseList = cloneDeep(expenseList4); - component.expensesToBeDeleted = cloneDeep(mockExpenseList); - component.selectedElements = cloneDeep([mockExpenseList[0]]); + component.expensesToBeDeleted = cloneDeep(apiExpenses1); + component.selectedElements = cloneDeep([apiExpenses1[0]]); component.openDeleteExpensesPopover(); tick(100); @@ -2620,45 +2752,51 @@ describe('MyExpensesPage', () => { describe('onSelectAll(): ', () => { beforeEach(() => { - transactionService.getAllExpenses.and.returnValue(of(cloneDeep(apiExpenseRes))); - transactionService.excludeCCCExpenses.and.returnValue(apiExpenseRes); - transactionService.getReportableExpenses.and.returnValue(apiExpenseRes); - apiV2Service.extendQueryParamsForTextSearch.and.returnValue({ - tx_report_id: 'is.null', - tx_state: 'in.(COMPLETE,DRAFT)', - }); + expensesService.getAllExpenses.and.returnValue(of(cloneDeep(apiExpenses1))); + sharedExpenseService.excludeCCCExpenses.and.returnValue(apiExpenses1); + sharedExpenseService.getReportableExpenses.and.returnValue(apiExpenses1); spyOn(component, 'setExpenseStatsOnSelect'); - component.loadData$ = new BehaviorSubject({ pageNumber: 1 }); + component.loadExpenses$ = new BehaviorSubject({ pageNumber: 1, searchString: 'Bus' }); }); it('should set selectedElement to empty array if checked is false', () => { - component.selectedElements = cloneDeep(apiExpenseRes); + component.selectedElements = cloneDeep(apiExpenses1); component.isReportableExpensesSelected = false; component.onSelectAll(false); expect(component.selectedElements).toEqual([]); - expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith([]); + expect(sharedExpenseService.getReportableExpenses).toHaveBeenCalledOnceWith([]); expect(component.isReportableExpensesSelected).toBeTrue(); expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); }); + it('should select all pending transactions and update stats', () => { + component.pendingTransactions = expenseList4; + transactionService.getReportableExpenses.and.returnValue(expenseList4); + spyOn(component, 'setOutboxExpenseStatsOnSelect'); + + component.onSelectAll(true); + + expect(transactionService.getReportableExpenses).toHaveBeenCalledOnceWith(expenseList4); + expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); + expect(component.isReportableExpensesSelected).toBeTrue(); + }); + it('should update selectedElements, allExpensesCount and call apiV2Service if checked is true', () => { - transactionService.getAllExpenses.and.returnValue(of(cloneDeep(expenseList4))); - component.pendingTransactions = cloneDeep(apiExpenseRes); + expensesService.getAllExpenses.and.returnValue(of(cloneDeep(apiExpenses1))); + component.outboxExpensesToBeDeleted = apiExpenseRes; + component.pendingTransactions = cloneDeep([]); component.onSelectAll(true); expect(component.isReportableExpensesSelected).toBeTrue(); - expect(apiV2Service.extendQueryParamsForTextSearch).toHaveBeenCalledOnceWith( - { tx_report_id: 'is.null', tx_state: 'in.(COMPLETE,DRAFT)' }, - undefined - ); - expect(transactionService.getAllExpenses).toHaveBeenCalledOnceWith({ - queryParams: { tx_report_id: 'is.null', tx_state: 'in.(COMPLETE,DRAFT)' }, + + expect(expensesService.getAllExpenses).toHaveBeenCalledOnceWith({ + queryParams: { report_id: 'is.null', state: 'in.(COMPLETE,DRAFT)', q: 'Bus:*' }, }); - expect(transactionService.excludeCCCExpenses).toHaveBeenCalledOnceWith([...apiExpenseRes, ...expenseList4]); - expect(component.cccExpenses).toBe(3); - expect(component.selectedElements).toEqual([...apiExpenseRes, ...expenseList4]); - expect(component.allExpensesCount).toBe(4); + expect(sharedExpenseService.excludeCCCExpenses).toHaveBeenCalledOnceWith(apiExpenses1); + expect(component.cccExpenses).toBe(0); + expect(component.selectedElements).toEqual([...apiExpenses1]); + expect(component.allExpensesCount).toBe(2); expect(component.isReportableExpensesSelected).toBeTrue(); - expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(2); + expect(component.setExpenseStatsOnSelect).toHaveBeenCalledTimes(1); }); }); @@ -2694,7 +2832,7 @@ describe('MyExpensesPage', () => { describe('onFilterClose(): ', () => { beforeEach(() => { - component.loadData$ = new BehaviorSubject({}); + component.loadExpenses$ = new BehaviorSubject({}); component.filters = { sortDir: 'asc', sortParam: 'tx_org_category', @@ -2713,7 +2851,7 @@ describe('MyExpensesPage', () => { expect(component.filters.sortParam).toBeUndefined(); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((data) => { + component.loadExpenses$.subscribe((data) => { expect(data).toEqual({ pageNumber: 3 }); }); expect(component.filterPills).toEqual(creditTxnFilterPill); @@ -2726,7 +2864,7 @@ describe('MyExpensesPage', () => { }); expect(component.currentPageNumber).toBe(1); expect(component.addNewFiltersToParams).toHaveBeenCalledTimes(1); - component.loadData$.subscribe((data) => { + component.loadExpenses$.subscribe((data) => { expect(data).toEqual({ pageNumber: 3 }); }); expect(component.filterPills).toEqual(creditTxnFilterPill); @@ -2778,15 +2916,14 @@ describe('MyExpensesPage', () => { })); it('mergeExpense(): should navigate to merge_expenses with payload data', () => { - component.selectedElements = apiExpenseRes; - + component.selectedElements = apiExpenses1; component.mergeExpenses(); expect(router.navigate).toHaveBeenCalledOnceWith([ '/', 'enterprise', 'merge_expense', { - expenseIDs: JSON.stringify(['tx3nHShG60zq']), + expenseIDs: JSON.stringify(['txDDLtRaflUW', 'tx5WDG9lxBDT']), from: 'MY_EXPENSES', }, ]); @@ -2805,4 +2942,129 @@ describe('MyExpensesPage', () => { expect(component.isCameraPreviewStarted).toBeTrue(); }); }); + + it('setOutboxExpenseStatsOnSelect(): should update stats on selecting outbox expenses', (done) => { + component.selectedOutboxExpenses = expenseList4; + + component.setOutboxExpenseStatsOnSelect(); + + component.allExpensesStats$.subscribe((res) => { + expect(res).toEqual({ + count: 3, + amount: 49475.76, + }); + done(); + }); + }); + + describe('selectOutboxExpense(): ', () => { + beforeEach(() => { + transactionService.getReportableExpenses.and.returnValue(apiExpenseRes); + component.allExpensesCount = 1; + spyOn(component, 'setExpenseStatsOnSelect'); + spyOn(component, 'setOutboxExpenseStatsOnSelect'); + component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); + transactionService.isMergeAllowed.and.returnValue(true); + transactionService.excludeCCCExpenses.and.returnValue(apiExpenseRes); + }); + + it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses', () => { + transactionService.getReportableExpenses.and.returnValue([]); + const expense = apiExpenseRes[0]; + component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); + + component.selectOutboxExpense(expense); + + expect(component.selectedOutboxExpenses).toEqual([]); + expect(component.isReportableExpensesSelected).toBeFalse(); + expect(component.selectAll).toBeFalse(); + expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); + expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); + expect(component.isMergeAllowed).toBeTrue(); + }); + + it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses', () => { + transactionService.getReportableExpenses.and.returnValue([]); + component.allExpensesCount = 4; + const expense = apiExpenseRes[0]; + component.selectedOutboxExpenses = cloneDeep(cloneDeep(expenseList4)); + + component.selectOutboxExpense(expense); + + expect(component.selectedOutboxExpenses).toEqual([...expenseList4, expense]); + expect(component.isReportableExpensesSelected).toBeFalse(); + expect(component.selectAll).toBeTrue(); + expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); + expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([...expenseList4, expense]); + expect(component.isMergeAllowed).toBeTrue(); + }); + + it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses and allExpenseCount is not equal to length of selectedOutboxExpenses', () => { + transactionService.getReportableExpenses.and.returnValue([]); + const expense = apiExpenseRes[0]; + component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); + + component.selectOutboxExpense(expense); + + expect(component.selectedOutboxExpenses).toEqual([]); + expect(component.isReportableExpensesSelected).toBeFalse(); + expect(component.selectAll).toBeFalse(); + expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); + expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); + expect(component.isMergeAllowed).toBeTrue(); + }); + + it('should update expenseToBeDeleted if selectedOutboxExpenses is an array of atleast 1', () => { + component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); + component.selectOutboxExpense(expenseData2); + + const expectedSelectedElements = [...apiExpenseRes, expenseData2]; + expect(component.selectedOutboxExpenses).toEqual(expectedSelectedElements); + expect(component.outboxExpensesToBeDeleted).toEqual(apiExpenseRes); + expect(component.cccExpenses).toBe(1); + expect(component.selectAll).toBeFalse(); + }); + + it('should remove an expense from selectedOutboxExpenses if it is present in selectedOutboxExpenses and tx_id is not present in expense', () => { + transactionService.getReportableExpenses.and.returnValue([]); + component.allExpensesCount = 0; + const expense = cloneDeep(apiExpenseRes[0]); + expense.tx_id = undefined; + component.selectedOutboxExpenses = cloneDeep(apiExpenseRes); + component.selectedOutboxExpenses[0].tx_id = undefined; + + component.selectOutboxExpense(expense); + + expect(component.selectedOutboxExpenses).toEqual([]); + expect(component.isReportableExpensesSelected).toBeFalse(); + expect(component.selectAll).toBeTrue(); + expect(component.setOutboxExpenseStatsOnSelect).toHaveBeenCalledTimes(1); + expect(transactionService.isMergeAllowed).toHaveBeenCalledOnceWith([]); + expect(component.isMergeAllowed).toBeTrue(); + }); + }); + + describe('checkDeleteDisabled():', () => { + it('should check and enable the button for online mode', (done) => { + component.isConnected$ = of(true); + component.selectedElements = apiExpenses1; + component.expensesToBeDeleted = []; + + component.checkDeleteDisabled().subscribe(() => { + expect(component.isDisabled).toBeFalse(); + done(); + }); + }); + + it('should check and enable the button for offline mode', (done) => { + component.isConnected$ = of(false); + component.selectedOutboxExpenses = apiExpenseRes; + component.outboxExpensesToBeDeleted = []; + + component.checkDeleteDisabled().subscribe(() => { + expect(component.isDisabled).toBeFalse(); + done(); + }); + }); + }); }); diff --git a/src/app/fyle/my-expenses/my-expenses.page.ts b/src/app/fyle/my-expenses/my-expenses.page.ts index b1b250f820..a426e94b9c 100644 --- a/src/app/fyle/my-expenses/my-expenses.page.ts +++ b/src/app/fyle/my-expenses/my-expenses.page.ts @@ -1,84 +1,82 @@ +import { getCurrencySymbol } from '@angular/common'; import { Component, ElementRef, EventEmitter, OnInit, ViewChild } from '@angular/core'; +import { MatBottomSheet } from '@angular/material/bottom-sheet'; +import { MatSnackBar } from '@angular/material/snack-bar'; +import { ActivatedRoute, Params, Router } from '@angular/router'; +import { ActionSheetController, ModalController, NavController, PopoverController } from '@ionic/angular'; +import { cloneDeep, isEqual } from 'lodash'; import { BehaviorSubject, + Observable, + Subject, + Subscription, concat, - EMPTY, forkJoin, from, fromEvent, iif, noop, - Observable, of, - Subject, - Subscription, } from 'rxjs'; -import { NetworkService } from 'src/app/core/services/network.service'; -import { LoaderService } from 'src/app/core/services/loader.service'; -import { ActionSheetController, ModalController, PopoverController, NavController } from '@ionic/angular'; -import { ActivatedRoute, Params, Router } from '@angular/router'; import { - catchError, debounceTime, distinctUntilChanged, + filter, finalize, map, shareReplay, switchMap, take, takeUntil, - tap, - filter, } from 'rxjs/operators'; -import { TransactionService } from 'src/app/core/services/transaction.service'; +import { BackButtonActionPriority } from 'src/app/core/models/back-button-action-priority.enum'; +import { CardAggregateStats } from 'src/app/core/models/card-aggregate-stats.model'; import { Expense } from 'src/app/core/models/expense.model'; -import { TransactionsOutboxService } from 'src/app/core/services/transactions-outbox.service'; -import { PopupService } from 'src/app/core/services/popup.service'; -import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog/add-txn-to-report-dialog.component'; -import { TrackingService } from '../../core/services/tracking.service'; -import { StorageService } from '../../core/services/storage.service'; -import { ModalPropertiesService } from 'src/app/core/services/modal-properties.service'; -import { ReportService } from 'src/app/core/services/report.service'; -import { cloneDeep, isEqual } from 'lodash'; -import { CreateNewReportComponent } from 'src/app/shared/components/create-new-report/create-new-report.component'; -import { PopupAlertComponent } from 'src/app/shared/components/popup-alert/popup-alert.component'; -import { MatBottomSheet } from '@angular/material/bottom-sheet'; -import { MatSnackBar } from '@angular/material/snack-bar'; +import { OrgSettings } from 'src/app/core/models/org-settings.model'; +import { ExpenseFilters } from 'src/app/core/models/platform/expense-filters.model'; +import { PlatformCategory } from 'src/app/core/models/platform/platform-category.model'; +import { Expense as PlatformExpense } from 'src/app/core/models/platform/v1/expense.model'; +import { GetExpenseQueryParam } from 'src/app/core/models/platform/v1/get-expenses-query.model'; +import { ReportV1 } from 'src/app/core/models/report-v1.model'; import { ExtendedReport } from 'src/app/core/models/report.model'; -import { ToastMessageComponent } from 'src/app/shared/components/toast-message/toast-message.component'; -import { TokenService } from 'src/app/core/services/token.service'; +import { UniqueCardStats } from 'src/app/core/models/unique-cards-stats.model'; +import { UniqueCards } from 'src/app/core/models/unique-cards.model'; +import { Transaction } from 'src/app/core/models/v1/transaction.model'; import { ApiV2Service } from 'src/app/core/services/api-v2.service'; -import { environment } from 'src/environments/environment'; -import { HeaderState } from '../../shared/components/fy-header/header-state.enum'; -import { FyDeleteDialogComponent } from '../../shared/components/fy-delete-dialog/fy-delete-dialog.component'; -import { FyFiltersComponent } from '../../shared/components/fy-filters/fy-filters.component'; -import { FilterOptions } from '../../shared/components/fy-filters/filter-options.interface'; -import { FilterOptionType } from '../../shared/components/fy-filters/filter-option-type.enum'; -import { FilterPill } from '../../shared/components/fy-filter-pills/filter-pill.interface'; -import { getCurrencySymbol } from '@angular/common'; -import { SnackbarPropertiesService } from '../../core/services/snackbar-properties.service'; -import { TasksService } from 'src/app/core/services/tasks.service'; +import { CategoriesService } from 'src/app/core/services/categories.service'; import { CorporateCreditCardExpenseService } from 'src/app/core/services/corporate-credit-card-expense.service'; -import { MaskNumber } from 'src/app/shared/pipes/mask-number.pipe'; -import { MyExpensesService } from './my-expenses.service'; -import { ExpenseFilters } from './expense-filters.model'; import { CurrencyService } from 'src/app/core/services/currency.service'; +import { LoaderService } from 'src/app/core/services/loader.service'; +import { ModalPropertiesService } from 'src/app/core/services/modal-properties.service'; +import { NetworkService } from 'src/app/core/services/network.service'; import { OrgSettingsService } from 'src/app/core/services/org-settings.service'; import { OrgUserSettingsService } from 'src/app/core/services/org-user-settings.service'; -import { BackButtonActionPriority } from 'src/app/core/models/back-button-action-priority.enum'; import { PlatformHandlerService } from 'src/app/core/services/platform-handler.service'; -import { OrgSettings } from 'src/app/core/models/org-settings.model'; -import { GetExpensesQueryParamsWithFilters } from 'src/app/core/models/get-expenses-query-params-with-filters.model'; -import { Transaction } from 'src/app/core/models/v1/transaction.model'; -import { Datum } from 'src/app/core/models/v2/stats-response.model'; -import { CardAggregateStats } from 'src/app/core/models/card-aggregate-stats.model'; -import { UniqueCardStats } from 'src/app/core/models/unique-cards-stats.model'; -import { FilterQueryParams } from 'src/app/core/models/filter-query-params.model'; +import { ExpensesService as SharedExpenseService } from 'src/app/core/services/platform/v1/shared/expenses.service'; +import { ExpensesService } from 'src/app/core/services/platform/v1/spender/expenses.service'; +import { PopupService } from 'src/app/core/services/popup.service'; +import { ReportService } from 'src/app/core/services/report.service'; +import { TasksService } from 'src/app/core/services/tasks.service'; +import { TokenService } from 'src/app/core/services/token.service'; +import { TransactionService } from 'src/app/core/services/transaction.service'; +import { TransactionsOutboxService } from 'src/app/core/services/transactions-outbox.service'; +import { CreateNewReportComponent } from 'src/app/shared/components/create-new-report-v2/create-new-report.component'; import { SelectedFilters } from 'src/app/shared/components/fy-filters/selected-filters.interface'; -import { UniqueCards } from 'src/app/core/models/unique-cards.model'; -import { CategoriesService } from 'src/app/core/services/categories.service'; -import { PlatformCategory } from 'src/app/core/models/platform/platform-category.model'; -import { ReportV1 } from 'src/app/core/models/report-v1.model'; +import { PopupAlertComponent } from 'src/app/shared/components/popup-alert/popup-alert.component'; +import { ToastMessageComponent } from 'src/app/shared/components/toast-message/toast-message.component'; +import { MaskNumber } from 'src/app/shared/pipes/mask-number.pipe'; +import { environment } from 'src/environments/environment'; +import { SnackbarPropertiesService } from '../../core/services/snackbar-properties.service'; +import { StorageService } from '../../core/services/storage.service'; +import { TrackingService } from '../../core/services/tracking.service'; +import { FyDeleteDialogComponent } from '../../shared/components/fy-delete-dialog/fy-delete-dialog.component'; +import { FilterPill } from '../../shared/components/fy-filter-pills/filter-pill.interface'; +import { FilterOptionType } from '../../shared/components/fy-filters/filter-option-type.enum'; +import { FilterOptions } from '../../shared/components/fy-filters/filter-options.interface'; +import { FyFiltersComponent } from '../../shared/components/fy-filters/fy-filters.component'; +import { HeaderState } from '../../shared/components/fy-header/header-state.enum'; +import { AddTxnToReportDialogComponent } from './add-txn-to-report-dialog/add-txn-to-report-dialog.component'; +import { MyExpensesService } from './my-expenses.service'; @Component({ selector: 'app-my-expenses', @@ -90,17 +88,17 @@ export class MyExpensesPage implements OnInit { isConnected$: Observable; - myExpenses$: Observable; + myExpenses$: Observable; count$: Observable; isInfiniteScrollRequired$: Observable; - loadData$: BehaviorSubject>; + loadExpenses$: BehaviorSubject>; currentPageNumber = 1; - acc: Expense[] = []; + acc: PlatformExpense[] = []; filters: Partial; @@ -126,7 +124,9 @@ export class MyExpensesPage implements OnInit { selectionMode = false; - selectedElements: Partial[]; + selectedElements: PlatformExpense[]; + + selectedOutboxExpenses: Partial[] = []; syncing = false; @@ -178,7 +178,9 @@ export class MyExpensesPage implements OnInit { maskNumber = new MaskNumber(); - expensesToBeDeleted: Partial[]; + expensesToBeDeleted: PlatformExpense[]; + + outboxExpensesToBeDeleted: Partial[] = []; cccExpenses: number; @@ -188,6 +190,10 @@ export class MyExpensesPage implements OnInit { isNewReportsFlowEnabled = false; + isDisabled = false; + + restrictPendingTransactionsEnabled = false; + constructor( private networkService: NetworkService, private loaderService: LoaderService, @@ -216,7 +222,9 @@ export class MyExpensesPage implements OnInit { private orgUserSettingsService: OrgUserSettingsService, private platformHandlerService: PlatformHandlerService, private categoriesService: CategoriesService, - private navController: NavController + private navController: NavController, + private expenseService: ExpensesService, + private sharedExpenseService: SharedExpenseService ) {} get HeaderState(): typeof HeaderState { @@ -253,10 +261,10 @@ export class MyExpensesPage implements OnInit { }); } - switchSelectionMode(expense?: Expense): void { + switchSelectionMode(expense?: PlatformExpense): void { this.selectionMode = !this.selectionMode; if (!this.selectionMode) { - if (this.loadData$.getValue().searchString) { + if (this.loadExpenses$.getValue().searchString) { this.headerState = HeaderState.simpleSearch; } else { this.headerState = HeaderState.base; @@ -278,6 +286,31 @@ export class MyExpensesPage implements OnInit { } } + switchOutboxSelectionMode(expense?: Expense): void { + this.selectionMode = !this.selectionMode; + if (!this.selectionMode) { + if (this.loadExpenses$.getValue().searchString) { + this.headerState = HeaderState.simpleSearch; + } else { + this.headerState = HeaderState.base; + } + + this.selectedOutboxExpenses = []; + this.setOutboxExpenseStatsOnSelect(); + } else { + this.headerState = HeaderState.multiselect; + // setting Expense amount & count stats to zero on select init + this.allExpensesStats$ = of({ + count: 0, + amount: 0, + }); + } + + if (expense) { + this.selectOutboxExpense(expense); + } + } + async sendFirstExpenseCreatedEvent(): Promise { // checking if the expense is first expense const isFirstExpenseCreated = await this.storageService.get('isFirstExpenseCreated'); @@ -294,37 +327,27 @@ export class MyExpensesPage implements OnInit { } setAllExpensesCountAndAmount(): void { - this.allExpensesStats$ = this.loadData$.pipe( + this.allExpensesStats$ = this.loadExpenses$.pipe( switchMap((params) => { - const queryParams: FilterQueryParams = - (JSON.parse(JSON.stringify(params.queryParams)) as FilterQueryParams) || {}; + const queryParams = cloneDeep(params.queryParams) || {}; - queryParams.tx_report_id = queryParams.tx_report_id || 'is.null'; - queryParams.tx_state = 'in.(COMPLETE,DRAFT)'; + queryParams.report_id = (queryParams.report_id || 'is.null') as string; + queryParams.state = 'in.(COMPLETE,DRAFT)'; - if (queryParams.corporate_credit_card_account_number) { - const cardParamsCopy = JSON.parse(JSON.stringify(queryParams.corporate_credit_card_account_number)) as string; - queryParams.or = queryParams.or || []; - queryParams.or.push('(corporate_credit_card_account_number.' + cardParamsCopy + ')'); - delete queryParams.corporate_credit_card_account_number; + if (queryParams['matched_corporate_card_transactions->0->corporate_card_number']) { + const cardParamsCopy = queryParams['matched_corporate_card_transactions->0->corporate_card_number'] as string; + + queryParams.or = (queryParams.or || []) as string[]; + queryParams.or.push('(matched_corporate_card_transactions->0->corporate_card_number.' + cardParamsCopy + ')'); + delete queryParams['matched_corporate_card_transactions->0->corporate_card_number']; } - return this.transactionService - .getTransactionStats('count(tx_id),sum(tx_amount)', { - scalar: true, - ...queryParams, - }) - .pipe( - catchError(() => EMPTY), - map((stats: Datum[]) => { - const count = stats[0].aggregates.find((stat) => stat.function_name === 'count(tx_id)'); - const amount = stats[0].aggregates.find((stat) => stat.function_name === 'sum(tx_amount)'); - return { - count: count.function_value, - amount: amount.function_value || 0, - }; - }) - ); + return this.expenseService.getExpenseStats(queryParams).pipe( + map((stats) => ({ + count: stats.data.count, + amount: stats.data.total_amount, + })) + ); }) ); } @@ -444,6 +467,10 @@ export class MyExpensesPage implements OnInit { this.orgSettings$.subscribe((orgSettings) => { this.isNewReportsFlowEnabled = orgSettings?.simplified_report_closure_settings?.enabled || false; + this.restrictPendingTransactionsEnabled = + (orgSettings?.corporate_credit_card_settings?.enabled && + orgSettings?.pending_cct_expense_restriction?.enabled) || + false; }); forkJoin({ @@ -487,7 +514,8 @@ export class MyExpensesPage implements OnInit { this.simpleSearchText = ''; this.currentPageNumber = 1; - this.loadData$ = new BehaviorSubject({ + + this.loadExpenses$ = new BehaviorSubject({ pageNumber: 1, }); @@ -518,105 +546,102 @@ export class MyExpensesPage implements OnInit { debounceTime(400) ) .subscribe((searchString) => { - const currentParams = this.loadData$.getValue(); + const currentParams = this.loadExpenses$.getValue(); currentParams.searchString = searchString; this.currentPageNumber = 1; currentParams.pageNumber = this.currentPageNumber; - this.loadData$.next(currentParams); + + this.loadExpenses$.next(currentParams); }); - const paginatedPipe = this.loadData$.pipe( + const paginatedPipe = this.loadExpenses$.pipe( switchMap((params) => { - let queryParams = params.queryParams || {}; + const queryParams = params.queryParams || {}; + + queryParams.report_id = queryParams.report_id || 'is.null'; + queryParams.state = 'in.(COMPLETE,DRAFT)'; - queryParams.tx_report_id = queryParams.tx_report_id || 'is.null'; - queryParams.tx_state = 'in.(COMPLETE,DRAFT)'; - queryParams = this.apiV2Service.extendQueryParamsForTextSearch(queryParams, params.searchString); - const orderByParams = params.sortParam && params.sortDir ? `${params.sortParam}.${params.sortDir}` : null; + if (params.searchString) { + queryParams.q = params.searchString; + queryParams.q = queryParams.q + ':*'; + } else if (params.searchString === '') { + delete queryParams.q; + } + const orderByParams = + params.sortParam && params.sortDir + ? `${params.sortParam}.${params.sortDir}` + : 'spent_at.desc,created_at.desc,id.desc'; this.isLoadingDataInInfiniteScroll = true; - return this.transactionService.getMyExpensesCount(queryParams).pipe( + + return this.expenseService.getExpensesCount(queryParams).pipe( switchMap((count) => { if (count > (params.pageNumber - 1) * 10) { - return this.transactionService.getMyExpenses({ + return this.expenseService.getExpenses({ offset: (params.pageNumber - 1) * 10, limit: 10, - queryParams, + ...queryParams, order: orderByParams, }); } else { - return of({ - data: [], - }); + return of([]); + } + }), + map((res) => { + this.isLoadingDataInInfiniteScroll = false; + if (this.currentPageNumber === 1) { + this.acc = []; } + this.acc = this.acc.concat(res as PlatformExpense[]); + return this.acc; }) ); - }), - map((res) => { - this.isLoadingDataInInfiniteScroll = false; - if (this.currentPageNumber === 1) { - this.acc = []; - } - this.acc = this.acc.concat(res.data); - return this.acc; - }), - tap(() => { - this.pendingTransactions = this.formatTransactions(this.transactionOutboxService.getPendingTransactions()); }) ); this.myExpenses$ = paginatedPipe.pipe(shareReplay(1)); - this.count$ = this.loadData$.pipe( + this.count$ = this.loadExpenses$.pipe( switchMap((params) => { - let queryParams = params.queryParams || {}; + const queryParams = params.queryParams || {}; - queryParams.tx_report_id = queryParams.tx_report_id || 'is.null'; - queryParams.tx_state = 'in.(COMPLETE,DRAFT)'; - queryParams = this.apiV2Service.extendQueryParamsForTextSearch(queryParams, params.searchString); - return this.transactionService.getMyExpensesCount(queryParams); + queryParams.report_id = queryParams.report_id || 'is.null'; + queryParams.state = 'in.(COMPLETE,DRAFT)'; + return this.expenseService.getExpensesCount(queryParams); }), shareReplay(1) ); - this.isNewUser$ = this.transactionService.getPaginatedETxncCount().pipe(map((res) => res.count === 0)); + this.isNewUser$ = this.expenseService.getExpensesCount({ offset: 0, limit: 200 }).pipe(map((res) => res === 0)); const paginatedScroll$ = this.myExpenses$.pipe( switchMap((etxns) => this.count$.pipe(map((count) => count > etxns.length))) ); - this.isInfiniteScrollRequired$ = this.loadData$.pipe(switchMap(() => paginatedScroll$)); + this.isInfiniteScrollRequired$ = this.loadExpenses$.pipe(switchMap(() => paginatedScroll$)); this.setAllExpensesCountAndAmount(); - this.allExpenseCountHeader$ = this.loadData$.pipe( + this.allExpenseCountHeader$ = this.loadExpenses$.pipe( switchMap(() => - this.transactionService.getTransactionStats('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_state: 'in.(COMPLETE,DRAFT)', - tx_report_id: 'is.null', + this.expenseService.getExpenseStats({ + state: 'in.(COMPLETE,DRAFT)', + report_id: 'is.null', }) ), - map((stats) => { - const count = stats && stats[0] && stats[0].aggregates.find((stat) => stat.function_name === 'count(tx_id)'); - return count && count.function_value; - }) + map((stats) => stats.data.count) ); - this.draftExpensesCount$ = this.loadData$.pipe( + this.draftExpensesCount$ = this.loadExpenses$.pipe( switchMap(() => - this.transactionService.getTransactionStats('count(tx_id),sum(tx_amount)', { - scalar: true, - tx_report_id: 'is.null', - tx_state: 'in.(DRAFT)', + this.expenseService.getExpenseStats({ + state: 'in.(DRAFT)', + report_id: 'is.null', }) ), - map((stats) => { - const count = stats && stats[0] && stats[0].aggregates.find((stat) => stat.function_name === 'count(tx_id)'); - return count && count.function_value; - }) + map((stats) => stats.data.count) ); - this.loadData$.subscribe(() => { + this.loadExpenses$.subscribe(() => { const queryParams: Params = { filters: JSON.stringify(this.filters) }; this.router.navigate([], { relativeTo: this.activatedRoute, @@ -636,25 +661,27 @@ export class MyExpensesPage implements OnInit { ); this.currentPageNumber = 1; const params = this.addNewFiltersToParams(); - this.loadData$.next(params); + + this.loadExpenses$.next(params); this.filterPills = this.generateFilterPills(this.filters); } else if (this.activatedRoute.snapshot.params.state) { let filters = {}; if ((this.activatedRoute.snapshot.params.state as string).toLowerCase() === 'needsreceipt') { - filters = { tx_receipt_required: 'eq.true', state: 'NEEDS_RECEIPT' }; + filters = { is_receipt_mandotary: 'eq.true', state: 'NEEDS_RECEIPT' }; } else if ((this.activatedRoute.snapshot.params.state as string).toLowerCase() === 'policyviolated') { filters = { - tx_policy_flag: 'eq.true', - or: '(tx_policy_amount.is.null,tx_policy_amount.gt.0.0001)', + is_policy_flagged: 'eq.true', + or: '(policy_amount.is.null,policy_amount.gt.0.0001)', state: 'POLICY_VIOLATED', }; } else if ((this.activatedRoute.snapshot.params.state as string).toLowerCase() === 'cannotreport') { - filters = { tx_policy_amount: 'lt.0.0001', state: 'CANNOT_REPORT' }; + filters = { policy_amount: 'lt.0.0001', state: 'CANNOT_REPORT' }; } this.filters = Object.assign({}, this.filters, filters); this.currentPageNumber = 1; const params = this.addNewFiltersToParams(); - this.loadData$.next(params); + + this.loadExpenses$.next(params); this.filterPills = this.generateFilterPills(this.filters); } else { this.clearFilters(); @@ -679,6 +706,8 @@ export class MyExpensesPage implements OnInit { ) ); this.doRefresh(); + + this.checkDeleteDisabled(); } setupNetworkWatcher(): void { @@ -690,9 +719,9 @@ export class MyExpensesPage implements OnInit { loadData(event: { target?: { complete?: () => void } }): void { this.currentPageNumber = this.currentPageNumber + 1; - const params = this.loadData$.getValue(); + const params = this.loadExpenses$.getValue(); params.pageNumber = this.currentPageNumber; - this.loadData$.next(params); + this.loadExpenses$.next(params); setTimeout(() => { event?.target?.complete(); @@ -724,10 +753,10 @@ export class MyExpensesPage implements OnInit { if (this.selectionMode) { this.setExpenseStatsOnSelect(); } - const params = this.loadData$.getValue(); + const params = this.loadExpenses$.getValue(); params.pageNumber = this.currentPageNumber; this.transactionService.clearCache().subscribe(() => { - this.loadData$.next(params); + this.loadExpenses$.next(params); if (event) { setTimeout(() => { event.target?.complete(); @@ -770,26 +799,26 @@ export class MyExpensesPage implements OnInit { return filterPills; } - addNewFiltersToParams(): Partial { - let currentParams = this.loadData$.getValue(); + addNewFiltersToParams(): Partial { + let currentParams = this.loadExpenses$.getValue(); currentParams.pageNumber = 1; - let newQueryParams: FilterQueryParams = { + let newQueryParams: Record = { or: [], }; - newQueryParams = this.transactionService.generateCardNumberParams(newQueryParams, this.filters); + newQueryParams = this.sharedExpenseService.generateCardNumberParams(newQueryParams, this.filters); - newQueryParams = this.transactionService.generateDateParams(newQueryParams, this.filters); + newQueryParams = this.sharedExpenseService.generateDateParams(newQueryParams, this.filters); - newQueryParams = this.transactionService.generateReceiptAttachedParams(newQueryParams, this.filters); + newQueryParams = this.sharedExpenseService.generateReceiptAttachedParams(newQueryParams, this.filters); - newQueryParams = this.transactionService.generateStateFilters(newQueryParams, this.filters); + newQueryParams = this.sharedExpenseService.generateStateFilters(newQueryParams, this.filters); - newQueryParams = this.transactionService.generateTypeFilters(newQueryParams, this.filters); + newQueryParams = this.sharedExpenseService.generateTypeFilters(newQueryParams, this.filters); - currentParams = this.transactionService.setSortParams(currentParams, this.filters); + currentParams = this.sharedExpenseService.setSortParams(currentParams, this.filters); - newQueryParams = this.transactionService.generateSplitExpenseParams(newQueryParams, this.filters); + newQueryParams = this.sharedExpenseService.generateSplitExpenseParams(newQueryParams, this.filters); currentParams.queryParams = newQueryParams; @@ -820,11 +849,13 @@ export class MyExpensesPage implements OnInit { } as FilterOptions); } + const selectedFilters = this.myExpensesService.generateSelectedFilters(this.filters); + const filterPopover = await this.modalController.create({ component: FyFiltersComponent, componentProps: { filterOptions: filterMain, - selectedFilterValues: this.myExpensesService.generateSelectedFilters(this.filters), + selectedFilterValues: selectedFilters, activeFilterInitialName, }, cssClass: 'dialog-popover', @@ -833,11 +864,16 @@ export class MyExpensesPage implements OnInit { await filterPopover.present(); const { data } = (await filterPopover.onWillDismiss()) as { data: SelectedFilters[] }; + if (data) { - this.filters = this.myExpensesService.convertFilters(data); + const filters1 = this.myExpensesService.convertSelectedOptionsToExpenseFilters(data); + this.filters = filters1; + this.currentPageNumber = 1; const params = this.addNewFiltersToParams(); - this.loadData$.next(params); + + this.loadExpenses$.next(params); + this.filterPills = this.generateFilterPills(this.filters); this.trackingService.myExpensesFilterApplied({ ...this.filters, @@ -849,7 +885,7 @@ export class MyExpensesPage implements OnInit { this.filters = {}; this.currentPageNumber = 1; const params = this.addNewFiltersToParams(); - this.loadData$.next(params); + this.loadExpenses$.next(params); this.filterPills = this.generateFilterPills(this.filters); } @@ -857,7 +893,7 @@ export class MyExpensesPage implements OnInit { this.isLoading = true; this.currentPageNumber = 1; const params = this.addNewFiltersToParams(); - this.loadData$.next(params); + this.loadExpenses$.next(params); setTimeout(() => { this.isLoading = false; }, 500); @@ -866,31 +902,76 @@ export class MyExpensesPage implements OnInit { setExpenseStatsOnSelect(): void { this.allExpensesStats$ = of({ count: this.selectedElements.length, - amount: this.selectedElements.reduce((acc, txnObj) => acc + txnObj.tx_amount, 0), + amount: this.selectedElements.reduce((acc, txnObj) => acc + txnObj.amount, 0), }); } - selectExpense(expense: Expense): void { + setOutboxExpenseStatsOnSelect(): void { + this.allExpensesStats$ = of({ + count: this.selectedOutboxExpenses.length, + amount: this.selectedOutboxExpenses.reduce((acc, txnObj) => acc + txnObj.tx_amount, 0), + }); + } + + selectOutboxExpense(expense: Expense): void { let isSelectedElementsIncludesExpense = false; if (expense.tx_id) { - isSelectedElementsIncludesExpense = this.selectedElements.some((txn) => expense.tx_id === txn.tx_id); + isSelectedElementsIncludesExpense = this.selectedOutboxExpenses.some((txn) => expense.tx_id === txn.tx_id); } else { - isSelectedElementsIncludesExpense = this.selectedElements.some((txn) => isEqual(txn, expense)); + isSelectedElementsIncludesExpense = this.selectedOutboxExpenses.some((txn) => isEqual(txn, expense)); } if (isSelectedElementsIncludesExpense) { if (expense.tx_id) { - this.selectedElements = this.selectedElements.filter((txn) => txn.tx_id !== expense.tx_id); + this.selectedOutboxExpenses = this.selectedOutboxExpenses.filter((txn) => txn.tx_id !== expense.tx_id); + } else { + this.selectedOutboxExpenses = this.selectedOutboxExpenses.filter((txn) => !isEqual(txn, expense)); + } + } else { + this.selectedOutboxExpenses.push(expense); + } + this.isReportableExpensesSelected = + this.transactionService.getReportableExpenses(this.selectedOutboxExpenses).length > 0; + + if (this.selectedOutboxExpenses.length > 0) { + this.outboxExpensesToBeDeleted = this.transactionService.excludeCCCExpenses(this.selectedOutboxExpenses); + + this.cccExpenses = this.selectedOutboxExpenses.length - this.outboxExpensesToBeDeleted.length; + } + + // setting Expenses count and amount stats on select + if (this.allExpensesCount === this.selectedOutboxExpenses.length) { + this.selectAll = true; + } else { + this.selectAll = false; + } + this.setOutboxExpenseStatsOnSelect(); + this.isMergeAllowed = this.transactionService.isMergeAllowed(this.selectedOutboxExpenses); + } + + selectExpense(expense: PlatformExpense): void { + let isSelectedElementsIncludesExpense = false; + if (expense.id) { + isSelectedElementsIncludesExpense = this.selectedElements.some((txn) => expense.id === txn.id); + } else { + isSelectedElementsIncludesExpense = this.selectedElements.some((txn) => isEqual(txn, expense)); + } + + if (isSelectedElementsIncludesExpense) { + if (expense.id) { + this.selectedElements = this.selectedElements.filter((txn) => txn.id !== expense.id); } else { this.selectedElements = this.selectedElements.filter((txn) => !isEqual(txn, expense)); } } else { this.selectedElements.push(expense); } - this.isReportableExpensesSelected = this.transactionService.getReportableExpenses(this.selectedElements).length > 0; + this.isReportableExpensesSelected = + this.sharedExpenseService.getReportableExpenses(this.selectedElements, this.restrictPendingTransactionsEnabled) + .length > 0; if (this.selectedElements.length > 0) { - this.expensesToBeDeleted = this.transactionService.excludeCCCExpenses(this.selectedElements); + this.expensesToBeDeleted = this.sharedExpenseService.excludeCCCExpenses(this.selectedElements); this.cccExpenses = this.selectedElements.length - this.expensesToBeDeleted.length; } @@ -902,22 +983,22 @@ export class MyExpensesPage implements OnInit { this.selectAll = false; } this.setExpenseStatsOnSelect(); - this.isMergeAllowed = this.transactionService.isMergeAllowed(this.selectedElements); + this.isMergeAllowed = this.sharedExpenseService.isMergeAllowed(this.selectedElements); } - goToTransaction({ etxn: expense }: { etxn: Expense; etxnIndex: number }): void { + goToTransaction(event: { expense: PlatformExpense; expenseIndex: number }): void { let category: string; - if (expense.tx_org_category) { - category = expense.tx_org_category.toLowerCase(); + if (event.expense?.category?.name) { + category = event.expense.category.name.toLowerCase(); } if (category === 'mileage') { - this.router.navigate(['/', 'enterprise', 'add_edit_mileage', { id: expense.tx_id, persist_filters: true }]); + this.router.navigate(['/', 'enterprise', 'add_edit_mileage', { id: event.expense.id, persist_filters: true }]); } else if (category === 'per diem') { - this.router.navigate(['/', 'enterprise', 'add_edit_per_diem', { id: expense.tx_id, persist_filters: true }]); + this.router.navigate(['/', 'enterprise', 'add_edit_per_diem', { id: event.expense.id, persist_filters: true }]); } else { - this.router.navigate(['/', 'enterprise', 'add_edit_expense', { id: expense.tx_id, persist_filters: true }]); + this.router.navigate(['/', 'enterprise', 'add_edit_expense', { id: event.expense.id, persist_filters: true }]); } } @@ -966,61 +1047,116 @@ export class MyExpensesPage implements OnInit { this.trackingService.showToastMessage({ ToastContent: message }); } + isSelectionContainsException( + policyViolationsCount: number, + draftCount: number, + pendingTransactionsCount: number + ): boolean { + return ( + policyViolationsCount > 0 || + draftCount > 0 || + (this.restrictPendingTransactionsEnabled && pendingTransactionsCount > 0) + ); + } + + unreportableExpenseExceptionHandler( + draftCount: number, + policyViolationsCount: number, + pendingTransactionsCount: number + ): void { + // This Map contains different messages based on different conditions , the first character in map key is draft, second is policy violation, third is pending transactions + // draft, policy, pending + const toastMessage = new Map([ + ['111', "You can't add draft expenses and expenses with critical policy violation & pending transactions."], + ['110', "You can't add draft expenses & expenses with critical policy violations to a report."], + ['101', "You can't add draft expenses & expenses with pending transactions to a report."], + ['011', "You can't add expenses with critical policy violation & pending transactions to a report."], + ['100', "You can't add draft expenses to a report."], + ['010', "You can't add expenses with critical policy violations to a report."], + ['001', "You can't add expenses with pending transactions to a report."], + ]); + const messageConfig = `${draftCount > 0 ? 1 : 0}${policyViolationsCount > 0 ? 1 : 0}${ + pendingTransactionsCount > 0 ? 1 : 0 + }`; + + if (toastMessage.has(messageConfig)) { + this.showNonReportableExpenseSelectedToast(toastMessage.get(messageConfig)); + } + if (pendingTransactionsCount > 0) { + this.trackingService.spenderSelectedPendingTxnFromMyExpenses(); + } + } + + reportableExpenseDialogHandler( + draftCount: number, + policyViolationsCount: number, + pendingTransactionsCount: number, + reportType: 'oldReport' | 'newReport' + ): void { + const title = "Can't add these expenses..."; + let message = ''; + + if (draftCount > 0) { + message += `${draftCount} ${draftCount > 1 ? 'expenses are' : 'expense is'} in draft state.`; + } + if (pendingTransactionsCount > 0) { + message += `${message.length ? '

' : ''}${pendingTransactionsCount} ${ + pendingTransactionsCount > 1 ? 'expenses' : 'expense' + } with pending transactions.`; + } + if (policyViolationsCount > 0) { + message += `${message.length ? '

' : ''}${policyViolationsCount} ${ + policyViolationsCount > 1 ? 'expenses' : 'expense' + } with Critical Policy Violations.`; + } + + this.openCriticalPolicyViolationPopOver({ title, message, reportType }); + } + async openCreateReportWithSelectedIds(reportType: 'oldReport' | 'newReport'): Promise { let selectedElements = cloneDeep(this.selectedElements); // Removing offline expenses from the list - selectedElements = selectedElements.filter((expense) => expense.tx_id); + selectedElements = selectedElements.filter((expense) => expense.id); if (!selectedElements.length) { this.showNonReportableExpenseSelectedToast('Please select one or more expenses to be reported'); return; } const expensesWithCriticalPolicyViolations = selectedElements.filter((expense) => - this.transactionService.getIsCriticalPolicyViolated(expense) + this.sharedExpenseService.isCriticalPolicyViolatedExpense(expense) + ); + const expensesInDraftState = selectedElements.filter((expense) => + this.sharedExpenseService.isExpenseInDraft(expense) ); - const expensesInDraftState = selectedElements.filter((expense) => this.transactionService.getIsDraft(expense)); + let expensesWithPendingTransactions = []; + //only handle pending txns if it is enabled from settings + if (this.restrictPendingTransactionsEnabled) { + expensesWithPendingTransactions = selectedElements.filter((expense) => + this.sharedExpenseService.doesExpenseHavePendingCardTransaction(expense) + ); + } const noOfExpensesWithCriticalPolicyViolations = expensesWithCriticalPolicyViolations.length; const noOfExpensesInDraftState = expensesInDraftState.length; + const noOfExpensesWithPendingTransactions = expensesWithPendingTransactions.length; - if (noOfExpensesWithCriticalPolicyViolations === selectedElements.length) { - this.showNonReportableExpenseSelectedToast('You cannot add critical policy violated expenses to a report'); - } else if (noOfExpensesInDraftState === selectedElements.length) { - this.showNonReportableExpenseSelectedToast('You cannot add draft expenses to a report'); - } else if (!this.isReportableExpensesSelected) { - this.showNonReportableExpenseSelectedToast( - 'You cannot add draft expenses and critical policy violated expenses to a report' + if (!this.isReportableExpensesSelected) { + this.unreportableExpenseExceptionHandler( + noOfExpensesInDraftState, + noOfExpensesWithCriticalPolicyViolations, + noOfExpensesWithPendingTransactions ); } else { this.trackingService.addToReport(); - const totalAmountofCriticalPolicyViolationExpenses = expensesWithCriticalPolicyViolations.reduce( - (prev, current) => { - const amount = current.tx_amount || current.tx_user_amount; - return prev + amount; - }, - 0 - ); - - let title = ''; - let message = ''; - - if (noOfExpensesWithCriticalPolicyViolations > 0 || noOfExpensesInDraftState > 0) { - this.homeCurrency$.subscribe(() => { - if (noOfExpensesWithCriticalPolicyViolations > 0 && noOfExpensesInDraftState > 0) { - title = `${noOfExpensesWithCriticalPolicyViolations} Critical Policy and \ - ${noOfExpensesInDraftState} Draft Expenses blocking the way`; - message = `Critical policy blocking these ${noOfExpensesWithCriticalPolicyViolations} expenses worth \ - ${this.homeCurrencySymbol}${totalAmountofCriticalPolicyViolationExpenses} from being submitted. \ - Also ${noOfExpensesInDraftState} other expenses are in draft states.`; - } else if (noOfExpensesWithCriticalPolicyViolations > 0) { - title = `${noOfExpensesWithCriticalPolicyViolations} Critical Policy Expenses blocking the way`; - message = `Critical policy blocking these ${noOfExpensesWithCriticalPolicyViolations} expenses worth \ - ${this.homeCurrencySymbol}${totalAmountofCriticalPolicyViolationExpenses} from being submitted.`; - } else if (noOfExpensesInDraftState > 0) { - title = `${noOfExpensesInDraftState} Draft Expenses blocking the way`; - message = `${noOfExpensesInDraftState} expenses are in draft states.`; - } - this.openCriticalPolicyViolationPopOver({ title, message, reportType }); - }); + const totalUnreportableCount = + noOfExpensesInDraftState + noOfExpensesWithCriticalPolicyViolations + noOfExpensesWithPendingTransactions; + + if (totalUnreportableCount > 0) { + this.reportableExpenseDialogHandler( + noOfExpensesInDraftState, + noOfExpensesWithCriticalPolicyViolations, + noOfExpensesWithPendingTransactions, + reportType + ); } else { if (reportType === 'oldReport') { this.showOldReportsMatBottomSheet(); @@ -1032,7 +1168,10 @@ export class MyExpensesPage implements OnInit { } async showNewReportModal(): Promise { - const reportAbleExpenses = this.transactionService.getReportableExpenses(this.selectedElements); + const reportAbleExpenses = this.sharedExpenseService.getReportableExpenses( + this.selectedElements, + this.restrictPendingTransactionsEnabled + ); const addExpenseToNewReportModal = await this.modalController.create({ component: CreateNewReportComponent, componentProps: { @@ -1058,18 +1197,21 @@ export class MyExpensesPage implements OnInit { } openReviewExpenses(): void { - const allDataPipe$ = this.loadData$.pipe( + const allDataPipe$ = this.loadExpenses$.pipe( take(1), switchMap((params) => { const queryParams = params.queryParams || {}; - queryParams.tx_report_id = queryParams.tx_report_id || 'is.null'; + queryParams.report_id = queryParams.report_id || 'is.null'; - queryParams.tx_state = 'in.(COMPLETE,DRAFT)'; + queryParams.state = 'in.(COMPLETE,DRAFT)'; - const orderByParams = params.sortParam && params.sortDir ? `${params.sortParam}.${params.sortDir}` : null; + const orderByParams = + params.sortParam && params.sortDir + ? `${params.sortParam}.${params.sortDir}` + : 'spent_at.desc,created_at.desc,id.desc'; - return this.transactionService + return this.expenseService .getAllExpenses({ queryParams, order: orderByParams, @@ -1086,21 +1228,21 @@ export class MyExpensesPage implements OnInit { ) ); }), - map((etxns) => etxns.map((etxn) => etxn.tx_id)) + map((etxns) => etxns.map((etxn) => etxn.id)) ); from(this.loaderService.showLoader()) .pipe( switchMap(() => { - const txnIds = this.selectedElements.map((expense) => expense.tx_id); + const txnIds = this.selectedElements.map((expense) => expense.id); return iif(() => this.selectedElements.length === 0, allDataPipe$, of(txnIds)); }), switchMap((selectedIds) => { const initial = selectedIds[0]; const allIds = selectedIds; - return this.transactionService.getETxnUnflattened(initial).pipe( - map((etxn) => ({ - inital: etxn, + return this.expenseService.getExpenseById(initial).pipe( + map((expense) => ({ + inital: expense, allIds, })) ); @@ -1110,28 +1252,28 @@ export class MyExpensesPage implements OnInit { .subscribe(({ inital, allIds }) => { let category: string; - if (inital.tx.org_category) { - category = inital.tx.org_category.toLowerCase(); + if (inital.category.name) { + category = inital.category.name.toLowerCase(); } - if (category === 'mileage') { + if (category.includes('mileage')) { this.router.navigate([ '/', 'enterprise', 'add_edit_mileage', { - id: inital.tx.id, + id: inital.id, txnIds: JSON.stringify(allIds), activeIndex: 0, }, ]); - } else if (category === 'per diem') { + } else if (category.includes('per diem')) { this.router.navigate([ '/', 'enterprise', 'add_edit_per_diem', { - id: inital.tx.id, + id: inital.id, txnIds: JSON.stringify(allIds), activeIndex: 0, }, @@ -1142,7 +1284,7 @@ export class MyExpensesPage implements OnInit { 'enterprise', 'add_edit_expense', { - id: inital.tx.id, + id: inital.id, txnIds: JSON.stringify(allIds), activeIndex: 0, }, @@ -1151,9 +1293,9 @@ export class MyExpensesPage implements OnInit { }); } - filterExpensesBySearchString(expense: Expense, searchString: string): boolean { + filterExpensesBySearchString(expense: PlatformExpense, searchString: string): boolean { return Object.values(expense) - .map((value: keyof Expense) => value && value.toString().toLowerCase()) + .map((value: keyof PlatformExpense) => value && value.toString().toLowerCase()) .filter((value) => !!value) .some((value) => value.toLowerCase().includes(searchString.toLowerCase())); } @@ -1206,8 +1348,11 @@ export class MyExpensesPage implements OnInit { } showOldReportsMatBottomSheet(): void { - const reportAbleExpenses = this.transactionService.getReportableExpenses(this.selectedElements); - const selectedExpensesId = reportAbleExpenses.map((expenses) => expenses.tx_id); + const reportAbleExpenses = this.sharedExpenseService.getReportableExpenses( + this.selectedElements, + this.restrictPendingTransactionsEnabled + ); + const selectedExpensesId = reportAbleExpenses.map((expenses) => expenses.id); this.openReports$ .pipe( @@ -1251,22 +1396,35 @@ export class MyExpensesPage implements OnInit { } deleteSelectedExpenses(offlineExpenses: Partial[]): Observable { - this.transactionOutboxService.deleteBulkOfflineExpenses(this.pendingTransactions, offlineExpenses); - - this.selectedElements = this.expensesToBeDeleted.filter((expense) => expense.tx_id); - if (this.selectedElements.length > 0) { - return this.transactionService.deleteBulk(this.selectedElements.map((selectedExpense) => selectedExpense.tx_id)); - } else { + if (offlineExpenses?.length > 0) { + this.transactionOutboxService.deleteBulkOfflineExpenses(this.pendingTransactions, offlineExpenses); return of(null); + } else { + this.selectedElements = this.expensesToBeDeleted.filter((expense) => expense.id); + if (this.selectedElements.length > 0) { + return this.transactionService.deleteBulk(this.selectedElements.map((selectedExpense) => selectedExpense.id)); + } else { + return of(null); + } } } async openDeleteExpensesPopover(): Promise { - const offlineExpenses = this.expensesToBeDeleted.filter((expense) => !expense.tx_id); + const offlineExpenses = this.outboxExpensesToBeDeleted.filter((expense) => !expense.tx_id); - const expenseDeletionMessage = this.transactionService.getExpenseDeletionMessage(this.expensesToBeDeleted); + let expenseDeletionMessage: string; + let cccExpensesMessage: string; + let totalDeleteLength = 0; - const cccExpensesMessage = this.transactionService.getCCCExpenseMessage(this.expensesToBeDeleted, this.cccExpenses); + if (offlineExpenses.length > 0) { + expenseDeletionMessage = this.transactionService.getExpenseDeletionMessage(offlineExpenses); + cccExpensesMessage = this.transactionService.getCCCExpenseMessage(offlineExpenses, this.cccExpenses); + totalDeleteLength = this.outboxExpensesToBeDeleted.length; + } else { + expenseDeletionMessage = this.sharedExpenseService.getExpenseDeletionMessage(this.expensesToBeDeleted); + cccExpensesMessage = this.sharedExpenseService.getCCCExpenseMessage(this.expensesToBeDeleted, this.cccExpenses); + totalDeleteLength = this.expensesToBeDeleted?.length; + } const deletePopover = await this.popoverController.create({ component: FyDeleteDialogComponent, @@ -1274,14 +1432,14 @@ export class MyExpensesPage implements OnInit { backdropDismiss: false, componentProps: { header: 'Delete Expense', - body: this.transactionService.getDeleteDialogBody( - this.expensesToBeDeleted, + body: this.sharedExpenseService.getDeleteDialogBody( + totalDeleteLength, this.cccExpenses, expenseDeletionMessage, cccExpensesMessage ), - ctaText: this.expensesToBeDeleted.length > 0 && this.cccExpenses > 0 ? 'Exclude and Delete' : 'Delete', - disableDelete: this.expensesToBeDeleted.length > 0 ? false : true, + ctaText: totalDeleteLength > 0 && this.cccExpenses > 0 ? 'Exclude and Delete' : 'Delete', + disableDelete: totalDeleteLength === 0 ? true : false, deleteMethod: () => this.deleteSelectedExpenses(offlineExpenses), }, }); @@ -1294,8 +1452,15 @@ export class MyExpensesPage implements OnInit { this.trackingService.myExpensesBulkDeleteExpenses({ count: this.selectedElements.length, }); + if (data.status === 'success') { - const totalNoOfSelectedExpenses = offlineExpenses.length + this.selectedElements.length; + let totalNoOfSelectedExpenses = 0; + if (offlineExpenses?.length > 0) { + totalNoOfSelectedExpenses = offlineExpenses.length + this.selectedElements.length; + } else { + totalNoOfSelectedExpenses = this.selectedElements.length; + } + const message = totalNoOfSelectedExpenses === 1 ? '1 expense has been deleted' @@ -1326,42 +1491,48 @@ export class MyExpensesPage implements OnInit { if (checked) { this.selectedElements = []; if (this.pendingTransactions.length > 0) { - this.selectedElements = this.pendingTransactions; - this.allExpensesCount = this.selectedElements.length; + this.selectedOutboxExpenses = this.pendingTransactions; + this.allExpensesCount = this.pendingTransactions.length; this.isReportableExpensesSelected = - this.transactionService.getReportableExpenses(this.selectedElements).length > 0; - this.setExpenseStatsOnSelect(); + this.transactionService.getReportableExpenses(this.selectedOutboxExpenses).length > 0; + this.outboxExpensesToBeDeleted = this.selectedOutboxExpenses; + this.setOutboxExpenseStatsOnSelect(); + } else { + this.loadExpenses$ + .pipe( + take(1), + map((params) => { + const queryParams = params.queryParams || {}; + + queryParams.report_id = queryParams.report_id || 'is.null'; + queryParams.state = 'in.(COMPLETE,DRAFT)'; + if (params.searchString) { + queryParams.q = params?.searchString + ':*'; + } + + return queryParams; + }), + switchMap((queryParams) => this.expenseService.getAllExpenses({ queryParams })) + ) + .subscribe((allExpenses) => { + this.selectedElements = this.selectedElements.concat(allExpenses); + if (this.selectedElements.length > 0) { + this.expensesToBeDeleted = this.sharedExpenseService.excludeCCCExpenses(this.selectedElements); + + this.cccExpenses = this.selectedElements.length - this.expensesToBeDeleted.length; + } + this.allExpensesCount = this.selectedElements.length; + this.isReportableExpensesSelected = + this.sharedExpenseService.getReportableExpenses(this.selectedElements).length > 0; + this.setExpenseStatsOnSelect(); + }); } - - this.loadData$ - .pipe( - take(1), - map((params) => { - let queryParams = params.queryParams || {}; - - queryParams.tx_report_id = queryParams.tx_report_id || 'is.null'; - queryParams.tx_state = 'in.(COMPLETE,DRAFT)'; - queryParams = this.apiV2Service.extendQueryParamsForTextSearch(queryParams, params.searchString); - return queryParams; - }), - switchMap((queryParams) => this.transactionService.getAllExpenses({ queryParams })) - ) - .subscribe((allExpenses) => { - this.selectedElements = this.selectedElements.concat(allExpenses); - if (this.selectedElements.length > 0) { - this.expensesToBeDeleted = this.transactionService.excludeCCCExpenses(this.selectedElements); - - this.cccExpenses = this.selectedElements.length - this.expensesToBeDeleted.length; - } - this.allExpensesCount = this.selectedElements.length; - this.isReportableExpensesSelected = - this.transactionService.getReportableExpenses(this.selectedElements).length > 0; - this.setExpenseStatsOnSelect(); - }); } else { this.selectedElements = []; + this.selectedOutboxExpenses = []; + this.outboxExpensesToBeDeleted = []; this.isReportableExpensesSelected = - this.transactionService.getReportableExpenses(this.selectedElements).length > 0; + this.sharedExpenseService.getReportableExpenses(this.selectedElements).length > 0; this.setExpenseStatsOnSelect(); } } @@ -1400,7 +1571,7 @@ export class MyExpensesPage implements OnInit { } this.currentPageNumber = 1; const params = this.addNewFiltersToParams(); - this.loadData$.next(params); + this.loadExpenses$.next(params); this.filterPills = this.generateFilterPills(this.filters); } @@ -1446,7 +1617,7 @@ export class MyExpensesPage implements OnInit { } mergeExpenses(): void { - const expenseIDs = this.selectedElements.map((expense) => expense.tx_id); + const expenseIDs = this.selectedElements.map((expense) => expense.id); this.router.navigate([ '/', 'enterprise', @@ -1461,4 +1632,19 @@ export class MyExpensesPage implements OnInit { showCamera(isCameraPreviewStarted: boolean): void { this.isCameraPreviewStarted = isCameraPreviewStarted; } + + checkDeleteDisabled(): Observable { + return this.isConnected$.pipe( + map((isConnected) => { + if (isConnected) { + this.isDisabled = + this.selectedElements?.length === 0 || + !this.expensesToBeDeleted || + (this.expensesToBeDeleted?.length === 0 && this.cccExpenses > 0); + } else if (!isConnected) { + this.isDisabled = this.selectedOutboxExpenses.length === 0 || !this.outboxExpensesToBeDeleted; + } + }) + ); + } } diff --git a/src/app/fyle/my-expenses/my-expenses.service.spec.ts b/src/app/fyle/my-expenses/my-expenses.service.spec.ts index a87412d739..522b6e28de 100644 --- a/src/app/fyle/my-expenses/my-expenses.service.spec.ts +++ b/src/app/fyle/my-expenses/my-expenses.service.spec.ts @@ -2,13 +2,10 @@ import { TestBed } from '@angular/core/testing'; import { MyExpensesService } from './my-expenses.service'; import { expenseFiltersData1, - expenseFiltersData1Old, expenseFiltersData3, expenseFiltersData4, expenseFiltersData5, - expenseFiltersData5Old, expenseFiltersData6, - expenseFiltersData7, } from 'src/app/core/mock-data/expense-filters.data'; import { cardFilterPill, @@ -37,7 +34,8 @@ import { expectedFilterPill9, } from 'src/app/core/mock-data/my-reports-filterpills.data'; import { filter1, filter2 } from 'src/app/core/mock-data/my-reports-filters.data'; -import { filterOptions2, filterOptions3 } from 'src/app/core/mock-data/filter-options.data'; +import { filterOptions2 } from 'src/app/core/mock-data/filter-options.data'; +import { ExpenseType } from 'src/app/core/enums/expense-type.enum'; describe('MyExpensesService', () => { let myExpensesService: MyExpensesService; @@ -73,7 +71,7 @@ describe('MyExpensesService', () => { spyOn(myExpensesService, 'convertSelectedSortFitlersToFilters'); const sortBy = { name: 'Sort By', value: 'dateNewToOld' }; - const convertedFilters = myExpensesService.convertFilters(selectedFilters7); + const convertedFilters = myExpensesService.convertSelectedOptionsToExpenseFilters(selectedFilters7); expect(myExpensesService.convertSelectedSortFitlersToFilters).toHaveBeenCalledOnceWith( sortBy, @@ -87,7 +85,7 @@ describe('MyExpensesService', () => { spyOn(myExpensesService, 'convertSelectedSortFitlersToFilters'); const sortBy = { name: 'Sort By', value: 'dateNewToOld' }; - const convertedFilters = myExpensesService.convertFilters(selectedFilters8); + const convertedFilters = myExpensesService.convertSelectedOptionsToExpenseFilters(selectedFilters8); expect(myExpensesService.convertSelectedSortFitlersToFilters).toHaveBeenCalledOnceWith( sortBy, @@ -101,13 +99,13 @@ describe('MyExpensesService', () => { describe('generateSortAmountPills():', () => { it('should add amount - high to low as sort params if sort direction is decreasing', () => { const filterPill = []; - myExpensesService.generateSortAmountPills(expenseFiltersData5Old, filterPill); + myExpensesService.generateSortAmountPills(expenseFiltersData5, filterPill); expect(filterPill).toEqual(sortByDescFilterPill); }); it('should add amount - low to high as sort params if sort direction is ascending', () => { const filterPill = []; - myExpensesService.generateSortAmountPills({ ...expenseFiltersData5Old, sortDir: 'asc' }, filterPill); + myExpensesService.generateSortAmountPills({ ...expenseFiltersData5, sortDir: 'asc' }, filterPill); expect(filterPill).toEqual(sortByAscFilterPill); }); }); @@ -115,13 +113,13 @@ describe('MyExpensesService', () => { describe('generateSortTxnDatePills():', () => { it('should add date - old to new as sort params if sort direction is ascending', () => { const filterPill = []; - myExpensesService.generateSortTxnDatePills(expenseFiltersData7, filterPill); + myExpensesService.generateSortTxnDatePills(expenseFiltersData6, filterPill); expect(filterPill).toEqual(sortByDateAscFilterPill); }); it('should add date - new to old as sort params if sort direction is descending', () => { const filterPill = []; - myExpensesService.generateSortTxnDatePills({ ...expenseFiltersData7, sortDir: 'desc' }, filterPill); + myExpensesService.generateSortTxnDatePills({ ...expenseFiltersData6, sortDir: 'desc' }, filterPill); expect(filterPill).toEqual(sortByDateDescFilterPill); }); }); @@ -129,7 +127,7 @@ describe('MyExpensesService', () => { it('generateTypeFilterPills(): should add combined expense types value in filter pills', () => { const filterPill = []; myExpensesService.generateTypeFilterPills( - { ...expenseFiltersData1, type: ['RegularExpenses', 'PerDiem', 'Mileage', 'custom'] }, + { ...expenseFiltersData1, type: [ExpenseType.EXPENSE, ExpenseType.PER_DIEM, ExpenseType.MILEAGE, 'custom'] }, filterPill ); expect(filterPill).toEqual([ @@ -272,7 +270,7 @@ describe('MyExpensesService', () => { myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); expect(generatedFilters).toEqual({ - sortParam: 'tx_txn_dt', + sortParam: 'spent_at', sortDir: 'desc', }); }); @@ -287,7 +285,7 @@ describe('MyExpensesService', () => { myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); expect(generatedFilters).toEqual({ - sortParam: 'tx_txn_dt', + sortParam: 'spent_at', sortDir: 'asc', }); }); @@ -302,7 +300,7 @@ describe('MyExpensesService', () => { myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); expect(generatedFilters).toEqual({ - sortParam: 'tx_amount', + sortParam: 'amount', sortDir: 'desc', }); }); @@ -317,7 +315,7 @@ describe('MyExpensesService', () => { myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); expect(generatedFilters).toEqual({ - sortParam: 'tx_amount', + sortParam: 'amount', sortDir: 'asc', }); }); @@ -332,7 +330,7 @@ describe('MyExpensesService', () => { myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); expect(generatedFilters).toEqual({ - sortParam: 'tx_org_category', + sortParam: 'category->name', sortDir: 'asc', }); }); @@ -347,7 +345,7 @@ describe('MyExpensesService', () => { myExpensesService.convertSelectedSortFitlersToFilters(sortBy, generatedFilters); expect(generatedFilters).toEqual({ - sortParam: 'tx_org_category', + sortParam: 'category->name', sortDir: 'desc', }); }); @@ -356,7 +354,7 @@ describe('MyExpensesService', () => { it('getFilters(): should return all the filters', () => { const filters = myExpensesService.getFilters(); - expect(filters).toEqual(filterOptions3); + expect(filters).toEqual(filterOptions2); }); it('generateSelectedFilters(): should generate selected filters', () => { @@ -394,7 +392,7 @@ describe('MyExpensesService', () => { it('should add categoryAToZ sort params if sort direction is ascending', () => { const generatedFilters = []; - myExpensesService.convertCategorySortToSelectedFilters(expenseFiltersData1Old, generatedFilters); + myExpensesService.convertCategorySortToSelectedFilters(expenseFiltersData1, generatedFilters); expect(generatedFilters).toEqual([ { @@ -408,7 +406,7 @@ describe('MyExpensesService', () => { const generatedFilters = []; myExpensesService.convertCategorySortToSelectedFilters( - { ...expenseFiltersData1Old, sortDir: 'desc' }, + { ...expenseFiltersData1, sortDir: 'desc' }, generatedFilters ); @@ -424,7 +422,7 @@ describe('MyExpensesService', () => { describe('convertAmountSortToSelectedFilters(): ', () => { it('should convert amount sort to selected filters for descending sort', () => { const filter = { - sortParam: 'tx_amount', + sortParam: 'amount', sortDir: 'desc', }; const generatedFilters = []; @@ -441,7 +439,7 @@ describe('MyExpensesService', () => { it('should convert amount sort to selected filters for ascending sort', () => { const filter = { - sortParam: 'tx_amount', + sortParam: 'amount', sortDir: 'asc', }; const generatedFilters = []; @@ -460,7 +458,7 @@ describe('MyExpensesService', () => { describe('convertTxnDtSortToSelectedFilters():', () => { it('should covert txn date sort to selected filters for descending sort', () => { const filter = { - sortParam: 'tx_txn_dt', + sortParam: 'spent_at', sortDir: 'desc', }; const generatedFilters = []; @@ -477,7 +475,7 @@ describe('MyExpensesService', () => { it('should covert txn date sort to selected filters for ascending sort', () => { const filter = { - sortParam: 'tx_txn_dt', + sortParam: 'spent_at', sortDir: 'asc', }; const generatedFilters = []; @@ -496,7 +494,7 @@ describe('MyExpensesService', () => { describe('generateSortCategoryPills():', () => { it('should add category - a to z as sort params if sort direction is ascending', () => { const filter = { - sortParam: 'tx_org_category', + sortParam: 'category->name', sortDir: 'asc', }; const filterPill = []; @@ -509,7 +507,7 @@ describe('MyExpensesService', () => { it('should add category - z to a as sort params if sort direction is descending', () => { const filter = { - sortParam: 'tx_org_category', + sortParam: 'category->name', sortDir: 'desc', }; const filterPill = []; diff --git a/src/app/fyle/my-expenses/my-expenses.service.ts b/src/app/fyle/my-expenses/my-expenses.service.ts index c01acbd94e..c9a07c6be7 100644 --- a/src/app/fyle/my-expenses/my-expenses.service.ts +++ b/src/app/fyle/my-expenses/my-expenses.service.ts @@ -6,7 +6,9 @@ import { FilterOptionType } from 'src/app/shared/components/fy-filters/filter-op import { FilterOptions } from 'src/app/shared/components/fy-filters/filter-options.interface'; import { SelectedFilters } from 'src/app/shared/components/fy-filters/selected-filters.interface'; import { MaskNumber } from 'src/app/shared/pipes/mask-number.pipe'; -import { ExpenseFilters } from './expense-filters.model'; +import { ExpenseType } from 'src/app/core/enums/expense-type.enum'; +import { ExpenseFilters } from 'src/app/core/models/platform/expense-filters.model'; + @Injectable({ providedIn: 'root', }) @@ -21,7 +23,9 @@ export class MyExpensesService { this.generateSortCategoryPills(filter, filterPills); } - convertFilters(selectedFilters: SelectedFilters[]): Partial { + convertSelectedOptionsToExpenseFilters( + selectedFilters: SelectedFilters[] + ): Partial { const generatedFilters: Partial = {}; const typeFilter = selectedFilters.find((filter) => filter.name === 'Type'); @@ -68,13 +72,13 @@ export class MyExpensesService { } generateSortAmountPills(filter: Partial, filterPills: FilterPill[]): void { - if (filter.sortParam === 'tx_amount' && filter.sortDir === 'desc') { + if (filter.sortParam === 'amount' && filter.sortDir === 'desc') { filterPills.push({ label: 'Sort By', type: 'sort', value: 'amount - high to low', }); - } else if (filter.sortParam === 'tx_amount' && filter.sortDir === 'asc') { + } else if (filter.sortParam === 'amount' && filter.sortDir === 'asc') { filterPills.push({ label: 'Sort By', type: 'sort', @@ -84,13 +88,13 @@ export class MyExpensesService { } generateSortTxnDatePills(filter: Partial, filterPills: FilterPill[]): void { - if (filter.sortParam === 'tx_txn_dt' && filter.sortDir === 'asc') { + if (filter.sortParam === 'spent_at' && filter.sortDir === 'asc') { filterPills.push({ label: 'Sort By', type: 'sort', value: 'date - old to new', }); - } else if (filter.sortParam === 'tx_txn_dt' && filter.sortDir === 'desc') { + } else if (filter.sortParam === 'spent_at' && filter.sortDir === 'desc') { filterPills.push({ label: 'Sort By', type: 'sort', @@ -102,11 +106,11 @@ export class MyExpensesService { generateTypeFilterPills(filter: Partial, filterPills: FilterPill[]): void { const combinedValue = filter.type .map((type) => { - if (type === 'RegularExpenses') { + if (type === 'EXPENSE') { return 'Regular Expenses'; - } else if (type === 'PerDiem') { + } else if (type === 'PER_DIEM') { return 'Per Diem'; - } else if (type === 'Mileage') { + } else if (type === 'MILEAGE') { return 'Mileage'; } else { return type; @@ -240,22 +244,22 @@ export class MyExpensesService { ): void { if (sortBy) { if (sortBy.value === 'dateNewToOld') { - generatedFilters.sortParam = 'tx_txn_dt'; + generatedFilters.sortParam = 'spent_at'; generatedFilters.sortDir = 'desc'; } else if (sortBy.value === 'dateOldToNew') { - generatedFilters.sortParam = 'tx_txn_dt'; + generatedFilters.sortParam = 'spent_at'; generatedFilters.sortDir = 'asc'; } else if (sortBy.value === 'amountHighToLow') { - generatedFilters.sortParam = 'tx_amount'; + generatedFilters.sortParam = 'amount'; generatedFilters.sortDir = 'desc'; } else if (sortBy.value === 'amountLowToHigh') { - generatedFilters.sortParam = 'tx_amount'; + generatedFilters.sortParam = 'amount'; generatedFilters.sortDir = 'asc'; } else if (sortBy.value === 'categoryAToZ') { - generatedFilters.sortParam = 'tx_org_category'; + generatedFilters.sortParam = 'category->name'; generatedFilters.sortDir = 'asc'; } else if (sortBy.value === 'categoryZToA') { - generatedFilters.sortParam = 'tx_org_category'; + generatedFilters.sortParam = 'category->name'; generatedFilters.sortDir = 'desc'; } } @@ -331,15 +335,15 @@ export class MyExpensesService { options: [ { label: 'Mileage', - value: 'Mileage', + value: ExpenseType.MILEAGE, }, { label: 'Per Diem', - value: 'PerDiem', + value: ExpenseType.PER_DIEM, }, { label: 'Regular Expenses', - value: 'RegularExpenses', + value: ExpenseType.EXPENSE, }, ], } as FilterOptions, @@ -461,12 +465,12 @@ export class MyExpensesService { filter: Partial, generatedFilters: SelectedFilters[] ): void { - if (filter.sortParam === 'tx_org_category' && filter.sortDir === 'asc') { + if (filter.sortParam === 'category->name' && filter.sortDir === 'asc') { generatedFilters.push({ name: 'Sort By', value: 'categoryAToZ', }); - } else if (filter.sortParam === 'tx_org_category' && filter.sortDir === 'desc') { + } else if (filter.sortParam === 'category->name' && filter.sortDir === 'desc') { generatedFilters.push({ name: 'Sort By', value: 'categoryZToA', @@ -478,12 +482,12 @@ export class MyExpensesService { filter: Partial, generatedFilters: SelectedFilters[] ): void { - if (filter.sortParam === 'tx_amount' && filter.sortDir === 'desc') { + if (filter.sortParam === 'amount' && filter.sortDir === 'desc') { generatedFilters.push({ name: 'Sort By', value: 'amountHighToLow', }); - } else if (filter.sortParam === 'tx_amount' && filter.sortDir === 'asc') { + } else if (filter.sortParam === 'amount' && filter.sortDir === 'asc') { generatedFilters.push({ name: 'Sort By', value: 'amountLowToHigh', @@ -495,12 +499,12 @@ export class MyExpensesService { filter: Partial, generatedFilters: SelectedFilters[] ): void { - if (filter.sortParam === 'tx_txn_dt' && filter.sortDir === 'asc') { + if (filter.sortParam === 'spent_at' && filter.sortDir === 'asc') { generatedFilters.push({ name: 'Sort By', value: 'dateOldToNew', }); - } else if (filter.sortParam === 'tx_txn_dt' && filter.sortDir === 'desc') { + } else if (filter.sortParam === 'spent_at' && filter.sortDir === 'desc') { generatedFilters.push({ name: 'Sort By', value: 'dateNewToOld', @@ -509,13 +513,13 @@ export class MyExpensesService { } private generateSortCategoryPills(filter: Partial, filterPills: FilterPill[]): void { - if (filter.sortParam === 'tx_org_category' && filter.sortDir === 'asc') { + if (filter.sortParam === 'category->name' && filter.sortDir === 'asc') { filterPills.push({ label: 'Sort By', type: 'sort', value: 'category - a to z', }); - } else if (filter.sortParam === 'tx_org_category' && filter.sortDir === 'desc') { + } else if (filter.sortParam === 'category->name' && filter.sortDir === 'desc') { filterPills.push({ label: 'Sort By', type: 'sort', diff --git a/src/app/fyle/my-reports/my-reports.page.spec.ts b/src/app/fyle/my-reports/my-reports.page.spec.ts index 07404963a5..89afdd819f 100644 --- a/src/app/fyle/my-reports/my-reports.page.spec.ts +++ b/src/app/fyle/my-reports/my-reports.page.spec.ts @@ -31,7 +31,7 @@ import { LoaderService } from 'src/app/core/services/loader.service'; import { TrackingService } from 'src/app/core/services/tracking.service'; import { SelectedFilters } from 'src/app/shared/components/fy-filters/selected-filters.interface'; import { FilterPill } from 'src/app/shared/components/fy-filter-pills/filter-pill.interface'; -import { Filters } from '../my-expenses-v2/my-expenses-filters.model'; +import { Filters } from '../my-expenses/my-expenses-filters.model'; import { selectedFilters1, selectedFilters2, diff --git a/src/app/shared/components/spent-cards/card-detail/card-detail.component.spec.ts b/src/app/shared/components/spent-cards/card-detail/card-detail.component.spec.ts index c3a35e69be..2641fafadb 100644 --- a/src/app/shared/components/spent-cards/card-detail/card-detail.component.spec.ts +++ b/src/app/shared/components/spent-cards/card-detail/card-detail.component.spec.ts @@ -75,13 +75,6 @@ describe('CardDetailComponent', () => { expect(component).toBeTruthy(); }); - it('ngOnInit():should set redirection flag', () => { - orgSettingService.get.and.returnValue(of(orgSettingsWithV2ExpensesPage)); - - expect(component.redirectToNewPage).toBeTrue(); - expect(orgSettingService.get).toHaveBeenCalledTimes(1); - }); - it('should display the card correctly', () => { const card = fixture.debugElement.query(By.directive(MockCorporateCardComponent)); expect(card).toBeTruthy(); @@ -101,7 +94,7 @@ describe('CardDetailComponent', () => { component.goToExpensesPage('incompleteExpenses', component.cardDetail); expect(trackingService.dashboardOnIncompleteCardExpensesClick).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses_v2'], { + expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses'], { queryParams, }); }); @@ -119,7 +112,7 @@ describe('CardDetailComponent', () => { component.goToExpensesPage('totalExpenses', component.cardDetail); expect(trackingService.dashboardOnTotalCardExpensesClick).toHaveBeenCalledTimes(1); - expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses_v2'], { + expect(router.navigate).toHaveBeenCalledOnceWith(['/', 'enterprise', 'my_expenses'], { queryParams, }); }); diff --git a/src/app/shared/components/spent-cards/card-detail/card-detail.component.ts b/src/app/shared/components/spent-cards/card-detail/card-detail.component.ts index 202262f4a5..1d802d9ac2 100644 --- a/src/app/shared/components/spent-cards/card-detail/card-detail.component.ts +++ b/src/app/shared/components/spent-cards/card-detail/card-detail.component.ts @@ -1,7 +1,5 @@ -import { Component, Input, OnInit } from '@angular/core'; +import { Component, Input } from '@angular/core'; import { Params, Router } from '@angular/router'; -import { map, noop } from 'rxjs'; -import { CardStatus } from 'src/app/core/enums/card-status.enum'; import { PlatformCorporateCardDetail } from 'src/app/core/models/platform-corporate-card-detail.model'; import { OrgSettingsService } from 'src/app/core/services/org-settings.service'; import { TrackingService } from 'src/app/core/services/tracking.service'; @@ -11,41 +9,26 @@ import { TrackingService } from 'src/app/core/services/tracking.service'; templateUrl: './card-detail.component.html', styleUrls: ['./card-detail.component.scss'], }) -export class CardDetailComponent implements OnInit { +export class CardDetailComponent { @Input() cardDetail: PlatformCorporateCardDetail; @Input() homeCurrency: string; @Input() currencySymbol: string; - redirectToNewPage = false; - constructor( private router: Router, private trackingService: TrackingService, private orgSettingService: OrgSettingsService ) {} - ngOnInit(): void { - this.orgSettingService - .get() - .pipe( - map((orgSettings) => { - if (orgSettings.mobile_app_my_expenses_beta_enabled) { - this.redirectToNewPage = true; - } - }) - ) - .subscribe(noop); - } - goToExpensesPage(state: string, cardDetail: PlatformCorporateCardDetail): void { if (state === 'incompleteExpenses' && cardDetail.stats.totalDraftTxns && cardDetail.stats.totalDraftTxns > 0) { const queryParams: Params = { filters: JSON.stringify({ state: ['DRAFT'], cardNumbers: [this.cardDetail?.card.card_number] }), }; - this.router.navigate(['/', 'enterprise', `${this.redirectToNewPage ? 'my_expenses_v2' : 'my_expenses'}`], { + this.router.navigate(['/', 'enterprise', 'my_expenses'], { queryParams, }); @@ -54,7 +37,7 @@ export class CardDetailComponent implements OnInit { const queryParams: Params = { filters: JSON.stringify({ state: ['DRAFT,READY_TO_REPORT'], cardNumbers: [this.cardDetail?.card.card_number] }), }; - this.router.navigate(['/', 'enterprise', `${this.redirectToNewPage ? 'my_expenses_v2' : 'my_expenses'}`], { + this.router.navigate(['/', 'enterprise', 'my_expenses'], { queryParams, });