taxtank-core 0.33.67 → 0.33.68

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -2,6 +2,7 @@ import first from 'lodash/first';
2
2
  import { PropertyEquityChartItem } from './property-equity-chart-item';
3
3
  import { FinancialYear } from '../financial-year';
4
4
  import { PropertyEquityChartTypeEnum } from './property-equity-chart-type.enum';
5
+ import { BankAccountTypeEnum } from '../../db/Enums';
5
6
  // count of years for future values
6
7
  const FORECAST_YEARS = 25;
7
8
  /**
@@ -122,7 +123,8 @@ export class PropertyEquityChartData {
122
123
  * @private
123
124
  */
124
125
  getLoanBalanceByYear(property, year, bankAccounts, loans) {
125
- return bankAccounts.items.reduce((totalSum, bankAccount) => totalSum + (bankAccount.getPropertyPercentage(property.id) * (loans.getByBankAccountId(bankAccount.id)?.getLastPaymentByYear(year)?.totalOwed || 0)), 0);
126
+ const offset = bankAccounts.filterBy('type', BankAccountTypeEnum.OFFSET).getPropertyBalanceAmount(property.id);
127
+ return bankAccounts.items.reduce((totalSum, bankAccount) => totalSum + (bankAccount.getPropertyPercentage(property.id) * (loans.getByBankAccountId(bankAccount.id)?.getLastPaymentByYear(year)?.totalOwed || 0)), 0) - offset;
126
128
  }
127
129
  getMarketValueForPropertyByYear(property, year) {
128
130
  if (new FinancialYear(property.contractDate).year > year) {
@@ -158,4 +160,4 @@ export class PropertyEquityChartData {
158
160
  });
159
161
  }
160
162
  }
161
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property-equity-chart-data.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/models/property/property-equity-chart-data.ts"],"names":[],"mappings":"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","sourcesContent":["import first from 'lodash/first';\nimport { PropertyEquityChartItem } from './property-equity-chart-item';\nimport {\n  BankAccountCollection,\n  LoanCollection,\n  PropertyCollection,\n  PropertyValuationCollection\n} from '../../collections';\nimport { FinancialYear } from '../financial-year';\nimport { PropertyEquityChartTypeEnum } from './property-equity-chart-type.enum';\nimport { Property } from './property';\nimport { BankAccount } from '../bank';\n\n// count of years for future values\nconst FORECAST_YEARS = 25;\n\n/**\n * class contain actual and forecasted market values and loan balances for passed properties\n * X axis - years from the first actual data year to the last forecasted year\n * Y axis - amounts of market values, loan balances\n * Also on hover appear equity positions point\n */\nexport class PropertyEquityChartData {\n  // list of equity positions for all years range\n  list: PropertyEquityChartItem[];\n  // current financial year\n  currentYear: number;\n\n  constructor(\n    private properties: PropertyCollection,\n    private valuations: PropertyValuationCollection,\n    private bankAccounts: BankAccountCollection,\n    private loans: LoanCollection,\n    private registerDate: Date\n  ) {\n    this.bankAccounts = bankAccounts.getLoanAndOffsetAccounts();\n    this.currentYear = new FinancialYear(new Date()).year;\n    this.initItems();\n    this.buildHistoryItems();\n    this.buildActualItem();\n    this.buildForecastedItems();\n    this.checkZeroLoanBalance();\n    this.checkZeroMarketValue();\n  }\n\n  /**\n   * get items list in chart series format\n   * @param type\n   */\n  getChartSeries(type: PropertyEquityChartTypeEnum): number[][] {\n    return this.list.map((item: PropertyEquityChartItem) => item.getSerie(type));\n  }\n\n  /**\n   * get prepared chart data\n   */\n  getChartData(): object[] {\n    return [\n      {\n        id: 'marketValue',\n        name: 'Market Value',\n        series: this.getChartSeries(PropertyEquityChartTypeEnum.MARKET_VALUE),\n        // display future market value with dash line and past market value with solid line\n        zones: [{\n          // line style after current calendar year\n          value: this.currentYear,\n          dashStyle: 'Solid'\n        }, {\n          // default line style\n          dashStyle: 'Dash'\n        }]\n      },\n      {\n        // id used to add/remove serie to chart\n        id: 'loanBalance',\n        name: 'Loan Balance',\n        series: this.getChartSeries(PropertyEquityChartTypeEnum.LOAN_BALANCE)\n      }\n    ];\n  }\n\n  /**\n   * create empty equity position item for each year in collection\n   */\n  private initItems(): void {\n    this.list = [];\n    for (let year: number = new FinancialYear(this.registerDate).year - 1; year <= (this.currentYear + FORECAST_YEARS); year++) {\n      this.list.push(new PropertyEquityChartItem(year));\n    }\n  }\n\n  /**\n   * collect equity position items with past values from properties forecasts\n   */\n  private buildHistoryItems(): void {\n    // get items for past and current years\n    const items: PropertyEquityChartItem[] = this.list.filter((item: PropertyEquityChartItem) => item.year <= this.currentYear);\n    items.forEach((item: PropertyEquityChartItem) => {\n      // increase item values with property actual forecasts for item year\n      item.marketValue = this.properties.items.reduce((sum: number, property: Property): number => sum + this.getMarketValueForPropertyByYear(property, item.year), 0);\n\n      item.loanBalance = Math.abs(this.properties.items.reduce((sum: number, property: Property): number => sum + this.getLoanBalanceForPropertyByYear(property, item.year), 0));\n    });\n  }\n\n  /**\n   * @TODO remove, there is no difference between actual and history calculations\n   * set actual year's real data\n   * @private\n   */\n  private buildActualItem(): void {\n    const item: PropertyEquityChartItem = this.list.find((i: PropertyEquityChartItem) => i.year === this.currentYear);\n    item.marketValue = this.valuations.getMarketValue(new Date());\n    item.loanBalance = this.properties.items.reduce((sum: number, property: Property): number => sum + Math.abs(this.bankAccounts.getPropertyBalanceAmount(property.id)), 0);\n  }\n\n  /**\n   * collect equity position items with forecasted (future) calculated values\n   */\n  private buildForecastedItems(): void {\n    // calculate future values for all properties separately\n    this.properties.items.forEach((property: Property) => {\n      const marketValue = this.valuations.filterBy('property.id', property.id).getMarketValue(new Date());\n      let forecastedMarketValue: number = (property.growthPercent / 100) * marketValue + marketValue;\n\n      // calculate future values for each future year for current handling property\n      this.list\n        .filter((item: PropertyEquityChartItem) => item.year > this.currentYear)\n        .forEach((item: PropertyEquityChartItem) => {\n          const propertyBankAccounts: BankAccountCollection = this.bankAccounts.getByPropertyId(property.id);\n          // increase loan balance amount for handling equity position item\n          item.loanBalance += this.getLoanBalanceByYear(property, item.year, propertyBankAccounts, this.loans.getByBankAccountsIds(propertyBankAccounts.getIds()));\n\n          // increase market value amount for handling equity position item\n          item.marketValue += forecastedMarketValue;\n\n          // increase current market value forecast for the next cycle iteration\n          forecastedMarketValue += forecastedMarketValue * (property.growthPercent / 100);\n        });\n    });\n  }\n\n  /**\n   * Get property loan balance by financial year\n   * @param property Property instance for filter\n   * @param year Financial year number\n   * @param bankAccounts List of bank accounts\n   * @param loans List of loans\n   * @private\n   */\n  private getLoanBalanceByYear(property: Property, year: number, bankAccounts: BankAccountCollection, loans: LoanCollection): number {\n    return bankAccounts.items.reduce((totalSum: number, bankAccount: BankAccount): number => totalSum + (bankAccount.getPropertyPercentage(property.id) * (loans.getByBankAccountId(bankAccount.id)?.getLastPaymentByYear(year)?.totalOwed || 0)), 0);\n  }\n\n  getMarketValueForPropertyByYear(property: Property, year: number): number {\n    if (new FinancialYear(property.contractDate).year > year) {\n      return 0;\n    }\n\n    return (property.getForecastByYear(year) || first(property.forecasts)).marketValue;\n  }\n\n  getLoanBalanceForPropertyByYear(property: Property, year: number): number {\n    if (new FinancialYear(property.contractDate).year > year) {\n      return 0;\n    }\n\n    // @TODO remove after tests `|| this.bankAccounts.getPropertyBalanceAmount(property.id)`\n    return property.getForecastByYear(year)?.loanBalance;\n  }\n\n  /**\n   * Check if loan balance is 0 and set it as null (to not to draw point on the chart)\n   */\n  private checkZeroLoanBalance(): void {\n    this.list.forEach((item: PropertyEquityChartItem) => {\n      if (item.loanBalance === 0) {\n        item.loanBalance = null;\n      }\n    });\n  }\n\n  /**\n   * Check if market value is 0 and set it as null (to not to draw point on the chart)\n   */\n  private checkZeroMarketValue(): void {\n    this.list.forEach((item: PropertyEquityChartItem) => {\n      if (item.marketValue === 0) {\n        item.marketValue = null;\n      }\n    });\n  }\n}\n"]}
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property-equity-chart-data.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/models/property/property-equity-chart-data.ts"],"names":[],"mappings":"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","sourcesContent":["import first from 'lodash/first';\nimport { PropertyEquityChartItem } from './property-equity-chart-item';\nimport {\n  BankAccountCollection,\n  LoanCollection,\n  PropertyCollection,\n  PropertyValuationCollection\n} from '../../collections';\nimport { FinancialYear } from '../financial-year';\nimport { PropertyEquityChartTypeEnum } from './property-equity-chart-type.enum';\nimport { Property } from './property';\nimport { BankAccount } from '../bank';\nimport { BankAccountTypeEnum } from '../../db/Enums';\n\n// count of years for future values\nconst FORECAST_YEARS = 25;\n\n/**\n * class contain actual and forecasted market values and loan balances for passed properties\n * X axis - years from the first actual data year to the last forecasted year\n * Y axis - amounts of market values, loan balances\n * Also on hover appear equity positions point\n */\nexport class PropertyEquityChartData {\n  // list of equity positions for all years range\n  list: PropertyEquityChartItem[];\n  // current financial year\n  currentYear: number;\n\n  constructor(\n    private properties: PropertyCollection,\n    private valuations: PropertyValuationCollection,\n    private bankAccounts: BankAccountCollection,\n    private loans: LoanCollection,\n    private registerDate: Date\n  ) {\n    this.bankAccounts = bankAccounts.getLoanAndOffsetAccounts();\n    this.currentYear = new FinancialYear(new Date()).year;\n    this.initItems();\n    this.buildHistoryItems();\n    this.buildActualItem();\n    this.buildForecastedItems();\n    this.checkZeroLoanBalance();\n    this.checkZeroMarketValue();\n  }\n\n  /**\n   * get items list in chart series format\n   * @param type\n   */\n  getChartSeries(type: PropertyEquityChartTypeEnum): number[][] {\n    return this.list.map((item: PropertyEquityChartItem) => item.getSerie(type));\n  }\n\n  /**\n   * get prepared chart data\n   */\n  getChartData(): object[] {\n    return [\n      {\n        id: 'marketValue',\n        name: 'Market Value',\n        series: this.getChartSeries(PropertyEquityChartTypeEnum.MARKET_VALUE),\n        // display future market value with dash line and past market value with solid line\n        zones: [{\n          // line style after current calendar year\n          value: this.currentYear,\n          dashStyle: 'Solid'\n        }, {\n          // default line style\n          dashStyle: 'Dash'\n        }]\n      },\n      {\n        // id used to add/remove serie to chart\n        id: 'loanBalance',\n        name: 'Loan Balance',\n        series: this.getChartSeries(PropertyEquityChartTypeEnum.LOAN_BALANCE)\n      }\n    ];\n  }\n\n  /**\n   * create empty equity position item for each year in collection\n   */\n  private initItems(): void {\n    this.list = [];\n    for (let year: number = new FinancialYear(this.registerDate).year - 1; year <= (this.currentYear + FORECAST_YEARS); year++) {\n      this.list.push(new PropertyEquityChartItem(year));\n    }\n  }\n\n  /**\n   * collect equity position items with past values from properties forecasts\n   */\n  private buildHistoryItems(): void {\n    // get items for past and current years\n    const items: PropertyEquityChartItem[] = this.list.filter((item: PropertyEquityChartItem) => item.year <= this.currentYear);\n    items.forEach((item: PropertyEquityChartItem) => {\n      // increase item values with property actual forecasts for item year\n      item.marketValue = this.properties.items.reduce((sum: number, property: Property): number => sum + this.getMarketValueForPropertyByYear(property, item.year), 0);\n\n      item.loanBalance = Math.abs(this.properties.items.reduce((sum: number, property: Property): number => sum + this.getLoanBalanceForPropertyByYear(property, item.year), 0));\n    });\n  }\n\n  /**\n   * @TODO remove, there is no difference between actual and history calculations\n   * set actual year's real data\n   * @private\n   */\n  private buildActualItem(): void {\n    const item: PropertyEquityChartItem = this.list.find((i: PropertyEquityChartItem) => i.year === this.currentYear);\n    item.marketValue = this.valuations.getMarketValue(new Date());\n    item.loanBalance = this.properties.items.reduce((sum: number, property: Property): number => sum + Math.abs(this.bankAccounts.getPropertyBalanceAmount(property.id)), 0);\n  }\n\n  /**\n   * collect equity position items with forecasted (future) calculated values\n   */\n  private buildForecastedItems(): void {\n    // calculate future values for all properties separately\n    this.properties.items.forEach((property: Property) => {\n      const marketValue = this.valuations.filterBy('property.id', property.id).getMarketValue(new Date());\n      let forecastedMarketValue: number = (property.growthPercent / 100) * marketValue + marketValue;\n\n      // calculate future values for each future year for current handling property\n      this.list\n        .filter((item: PropertyEquityChartItem) => item.year > this.currentYear)\n        .forEach((item: PropertyEquityChartItem) => {\n          const propertyBankAccounts: BankAccountCollection = this.bankAccounts.getByPropertyId(property.id);\n          // increase loan balance amount for handling equity position item\n          item.loanBalance += this.getLoanBalanceByYear(property, item.year, propertyBankAccounts, this.loans.getByBankAccountsIds(propertyBankAccounts.getIds()));\n\n          // increase market value amount for handling equity position item\n          item.marketValue += forecastedMarketValue;\n\n          // increase current market value forecast for the next cycle iteration\n          forecastedMarketValue += forecastedMarketValue * (property.growthPercent / 100);\n        });\n    });\n  }\n\n  /**\n   * Get property loan balance by financial year\n   * @param property Property instance for filter\n   * @param year Financial year number\n   * @param bankAccounts List of bank accounts\n   * @param loans List of loans\n   * @private\n   */\n  private getLoanBalanceByYear(property: Property, year: number, bankAccounts: BankAccountCollection, loans: LoanCollection): number {\n    const offset = bankAccounts.filterBy('type', BankAccountTypeEnum.OFFSET).getPropertyBalanceAmount(property.id);\n\n    return bankAccounts.items.reduce((totalSum: number, bankAccount: BankAccount) => totalSum + (bankAccount.getPropertyPercentage(property.id) * (loans.getByBankAccountId(bankAccount.id)?.getLastPaymentByYear(year)?.totalOwed || 0)), 0) - offset;\n  }\n\n  getMarketValueForPropertyByYear(property: Property, year: number): number {\n    if (new FinancialYear(property.contractDate).year > year) {\n      return 0;\n    }\n\n    return (property.getForecastByYear(year) || first(property.forecasts)).marketValue;\n  }\n\n  getLoanBalanceForPropertyByYear(property: Property, year: number): number {\n    if (new FinancialYear(property.contractDate).year > year) {\n      return 0;\n    }\n\n    // @TODO remove after tests `|| this.bankAccounts.getPropertyBalanceAmount(property.id)`\n    return property.getForecastByYear(year)?.loanBalance;\n  }\n\n  /**\n   * Check if loan balance is 0 and set it as null (to not to draw point on the chart)\n   */\n  private checkZeroLoanBalance(): void {\n    this.list.forEach((item: PropertyEquityChartItem) => {\n      if (item.loanBalance === 0) {\n        item.loanBalance = null;\n      }\n    });\n  }\n\n  /**\n   * Check if market value is 0 and set it as null (to not to draw point on the chart)\n   */\n  private checkZeroMarketValue(): void {\n    this.list.forEach((item: PropertyEquityChartItem) => {\n      if (item.marketValue === 0) {\n        item.marketValue = null;\n      }\n    });\n  }\n}\n"]}
@@ -39,14 +39,7 @@ export class TransactionService extends RestService {
39
39
  this.listenNotifications();
40
40
  }
41
41
  get(path = this.apiUrl) {
42
- return super.get(path).pipe(map(transactions => {
43
- transactions.forEach((transaction) => {
44
- transaction.transactions = transactions
45
- // get list of child transactions
46
- .filter((t) => t.parentTransaction && t.parentTransaction.id === transaction.id);
47
- });
48
- return transactions;
49
- }));
42
+ return super.get(path).pipe(map(transactions => this.groupByParent(transactions)));
50
43
  }
51
44
  getCurrentYear() {
52
45
  return super.get().pipe(map((transactions) => new TransactionCollection(transactions).filterByFinancialYear('date').toArray()));
@@ -85,7 +78,15 @@ export class TransactionService extends RestService {
85
78
  return this.fetch(`${this.environment.apiV2}/properties/${propertyId}/holding-costs`, false);
86
79
  }
87
80
  getWithoutTaxFreeProperty() {
88
- return this.fetch(`${this.apiUrl}?excludeTaxFreePropertyCategories`, false);
81
+ return this.fetch(`${this.apiUrl}?excludeTaxFreePropertyCategories`, false).pipe(map(transactions => this.groupByParent(transactions)));
82
+ }
83
+ groupByParent(transactions) {
84
+ transactions.forEach((transaction) => {
85
+ transaction.transactions = transactions
86
+ // get list of child transactions
87
+ .filter((t) => t.parentTransaction && t.parentTransaction.id === transaction.id);
88
+ });
89
+ return transactions;
89
90
  }
90
91
  /**
91
92
  * get list of taxable transactions with tank type 'Work'
@@ -269,4 +270,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
269
270
  type: Inject,
270
271
  args: ['environment']
271
272
  }] }] });
272
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tt-core/src/lib/services/http/transaction/transaction.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter, Inject, Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\nimport { Transaction as TransactionBase } from '../../../db/Models/transaction/transaction';\nimport { map } from 'rxjs/operators';\nimport { replace } from '../../../functions/array';\nimport { TransactionOperationEnum } from '../../../db/Enums/transaction-operation.enum';\nimport { RestService } from '../rest/rest-old.service';\nimport {\n  AppEvent,\n  AppEventTypeEnum,\n  Depreciation, HomeOfficeClaim, HomeOfficeLog,\n  Notification, PropertyShare,\n  Transaction,\n  VehicleClaim\n} from '../../../models';\nimport _ from 'lodash';\nimport { HttpClient } from '@angular/common/http';\nimport { EventDispatcherService } from '../../event';\nimport { IEventListener } from '../../../interfaces';\nimport { classToPlain } from 'class-transformer';\nimport { UserEventTypeTypeEnum } from '../../../db/Enums/user-event-type-type.enum';\nimport { TransactionCollection } from '../../../collections';\n\n/**\n * Service for transactions business logic\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class TransactionService extends RestService<TransactionBase, Transaction> implements IEventListener {\n  // url part for Transaction API\n  url = 'transactions';\n  modelClass = Transaction;\n  transactionDeleted: EventEmitter<Transaction> = new EventEmitter<Transaction>();\n\n  constructor(\n    protected http: HttpClient,\n    protected eventDispatcherService: EventDispatcherService,\n    @Inject('environment') protected environment: any,\n  ) {\n    super(environment);\n    this.listenEvents();\n  }\n\n  /**\n   * Listen events from Event Dispatcher services\n   */\n  listenEvents(): void {\n    this.eventDispatcherService.on2(PropertyShare.getEventName('put')).subscribe(() => this.refreshCache());\n    this.listenDepreciationChange();\n    this.listenVehicleClaimChanges();\n    this.listenHomeOfficeClaim();\n    this.listenHomeOfficeLog();\n    this.listenPropertyForecastUpdate();\n    this.listenNotifications();\n  }\n\n  get(path = this.apiUrl): Observable<Transaction[]> {\n    return super.get(path).pipe(\n      map(transactions => {\n        transactions.forEach((transaction: Transaction): void => {\n          transaction.transactions = transactions\n            // get list of child transactions\n            .filter((t: Transaction): boolean => t.parentTransaction && t.parentTransaction.id === transaction.id);\n        });\n\n        return transactions;\n      })\n    )\n  }\n\n  getCurrentYear(): Observable<Transaction[]> {\n    return super.get().pipe(\n      map((transactions: Transaction[]): Transaction[] => new TransactionCollection(transactions).filterByFinancialYear('date').toArray())\n    );\n  }\n\n  /**\n   * Add single new transaction\n   * @param model New Transaction instance for saving\n   */\n  add(model: Transaction): Observable<Transaction> {\n    // we don't have POST API endpoint for single transaction\n    return this.addBatch([model])\n      .pipe(\n        map((newTransactions: Transaction[]): Transaction => {\n          // @TODO alex\n          this.eventDispatcherService.dispatch(\n            new AppEvent<Transaction>(AppEventTypeEnum.TRANSACTION_CREATED, newTransactions[0])\n          );\n\n          return newTransactions[0];\n        })\n      );\n  }\n\n  /**\n   * get transactions related with property\n   */\n  getByPropertyId(propertyId: number): Observable<Transaction[]> {\n    return this.getCurrentYear()\n      .pipe(\n        map((transactions: Transaction[]): Transaction[] => transactions.filter((transaction: Transaction): boolean => transaction.property?.id === propertyId))\n      );\n  }\n\n  /**\n   * get list of transactions with tank type 'Work'\n   */\n  getWorkTransactions(): Observable<Transaction[]> {\n    return this.getCurrentYear()\n      .pipe(\n        map((transactions: Transaction[]): Transaction[] => transactions.filter((transaction: Transaction): boolean => transaction.isWorkTank()))\n      );\n  }\n\n  /**\n   * Get list of property holding costs (transactions related to vacant land property)\n   */\n  getPropertyHoldingCosts(propertyId: number): Observable<Transaction[]> {\n    return this.fetch(`${this.environment.apiV2}/properties/${propertyId}/holding-costs`, false);\n  }\n\n  getWithoutTaxFreeProperty(): Observable<Transaction[]> {\n    return this.fetch(`${this.apiUrl}?excludeTaxFreePropertyCategories`, false);\n  }\n\n  /**\n   * get list of taxable transactions with tank type 'Work'\n   */\n  getTaxableWorkTransactions(): Observable<Transaction[]> {\n    return this.getWorkTransactions()\n      .pipe(\n        map((transactions: Transaction[]): Transaction[] => transactions.filter((transaction: Transaction): boolean => !!transaction.chartAccounts.taxReturnItem))\n      );\n  }\n\n  /**\n   * add multiple transactions\n   * @param transactions List of new Transaction instances for saving\n   */\n  addBatch(transactions: Transaction[]): Observable<Transaction[]> {\n    transactions = _.cloneDeep(transactions);\n    return this.http.post(`${this.environment.apiV2}/${this.url}`, classToPlain(transactions))\n      .pipe(\n        map((response: TransactionBase[]) => {\n          const addedTransactions: Transaction[] = response.map((item: TransactionBase) => this.createModelInstance(item));\n\n          transactions.forEach((transaction: Transaction, index: number) => {\n            // @TODO Viktor: implement API for saving of nested transactions\n            // add child transactions if exist\n            if (transaction.transactions.length) {\n              transaction.transactions.forEach((childTransaction: Transaction) => {\n                childTransaction.parentTransaction = this.createModelInstance({ id: addedTransactions[index].id });\n              });\n              this.addBatch(transaction.transactions).subscribe();\n            }\n\n            // add transfer transaction to cache\n            if (transaction.operation === TransactionOperationEnum.TRANSFER) {\n              addedTransactions.push(addedTransactions[0].transfer);\n            }\n          });\n\n          if (this.cache) {\n            this.cache.push(...addedTransactions);\n            this.updateCache();\n          }\n\n          this.eventDispatcherService.dispatch(\n            new AppEvent<Transaction[]>(AppEventTypeEnum.TRANSACTIONS_CREATED, addedTransactions)\n          );\n\n          return addedTransactions;\n        })\n      );\n  }\n\n  /**\n   * update existing transaction\n   * @param transaction Transaction instance for updating\n   */\n  update(transaction: Transaction): Observable<Transaction> {\n    return this.http.put(`${this.environment.apiV2}/${this.url}/${transaction.id}`, classToPlain(transaction))\n      .pipe(\n        map((response: TransactionBase) => {\n          const updatedTransaction: Transaction = this.createModelInstance(response);\n          // @TODO Viktor: implement API for saving of nested transactions\n          if (transaction.transactions.length) {\n            // add parent transaction to child transactions\n            transaction.transactions.forEach((childTransaction: Transaction) => {\n              childTransaction.parentTransaction = this.createModelInstance({ id: updatedTransaction.id });\n            });\n            // separate child transactions by id existing to define add or update action.\n            const childTransactionsToUpdate: Transaction[] = transaction.transactions.filter((t: Transaction) => t.id);\n            const childTransactionsToAdd: Transaction[] = transaction.transactions.filter((t: Transaction) => !t.id);\n\n            // update child transactions\n            if (childTransactionsToUpdate.length) {\n              this.updateBatch(childTransactionsToUpdate).subscribe();\n            }\n\n            // add child transactions\n            if (childTransactionsToAdd.length) {\n              this.addBatch(childTransactionsToAdd).subscribe();\n            }\n          }\n\n          this.replaceInCache(updatedTransaction);\n\n          return updatedTransaction;\n        })\n      );\n  }\n\n  /**\n   * update multiple transactions\n   * @param transactions list of transactions for updating\n   */\n  updateBatch(transactions: Transaction[]): Observable<Transaction[]> {\n    return this.http.put(`${this.environment.apiV2}/${this.url}`, classToPlain(transactions))\n      .pipe(\n        map((response: TransactionBase[]) => {\n          const updatedTransactions: Transaction[] = response.map((item: TransactionBase) => this.createModelInstance(item));\n\n          updatedTransactions.forEach((updatedTransaction: Transaction) => {\n            replace(this.cache, updatedTransaction);\n          });\n\n          this.updateCache();\n          return updatedTransactions;\n        })\n      );\n  }\n\n  deleteBatch(models: Transaction[]): Observable<void> {\n    return super.deleteBatch(models).pipe(\n      map(() => {\n        this.eventDispatcherService.dispatch(\n          new AppEvent<Transaction>(AppEventTypeEnum.TRANSACTION_DELETED, models[0])\n        );\n      })\n    );\n  }\n\n  /**\n   * delete transaction and related transactions\n   * @param model\n   */\n  delete(model: Transaction): Observable<void> {\n    return this.http.delete(`${this.environment.apiV2}/${this.url}/${model.id}`)\n      .pipe(\n        map((): void => {\n          this.cache = this.cache.filter((transaction: Transaction): boolean => {\n            // when delete transfer we delete actually 2 transactions\n            if (model.isTransfer) {\n              const ids: number[] = [model.id];\n\n              // get id of related transfer transaction\n              if (model.transfer) {\n                // just take id if we delete source transaction\n                ids.push(model.transfer.id);\n              } else {\n                // find source transaction id if we delete destination transaction\n                ids.push(this.cache.find((t: Transaction): boolean => t.transfer.id === model.id).id);\n              }\n\n              return !ids.includes(transaction.id);\n            }\n\n            return transaction.id !== model.id && transaction.parentTransaction?.id !== model.id;\n          });\n\n          this.eventDispatcherService.dispatch(\n            new AppEvent<Transaction>(AppEventTypeEnum.TRANSACTION_DELETED, model)\n          );\n\n          this.updateCache();\n          this.transactionDeleted.emit(model);\n        })\n      );\n  }\n\n  /**\n   * Listen to EventDispatcherService event related to Depreciation changing\n   */\n  private listenDepreciationChange(): void {\n    // @TODO reset cache only when required (depreciation has transaction)\n    this.eventDispatcherService.on2(...Depreciation.getEventNames('post', 'delete')).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  private listenVehicleClaimChanges(): void {\n    this.eventDispatcherService.on2(...VehicleClaim.getEventNames('post', 'put')).subscribe(() => this.refreshCache());\n    this.eventDispatcherService.on([AppEventTypeEnum.VEHICLE_CLAIM_DETAILS_UPDATED]).subscribe(() => {\n      this.refreshCache();\n    })\n  }\n\n  private listenHomeOfficeClaim(): void {\n    this.eventDispatcherService.on2(...HomeOfficeClaim.getEventNames('post', 'put')).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  private listenHomeOfficeLog(): void {\n    this.eventDispatcherService.on2(...HomeOfficeLog.getEventNames('post', 'put', 'delete')).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  /**\n   * property forecast includes claimPercent which affects property related expenses\n   * @TODO Alex move to forecast api instead of property\n   */\n  private listenPropertyForecastUpdate(): void {\n    this.eventDispatcherService.on([AppEventTypeEnum.PROPERTY_UPDATED]).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  private replaceInCache(transactionToReplace: Transaction): void {\n    replace(this.cache, transactionToReplace);\n    this.updateCache();\n  }\n\n  private listenNotifications(): void {\n    this.eventDispatcherService.on(AppEventTypeEnum.NOTIFICATION_ADDED).subscribe((notification: Notification) => {\n      const events = [\n        UserEventTypeTypeEnum.INVOICE_AUTO_ALLOCATED,\n        UserEventTypeTypeEnum.INVOICE_AUTO_PART_ALLOCATED,\n        UserEventTypeTypeEnum.RULES_AUTO_ALLOCATED\n      ];\n\n      if (!notification.isRead && events.includes(notification.eventType)) {\n        // @TODO TT-3826 move to unified mercure event\n        setTimeout(() => {\n          this.refreshCache();\n        }, 3000)\n      }\n    });\n  }\n}\n"]}
273
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tt-core/src/lib/services/http/transaction/transaction.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter, Inject, Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\nimport { Transaction as TransactionBase } from '../../../db/Models/transaction/transaction';\nimport { map } from 'rxjs/operators';\nimport { replace } from '../../../functions/array';\nimport { TransactionOperationEnum } from '../../../db/Enums/transaction-operation.enum';\nimport { RestService } from '../rest/rest-old.service';\nimport {\n  AppEvent,\n  AppEventTypeEnum,\n  Depreciation, HomeOfficeClaim, HomeOfficeLog,\n  Notification, PropertyShare,\n  Transaction,\n  VehicleClaim\n} from '../../../models';\nimport _ from 'lodash';\nimport { HttpClient } from '@angular/common/http';\nimport { EventDispatcherService } from '../../event';\nimport { IEventListener } from '../../../interfaces';\nimport { classToPlain } from 'class-transformer';\nimport { UserEventTypeTypeEnum } from '../../../db/Enums/user-event-type-type.enum';\nimport { TransactionCollection } from '../../../collections';\n\n/**\n * Service for transactions business logic\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class TransactionService extends RestService<TransactionBase, Transaction> implements IEventListener {\n  // url part for Transaction API\n  url = 'transactions';\n  modelClass = Transaction;\n  transactionDeleted: EventEmitter<Transaction> = new EventEmitter<Transaction>();\n\n  constructor(\n    protected http: HttpClient,\n    protected eventDispatcherService: EventDispatcherService,\n    @Inject('environment') protected environment: any,\n  ) {\n    super(environment);\n    this.listenEvents();\n  }\n\n  /**\n   * Listen events from Event Dispatcher services\n   */\n  listenEvents(): void {\n    this.eventDispatcherService.on2(PropertyShare.getEventName('put')).subscribe(() => this.refreshCache());\n    this.listenDepreciationChange();\n    this.listenVehicleClaimChanges();\n    this.listenHomeOfficeClaim();\n    this.listenHomeOfficeLog();\n    this.listenPropertyForecastUpdate();\n    this.listenNotifications();\n  }\n\n  get(path = this.apiUrl): Observable<Transaction[]> {\n    return super.get(path).pipe(map(transactions => this.groupByParent(transactions)))\n  }\n\n  getCurrentYear(): Observable<Transaction[]> {\n    return super.get().pipe(\n      map((transactions: Transaction[]): Transaction[] => new TransactionCollection(transactions).filterByFinancialYear('date').toArray())\n    );\n  }\n\n  /**\n   * Add single new transaction\n   * @param model New Transaction instance for saving\n   */\n  add(model: Transaction): Observable<Transaction> {\n    // we don't have POST API endpoint for single transaction\n    return this.addBatch([model])\n      .pipe(\n        map((newTransactions: Transaction[]): Transaction => {\n          // @TODO alex\n          this.eventDispatcherService.dispatch(\n            new AppEvent<Transaction>(AppEventTypeEnum.TRANSACTION_CREATED, newTransactions[0])\n          );\n\n          return newTransactions[0];\n        })\n      );\n  }\n\n  /**\n   * get transactions related with property\n   */\n  getByPropertyId(propertyId: number): Observable<Transaction[]> {\n    return this.getCurrentYear()\n      .pipe(\n        map((transactions: Transaction[]): Transaction[] => transactions.filter((transaction: Transaction): boolean => transaction.property?.id === propertyId))\n      );\n  }\n\n  /**\n   * get list of transactions with tank type 'Work'\n   */\n  getWorkTransactions(): Observable<Transaction[]> {\n    return this.getCurrentYear()\n      .pipe(\n        map((transactions: Transaction[]): Transaction[] => transactions.filter((transaction: Transaction): boolean => transaction.isWorkTank()))\n      );\n  }\n\n  /**\n   * Get list of property holding costs (transactions related to vacant land property)\n   */\n  getPropertyHoldingCosts(propertyId: number): Observable<Transaction[]> {\n    return this.fetch(`${this.environment.apiV2}/properties/${propertyId}/holding-costs`, false);\n  }\n\n  getWithoutTaxFreeProperty(): Observable<Transaction[]> {\n    return this.fetch(`${this.apiUrl}?excludeTaxFreePropertyCategories`, false).pipe(\n      map(transactions => this.groupByParent(transactions))\n    );\n  }\n\n  private groupByParent(transactions: Transaction[]): Transaction[] {\n    transactions.forEach((transaction: Transaction): void => {\n      transaction.transactions = transactions\n        // get list of child transactions\n        .filter((t: Transaction): boolean => t.parentTransaction && t.parentTransaction.id === transaction.id);\n    });\n\n    return transactions;\n  }\n\n  /**\n   * get list of taxable transactions with tank type 'Work'\n   */\n  getTaxableWorkTransactions(): Observable<Transaction[]> {\n    return this.getWorkTransactions()\n      .pipe(\n        map((transactions: Transaction[]): Transaction[] => transactions.filter((transaction: Transaction): boolean => !!transaction.chartAccounts.taxReturnItem))\n      );\n  }\n\n  /**\n   * add multiple transactions\n   * @param transactions List of new Transaction instances for saving\n   */\n  addBatch(transactions: Transaction[]): Observable<Transaction[]> {\n    transactions = _.cloneDeep(transactions);\n    return this.http.post(`${this.environment.apiV2}/${this.url}`, classToPlain(transactions))\n      .pipe(\n        map((response: TransactionBase[]) => {\n          const addedTransactions: Transaction[] = response.map((item: TransactionBase) => this.createModelInstance(item));\n\n          transactions.forEach((transaction: Transaction, index: number) => {\n            // @TODO Viktor: implement API for saving of nested transactions\n            // add child transactions if exist\n            if (transaction.transactions.length) {\n              transaction.transactions.forEach((childTransaction: Transaction) => {\n                childTransaction.parentTransaction = this.createModelInstance({ id: addedTransactions[index].id });\n              });\n              this.addBatch(transaction.transactions).subscribe();\n            }\n\n            // add transfer transaction to cache\n            if (transaction.operation === TransactionOperationEnum.TRANSFER) {\n              addedTransactions.push(addedTransactions[0].transfer);\n            }\n          });\n\n          if (this.cache) {\n            this.cache.push(...addedTransactions);\n            this.updateCache();\n          }\n\n          this.eventDispatcherService.dispatch(\n            new AppEvent<Transaction[]>(AppEventTypeEnum.TRANSACTIONS_CREATED, addedTransactions)\n          );\n\n          return addedTransactions;\n        })\n      );\n  }\n\n  /**\n   * update existing transaction\n   * @param transaction Transaction instance for updating\n   */\n  update(transaction: Transaction): Observable<Transaction> {\n    return this.http.put(`${this.environment.apiV2}/${this.url}/${transaction.id}`, classToPlain(transaction))\n      .pipe(\n        map((response: TransactionBase) => {\n          const updatedTransaction: Transaction = this.createModelInstance(response);\n          // @TODO Viktor: implement API for saving of nested transactions\n          if (transaction.transactions.length) {\n            // add parent transaction to child transactions\n            transaction.transactions.forEach((childTransaction: Transaction) => {\n              childTransaction.parentTransaction = this.createModelInstance({ id: updatedTransaction.id });\n            });\n            // separate child transactions by id existing to define add or update action.\n            const childTransactionsToUpdate: Transaction[] = transaction.transactions.filter((t: Transaction) => t.id);\n            const childTransactionsToAdd: Transaction[] = transaction.transactions.filter((t: Transaction) => !t.id);\n\n            // update child transactions\n            if (childTransactionsToUpdate.length) {\n              this.updateBatch(childTransactionsToUpdate).subscribe();\n            }\n\n            // add child transactions\n            if (childTransactionsToAdd.length) {\n              this.addBatch(childTransactionsToAdd).subscribe();\n            }\n          }\n\n          this.replaceInCache(updatedTransaction);\n\n          return updatedTransaction;\n        })\n      );\n  }\n\n  /**\n   * update multiple transactions\n   * @param transactions list of transactions for updating\n   */\n  updateBatch(transactions: Transaction[]): Observable<Transaction[]> {\n    return this.http.put(`${this.environment.apiV2}/${this.url}`, classToPlain(transactions))\n      .pipe(\n        map((response: TransactionBase[]) => {\n          const updatedTransactions: Transaction[] = response.map((item: TransactionBase) => this.createModelInstance(item));\n\n          updatedTransactions.forEach((updatedTransaction: Transaction) => {\n            replace(this.cache, updatedTransaction);\n          });\n\n          this.updateCache();\n          return updatedTransactions;\n        })\n      );\n  }\n\n  deleteBatch(models: Transaction[]): Observable<void> {\n    return super.deleteBatch(models).pipe(\n      map(() => {\n        this.eventDispatcherService.dispatch(\n          new AppEvent<Transaction>(AppEventTypeEnum.TRANSACTION_DELETED, models[0])\n        );\n      })\n    );\n  }\n\n  /**\n   * delete transaction and related transactions\n   * @param model\n   */\n  delete(model: Transaction): Observable<void> {\n    return this.http.delete(`${this.environment.apiV2}/${this.url}/${model.id}`)\n      .pipe(\n        map((): void => {\n          this.cache = this.cache.filter((transaction: Transaction): boolean => {\n            // when delete transfer we delete actually 2 transactions\n            if (model.isTransfer) {\n              const ids: number[] = [model.id];\n\n              // get id of related transfer transaction\n              if (model.transfer) {\n                // just take id if we delete source transaction\n                ids.push(model.transfer.id);\n              } else {\n                // find source transaction id if we delete destination transaction\n                ids.push(this.cache.find((t: Transaction): boolean => t.transfer.id === model.id).id);\n              }\n\n              return !ids.includes(transaction.id);\n            }\n\n            return transaction.id !== model.id && transaction.parentTransaction?.id !== model.id;\n          });\n\n          this.eventDispatcherService.dispatch(\n            new AppEvent<Transaction>(AppEventTypeEnum.TRANSACTION_DELETED, model)\n          );\n\n          this.updateCache();\n          this.transactionDeleted.emit(model);\n        })\n      );\n  }\n\n  /**\n   * Listen to EventDispatcherService event related to Depreciation changing\n   */\n  private listenDepreciationChange(): void {\n    // @TODO reset cache only when required (depreciation has transaction)\n    this.eventDispatcherService.on2(...Depreciation.getEventNames('post', 'delete')).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  private listenVehicleClaimChanges(): void {\n    this.eventDispatcherService.on2(...VehicleClaim.getEventNames('post', 'put')).subscribe(() => this.refreshCache());\n    this.eventDispatcherService.on([AppEventTypeEnum.VEHICLE_CLAIM_DETAILS_UPDATED]).subscribe(() => {\n      this.refreshCache();\n    })\n  }\n\n  private listenHomeOfficeClaim(): void {\n    this.eventDispatcherService.on2(...HomeOfficeClaim.getEventNames('post', 'put')).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  private listenHomeOfficeLog(): void {\n    this.eventDispatcherService.on2(...HomeOfficeLog.getEventNames('post', 'put', 'delete')).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  /**\n   * property forecast includes claimPercent which affects property related expenses\n   * @TODO Alex move to forecast api instead of property\n   */\n  private listenPropertyForecastUpdate(): void {\n    this.eventDispatcherService.on([AppEventTypeEnum.PROPERTY_UPDATED]).subscribe(() => {\n      this.refreshCache();\n    });\n  }\n\n  private replaceInCache(transactionToReplace: Transaction): void {\n    replace(this.cache, transactionToReplace);\n    this.updateCache();\n  }\n\n  private listenNotifications(): void {\n    this.eventDispatcherService.on(AppEventTypeEnum.NOTIFICATION_ADDED).subscribe((notification: Notification) => {\n      const events = [\n        UserEventTypeTypeEnum.INVOICE_AUTO_ALLOCATED,\n        UserEventTypeTypeEnum.INVOICE_AUTO_PART_ALLOCATED,\n        UserEventTypeTypeEnum.RULES_AUTO_ALLOCATED\n      ];\n\n      if (!notification.isRead && events.includes(notification.eventType)) {\n        // @TODO TT-3826 move to unified mercure event\n        setTimeout(() => {\n          this.refreshCache();\n        }, 3000)\n      }\n    });\n  }\n}\n"]}
@@ -6144,7 +6144,8 @@ class PropertyEquityChartData {
6144
6144
  * @private
6145
6145
  */
6146
6146
  getLoanBalanceByYear(property, year, bankAccounts, loans) {
6147
- return bankAccounts.items.reduce((totalSum, bankAccount) => totalSum + (bankAccount.getPropertyPercentage(property.id) * (loans.getByBankAccountId(bankAccount.id)?.getLastPaymentByYear(year)?.totalOwed || 0)), 0);
6147
+ const offset = bankAccounts.filterBy('type', BankAccountTypeEnum.OFFSET).getPropertyBalanceAmount(property.id);
6148
+ return bankAccounts.items.reduce((totalSum, bankAccount) => totalSum + (bankAccount.getPropertyPercentage(property.id) * (loans.getByBankAccountId(bankAccount.id)?.getLastPaymentByYear(year)?.totalOwed || 0)), 0) - offset;
6148
6149
  }
6149
6150
  getMarketValueForPropertyByYear(property, year) {
6150
6151
  if (new FinancialYear(property.contractDate).year > year) {
@@ -15559,14 +15560,7 @@ class TransactionService extends RestService {
15559
15560
  this.listenNotifications();
15560
15561
  }
15561
15562
  get(path = this.apiUrl) {
15562
- return super.get(path).pipe(map(transactions => {
15563
- transactions.forEach((transaction) => {
15564
- transaction.transactions = transactions
15565
- // get list of child transactions
15566
- .filter((t) => t.parentTransaction && t.parentTransaction.id === transaction.id);
15567
- });
15568
- return transactions;
15569
- }));
15563
+ return super.get(path).pipe(map(transactions => this.groupByParent(transactions)));
15570
15564
  }
15571
15565
  getCurrentYear() {
15572
15566
  return super.get().pipe(map((transactions) => new TransactionCollection(transactions).filterByFinancialYear('date').toArray()));
@@ -15605,7 +15599,15 @@ class TransactionService extends RestService {
15605
15599
  return this.fetch(`${this.environment.apiV2}/properties/${propertyId}/holding-costs`, false);
15606
15600
  }
15607
15601
  getWithoutTaxFreeProperty() {
15608
- return this.fetch(`${this.apiUrl}?excludeTaxFreePropertyCategories`, false);
15602
+ return this.fetch(`${this.apiUrl}?excludeTaxFreePropertyCategories`, false).pipe(map(transactions => this.groupByParent(transactions)));
15603
+ }
15604
+ groupByParent(transactions) {
15605
+ transactions.forEach((transaction) => {
15606
+ transaction.transactions = transactions
15607
+ // get list of child transactions
15608
+ .filter((t) => t.parentTransaction && t.parentTransaction.id === transaction.id);
15609
+ });
15610
+ return transactions;
15609
15611
  }
15610
15612
  /**
15611
15613
  * get list of taxable transactions with tank type 'Work'