taxtank-core 0.30.127 → 0.30.129

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.
@@ -132,8 +132,26 @@ export class BankAccountCollection extends Collection {
132
132
  getPropertyBalanceAmount(propertyId) {
133
133
  return this.items.reduce((sum, bankAccount) => sum + bankAccount.getPropertyBalanceAmount(propertyId), 0);
134
134
  }
135
- getMonthlyRepaymentAmount(propertyId) {
136
- return this.getLoanAccounts().reduce((sum, bankAccount) => sum + bankAccount.loan.monthlyRepaymentAmount * bankAccount.getPropertyPercentage(propertyId), 0);
135
+ get propertiesBalanceAmount() {
136
+ let balanceAmount = 0;
137
+ this.items.forEach((bankAccount) => {
138
+ bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {
139
+ balanceAmount += Math.abs(bankAccount.getPropertyBalanceAmount(bankAccountProperty.property.id));
140
+ });
141
+ });
142
+ return balanceAmount;
143
+ }
144
+ get propertiesMonthlyRepaymentAmount() {
145
+ let repaymentAmount = 0;
146
+ this.getLoanAccounts().items.forEach((bankAccount) => {
147
+ bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {
148
+ repaymentAmount += bankAccount.getMonthlyRepaymentAmount(bankAccountProperty.property.id);
149
+ });
150
+ });
151
+ return repaymentAmount;
152
+ }
153
+ getPropertyMonthlyRepaymentAmount(propertyId) {
154
+ return this.getLoanAccounts().reduce((sum, bankAccount) => sum + bankAccount.getMonthlyRepaymentAmount(propertyId), 0);
137
155
  }
138
156
  getLVR(property) {
139
157
  return this.getPropertyBalanceAmount(property.id) / property.currentYearForecast.marketValue;
@@ -142,5 +160,8 @@ export class BankAccountCollection extends Collection {
142
160
  // TODO Alex/Vik: maybe we should get it from jwtToken or think about some localStorageService?
143
161
  return this.filter((bankAccount) => bankAccount.isOwner(+localStorage.getItem('userId')));
144
162
  }
163
+ getActiveLoanAccountsByProperties(ids) {
164
+ return this.getActiveBankAccounts().getLoanAccounts().getByPropertiesIds(ids);
165
+ }
145
166
  }
146
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bank-account.collection.js","sourceRoot":"","sources":["../../../../../projects/tt-core/src/lib/collections/bank-account.collection.ts"],"names":[],"mappings":"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","sourcesContent":["import { Collection } from './collection';\nimport { BankAccountTypeEnum } from '../db/Enums';\nimport { TankTypeEnum } from '../db/Enums/tank-type.enum';\nimport { BankAccount, BankAccountProperty, Property, TYPE_LOAN } from '../models';\nimport flatten from 'lodash/flatten';\nimport uniqBy from 'lodash/uniqBy';\nimport uniq from 'lodash/uniq';\nimport concat from 'lodash/concat';\nimport { CollectionDictionary } from './collection-dictionary';\nimport { LoanCollection } from './loan';\n\n/**\n * Collection of bank accounts.\n */\nexport class BankAccountCollection extends Collection<BankAccount> {\n  /**\n   * get list of bank accounts with passed types\n   */\n  getByType(types: BankAccountTypeEnum | BankAccountTypeEnum[], isExclude = false): BankAccount[] {\n    // get types always as array (if only one passed)\n    const typesArray: BankAccountTypeEnum[] = concat(types);\n    return this.items.filter((bankAccount: BankAccount) => {\n      if (isExclude) {\n        return !typesArray.includes(bankAccount.type);\n      }\n      return typesArray.includes(bankAccount.type);\n    });\n  }\n\n  /**\n   * get amount of initial loans\n   */\n  getInitialLoanAmount(): number {\n    return this.getByType(TYPE_LOAN)\n      .reduce((sum: number, bankAccount: BankAccount) => sum += bankAccount.balances.length ? bankAccount.getOpeningBalance() : 0, 0);\n  }\n\n  /**\n   * get amount of current loans\n   */\n  getCurrentLoanAmount(): number {\n    return this.getByType(TYPE_LOAN)\n      .reduce((sum: number, bankAccount: BankAccount) => sum += bankAccount.currentBalance, 0);\n  }\n\n  /**\n   * get collection filtered by property id\n   */\n  getByPropertyId(id: number): BankAccountCollection {\n    return new BankAccountCollection(this.items.filter((bankAccount: BankAccount) => bankAccount.bankAccountProperties.find((bankAccountProperty: BankAccountProperty) => bankAccountProperty.property.id === id)));\n  }\n\n  groupByProperty(): CollectionDictionary<BankAccountCollection> {\n    const dictionary = new CollectionDictionary(new BankAccountCollection());\n\n    this.items.forEach((bankAccount) => {\n      bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {\n        dictionary.addTo(bankAccountProperty.property.id, bankAccount);\n      });\n    });\n\n    return dictionary;\n  }\n\n  /**\n   * get collection filtered by properties ids\n   */\n  getByPropertiesIds(ids: number[]): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount) => bankAccount.bankAccountProperties.find((bankAccountProperty: BankAccountProperty) => ids.includes(bankAccountProperty.property.id)))\n    );\n  }\n\n  /**\n   * get collection of active bank accounts\n   */\n  getActiveBankAccounts(): BankAccountCollection {\n    return new BankAccountCollection(this.items.filter((bankAccount: BankAccount) => bankAccount.isActive()));\n  }\n\n  /**\n   * get reduction of loan (reduction of principle) percentage\n   */\n  getPrincipleReductionPercent(): number {\n    const initialLoans: number = this.getInitialLoanAmount();\n    const currentLoans: number = this.getCurrentLoanAmount();\n\n    if (!initialLoans) {\n      return 0;\n    }\n\n    return (initialLoans - currentLoans) / initialLoans * 100;\n  }\n\n  /**\n   * Get collection of loan bank accounts\n   */\n  getLoanAccounts(): BankAccountCollection {\n    return new BankAccountCollection(this.getByType(TYPE_LOAN));\n  }\n\n  get loans(): LoanCollection {\n    return new LoanCollection(this.getLoanAccounts().map((bankAccount) => bankAccount.loan));\n  }\n\n  getSavingsAccounts(): BankAccountCollection {\n    return new BankAccountCollection(this.getByType(TYPE_LOAN, true));\n  }\n\n  getOpeningBalance(): number {\n    return this.items.reduce((sum: number, bankAccount: BankAccount): number => sum + bankAccount.getOpeningBalance(), 0);\n  }\n\n  get currentBalance(): number {\n    return this.sumBy('currentBalance');\n  }\n\n  get bankShortNames(): string {\n    return uniq(this.map((bankAccount) => bankAccount.bank.shortName)).join(', ');\n  }\n\n  /**\n   * Get Collection of bank accounts with property tank type\n   */\n  getPropertyBankAccounts(): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount): boolean => bankAccount.isPropertyTank())\n    );\n  }\n\n  /**\n   * Get Collection of bank accounts with work tank type\n   */\n  getWorkBankAccounts(): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount): boolean => bankAccount.isWorkTank())\n    );\n  }\n\n  /**\n   * Get Collection of bank accounts by tank type\n   */\n  getByTankType(tankType: TankTypeEnum): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount): boolean => bankAccount.tankType === tankType)\n    );\n  }\n\n  /**\n   * Get list of all bank account properties\n   */\n  getBankAccountPropertiesList(): BankAccountProperty[] {\n    return flatten(\n      this.items.map((bankAccount: BankAccount): BankAccountProperty[] => bankAccount.bankAccountProperties)\n    );\n  }\n\n  /**\n   * Get lis of unique properties from all bank accounts properties\n   */\n  getProperties(): Property[] {\n    return uniqBy(\n      this.getBankAccountPropertiesList().map((bankAccountProperty: BankAccountProperty) => bankAccountProperty.property),\n      'id'\n    );\n  }\n\n  /**\n   * @TODO Alex not efficient, filter bankAccounts by property first\n   */\n  getPropertyBalanceAmount(propertyId: number): number {\n    return this.items.reduce((sum: number, bankAccount: BankAccount): number => sum + bankAccount.getPropertyBalanceAmount(propertyId), 0);\n  }\n\n  getMonthlyRepaymentAmount(propertyId: number): number {\n    return this.getLoanAccounts().reduce((sum: number, bankAccount: BankAccount): number =>\n      sum + bankAccount.loan.monthlyRepaymentAmount * bankAccount.getPropertyPercentage(propertyId), 0\n    );\n  }\n\n  getLVR(property: Property): number {\n    return this.getPropertyBalanceAmount(property.id) / property.currentYearForecast.marketValue;\n  }\n\n  getOwn(): this {\n    // TODO Alex/Vik: maybe we should get it from jwtToken or think about some localStorageService?\n    return this.filter((bankAccount: BankAccount) => bankAccount.isOwner(+localStorage.getItem('userId')));\n  }\n}\n"]}
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bank-account.collection.js","sourceRoot":"","sources":["../../../../../projects/tt-core/src/lib/collections/bank-account.collection.ts"],"names":[],"mappings":"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","sourcesContent":["import { Collection } from './collection';\nimport { BankAccountTypeEnum } from '../db/Enums';\nimport { TankTypeEnum } from '../db/Enums/tank-type.enum';\nimport { BankAccount, BankAccountProperty, Property, TYPE_LOAN } from '../models';\nimport flatten from 'lodash/flatten';\nimport uniqBy from 'lodash/uniqBy';\nimport uniq from 'lodash/uniq';\nimport concat from 'lodash/concat';\nimport { CollectionDictionary } from './collection-dictionary';\nimport { LoanCollection } from './loan';\n\n/**\n * Collection of bank accounts.\n */\nexport class BankAccountCollection extends Collection<BankAccount> {\n  /**\n   * get list of bank accounts with passed types\n   */\n  getByType(types: BankAccountTypeEnum | BankAccountTypeEnum[], isExclude = false): BankAccount[] {\n    // get types always as array (if only one passed)\n    const typesArray: BankAccountTypeEnum[] = concat(types);\n    return this.items.filter((bankAccount: BankAccount) => {\n      if (isExclude) {\n        return !typesArray.includes(bankAccount.type);\n      }\n      return typesArray.includes(bankAccount.type);\n    });\n  }\n\n  /**\n   * get amount of initial loans\n   */\n  getInitialLoanAmount(): number {\n    return this.getByType(TYPE_LOAN)\n      .reduce((sum: number, bankAccount: BankAccount) => sum += bankAccount.balances.length ? bankAccount.getOpeningBalance() : 0, 0);\n  }\n\n  /**\n   * get amount of current loans\n   */\n  getCurrentLoanAmount(): number {\n    return this.getByType(TYPE_LOAN)\n      .reduce((sum: number, bankAccount: BankAccount) => sum += bankAccount.currentBalance, 0);\n  }\n\n  /**\n   * get collection filtered by property id\n   */\n  getByPropertyId(id: number): BankAccountCollection {\n    return new BankAccountCollection(this.items.filter((bankAccount: BankAccount) => bankAccount.bankAccountProperties.find((bankAccountProperty: BankAccountProperty) => bankAccountProperty.property.id === id)));\n  }\n\n  groupByProperty(): CollectionDictionary<BankAccountCollection> {\n    const dictionary = new CollectionDictionary(new BankAccountCollection());\n\n    this.items.forEach((bankAccount) => {\n      bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {\n        dictionary.addTo(bankAccountProperty.property.id, bankAccount);\n      });\n    });\n\n    return dictionary;\n  }\n\n  /**\n   * get collection filtered by properties ids\n   */\n  getByPropertiesIds(ids: number[]): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount) => bankAccount.bankAccountProperties.find((bankAccountProperty: BankAccountProperty) => ids.includes(bankAccountProperty.property.id)))\n    );\n  }\n\n  /**\n   * get collection of active bank accounts\n   */\n  getActiveBankAccounts(): BankAccountCollection {\n    return new BankAccountCollection(this.items.filter((bankAccount: BankAccount) => bankAccount.isActive()));\n  }\n\n  /**\n   * get reduction of loan (reduction of principle) percentage\n   */\n  getPrincipleReductionPercent(): number {\n    const initialLoans: number = this.getInitialLoanAmount();\n    const currentLoans: number = this.getCurrentLoanAmount();\n\n    if (!initialLoans) {\n      return 0;\n    }\n\n    return (initialLoans - currentLoans) / initialLoans * 100;\n  }\n\n  /**\n   * Get collection of loan bank accounts\n   */\n  getLoanAccounts(): BankAccountCollection {\n    return new BankAccountCollection(this.getByType(TYPE_LOAN));\n  }\n\n  get loans(): LoanCollection {\n    return new LoanCollection(this.getLoanAccounts().map((bankAccount) => bankAccount.loan));\n  }\n\n  getSavingsAccounts(): BankAccountCollection {\n    return new BankAccountCollection(this.getByType(TYPE_LOAN, true));\n  }\n\n  getOpeningBalance(): number {\n    return this.items.reduce((sum: number, bankAccount: BankAccount): number => sum + bankAccount.getOpeningBalance(), 0);\n  }\n\n  get currentBalance(): number {\n    return this.sumBy('currentBalance');\n  }\n\n  get bankShortNames(): string {\n    return uniq(this.map((bankAccount) => bankAccount.bank.shortName)).join(', ');\n  }\n\n  /**\n   * Get Collection of bank accounts with property tank type\n   */\n  getPropertyBankAccounts(): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount): boolean => bankAccount.isPropertyTank())\n    );\n  }\n\n  /**\n   * Get Collection of bank accounts with work tank type\n   */\n  getWorkBankAccounts(): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount): boolean => bankAccount.isWorkTank())\n    );\n  }\n\n  /**\n   * Get Collection of bank accounts by tank type\n   */\n  getByTankType(tankType: TankTypeEnum): BankAccountCollection {\n    return new BankAccountCollection(\n      this.items.filter((bankAccount: BankAccount): boolean => bankAccount.tankType === tankType)\n    );\n  }\n\n  /**\n   * Get list of all bank account properties\n   */\n  getBankAccountPropertiesList(): BankAccountProperty[] {\n    return flatten(\n      this.items.map((bankAccount: BankAccount): BankAccountProperty[] => bankAccount.bankAccountProperties)\n    );\n  }\n\n  /**\n   * Get lis of unique properties from all bank accounts properties\n   */\n  getProperties(): Property[] {\n    return uniqBy(\n      this.getBankAccountPropertiesList().map((bankAccountProperty: BankAccountProperty) => bankAccountProperty.property),\n      'id'\n    );\n  }\n\n  /**\n   * @TODO Alex not efficient, filter bankAccounts by property first\n   */\n  getPropertyBalanceAmount(propertyId: number): number {\n    return this.items.reduce((sum: number, bankAccount: BankAccount): number => sum + bankAccount.getPropertyBalanceAmount(propertyId), 0);\n  }\n\n  get propertiesBalanceAmount(): number {\n    let balanceAmount = 0;\n    this.items.forEach((bankAccount) => {\n      bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {\n        balanceAmount += Math.abs(bankAccount.getPropertyBalanceAmount(bankAccountProperty.property.id));\n      });\n    });\n\n    return balanceAmount;\n  }\n\n  get propertiesMonthlyRepaymentAmount(): number {\n    let repaymentAmount = 0;\n    this.getLoanAccounts().items.forEach((bankAccount) => {\n      bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {\n        repaymentAmount += bankAccount.getMonthlyRepaymentAmount(bankAccountProperty.property.id);\n      });\n    });\n\n    return repaymentAmount;\n  }\n\n  getPropertyMonthlyRepaymentAmount(propertyId: number): number {\n    return this.getLoanAccounts().reduce((sum: number, bankAccount: BankAccount): number =>\n      sum + bankAccount.getMonthlyRepaymentAmount(propertyId), 0\n    );\n  }\n\n  getLVR(property: Property): number {\n    return this.getPropertyBalanceAmount(property.id) / property.currentYearForecast.marketValue;\n  }\n\n  getOwn(): this {\n    // TODO Alex/Vik: maybe we should get it from jwtToken or think about some localStorageService?\n    return this.filter((bankAccount: BankAccount) => bankAccount.isOwner(+localStorage.getItem('userId')));\n  }\n\n  getActiveLoanAccountsByProperties(ids: number[]): BankAccountCollection {\n    return this.getActiveBankAccounts().getLoanAccounts().getByPropertiesIds(ids);\n  }\n}\n"]}
@@ -2,6 +2,7 @@ import get from 'lodash/get';
2
2
  import flatten from 'lodash/flatten';
3
3
  import hasIn from 'lodash/hasIn';
4
4
  import intersection from 'lodash/intersection';
5
+ import uniqBy from 'lodash/uniqBy';
5
6
  /**
6
7
  * List of collections grouped by passed property
7
8
  */
@@ -45,13 +46,13 @@ export class CollectionDictionary {
45
46
  return this.items[key] ? this.items[key] : this.createCollection([]);
46
47
  }
47
48
  /**
48
- * Join several collections by ids
49
+ * Join several collections by ids, return collection of uniq models (skip duplicates)
49
50
  */
50
51
  merge(keys) {
51
52
  if (!this.length) {
52
53
  return this.createCollection([]);
53
54
  }
54
- return this.createCollection(flatten(keys.map((key) => this.get(key.toString()).items)));
55
+ return this.createCollection(uniqBy(flatten(keys.map((key) => this.get(key.toString()).items)), 'id'));
55
56
  }
56
57
  /**
57
58
  * Create instance of collection
@@ -104,4 +105,4 @@ export class CollectionDictionary {
104
105
  });
105
106
  }
106
107
  }
107
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collection-dictionary.js","sourceRoot":"","sources":["../../../../../projects/tt-core/src/lib/collections/collection-dictionary.ts"],"names":[],"mappings":"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","sourcesContent":["import get from 'lodash/get';\nimport flatten from 'lodash/flatten';\nimport hasIn from 'lodash/hasIn';\nimport { Collection as BaseCollection } from './collection';\nimport {AbstractModel} from '../db/Models';\nimport intersection from 'lodash/intersection';\n\n/**\n * List of collections grouped by passed property\n */\nexport class CollectionDictionary<Collection extends BaseCollection<AbstractModel>> {\n  /**\n   * List of grouped collections\n   */\n  items: { [key: string]: Collection } = {};\n\n  /**\n   * Constructor function of passed collection for creating new instances\n   */\n  private collectionConstructor: new (items: object[]) => Collection;\n\n  constructor(collection: Collection, path: string = 'id') {\n    this.collectionConstructor = collection.constructor as new (items: object[]) => Collection;\n    // group if provided path was found in the 1st collection's item\n    if (collection.length && hasIn(collection.items[0], path.split('.')[0])) {\n      this.groupItems(collection, path);\n    }\n  }\n\n  /**\n   * List of collections keys\n   */\n  get keys(): string[] {\n    return Object.keys(this.items);\n  }\n\n  /**\n   * push new elements in existing item\n   */\n  addTo(key: string | number, ...models: AbstractModel[]): void {\n    if (!this.items[key]) {\n      this.add(key);\n    }\n\n    this.items[key].push(...models);\n  }\n\n  /**\n   * Add new item by key\n   */\n  add(key: string | number, value: Collection = this.createCollection([])) {\n    this.items[key] = value;\n  }\n\n  /**\n   * Get collection by key or return empty collection if key does not exist\n   */\n  get(key: string | number): Collection {\n    return this.items[key] ? this.items[key] : this.createCollection([]);\n  }\n\n  /**\n   * Join several collections by ids\n   */\n  merge(keys: number[]): Collection {\n    if (!this.length) {\n      return this.createCollection([]);\n    }\n\n    return this.createCollection(\n      flatten(\n        keys.map((key: number): object[] => this.get(key.toString()).items)\n      )\n    );\n  }\n\n  /**\n   * Create instance of collection\n   */\n  createCollection(items: object[] = []): Collection {\n    return new this.collectionConstructor(items);\n  }\n\n  get length(): number {\n    return this.keys.length;\n  }\n\n  /**\n   * @Todo find a better solution to get list of the sorted keys\n   * Get array of the \"keys\" by provided order\n   */\n  getSortedKeys(isDesc = false): string[] {\n    return this.keys.sort((a: any, b: any): number => {\n      if (a > b) { return !isDesc ? 1 : -1; }\n      if (a < b) { return !isDesc ? -1 : 1; }\n      return 0;\n    });\n  }\n\n  keysIntersection(keys: string[] | number[]): string[] {\n    return intersection(keys.map((key: string | number) => key.toString()), this.keys);\n  }\n\n  /**\n   * Group collection items by passed path into items object\n   */\n  private groupItems(collection: Collection, path: string): void {\n    // Create empty initial object for groups\n    const obj: { [key: string]: Array<object> } = {};\n\n    // Group collection items\n    collection.items.forEach((item: object): void => {\n      let key: string = get(item, path);\n\n      // if object does not have property for grouping it will be grouped as 'other'\n      if (key === undefined) {\n        key = 'other';\n      }\n\n      if (!obj[key]) {\n        obj[key] = [];\n      }\n\n      obj[key].push(item);\n    });\n\n    // Create collections from groups\n    Object.keys(obj).forEach((key: string): void => {\n      this.items[key] = this.createCollection(obj[key]);\n    });\n  }\n}\n"]}
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collection-dictionary.js","sourceRoot":"","sources":["../../../../../projects/tt-core/src/lib/collections/collection-dictionary.ts"],"names":[],"mappings":"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","sourcesContent":["import get from 'lodash/get';\nimport flatten from 'lodash/flatten';\nimport hasIn from 'lodash/hasIn';\nimport { Collection as BaseCollection } from './collection';\nimport {AbstractModel} from '../db/Models';\nimport intersection from 'lodash/intersection';\nimport uniqBy from 'lodash/uniqBy';\n\n/**\n * List of collections grouped by passed property\n */\nexport class CollectionDictionary<Collection extends BaseCollection<AbstractModel>> {\n  /**\n   * List of grouped collections\n   */\n  items: { [key: string]: Collection } = {};\n\n  /**\n   * Constructor function of passed collection for creating new instances\n   */\n  private collectionConstructor: new (items: object[]) => Collection;\n\n  constructor(collection: Collection, path: string = 'id') {\n    this.collectionConstructor = collection.constructor as new (items: object[]) => Collection;\n    // group if provided path was found in the 1st collection's item\n    if (collection.length && hasIn(collection.items[0], path.split('.')[0])) {\n      this.groupItems(collection, path);\n    }\n  }\n\n  /**\n   * List of collections keys\n   */\n  get keys(): string[] {\n    return Object.keys(this.items);\n  }\n\n  /**\n   * push new elements in existing item\n   */\n  addTo(key: string | number, ...models: AbstractModel[]): void {\n    if (!this.items[key]) {\n      this.add(key);\n    }\n\n    this.items[key].push(...models);\n  }\n\n  /**\n   * Add new item by key\n   */\n  add(key: string | number, value: Collection = this.createCollection([])) {\n    this.items[key] = value;\n  }\n\n  /**\n   * Get collection by key or return empty collection if key does not exist\n   */\n  get(key: string | number): Collection {\n    return this.items[key] ? this.items[key] : this.createCollection([]);\n  }\n\n  /**\n   * Join several collections by ids, return collection of uniq models (skip duplicates)\n   */\n  merge(keys: number[] | string[]): Collection {\n    if (!this.length) {\n      return this.createCollection([]);\n    }\n\n    return this.createCollection(\n      uniqBy(flatten(keys.map((key: number | string): object[] => this.get(key.toString()).items)), 'id')\n    );\n  }\n\n  /**\n   * Create instance of collection\n   */\n  createCollection(items: object[] = []): Collection {\n    return new this.collectionConstructor(items);\n  }\n\n  get length(): number {\n    return this.keys.length;\n  }\n\n  /**\n   * @Todo find a better solution to get list of the sorted keys\n   * Get array of the \"keys\" by provided order\n   */\n  getSortedKeys(isDesc = false): string[] {\n    return this.keys.sort((a: any, b: any): number => {\n      if (a > b) { return !isDesc ? 1 : -1; }\n      if (a < b) { return !isDesc ? -1 : 1; }\n      return 0;\n    });\n  }\n\n  keysIntersection(keys: string[] | number[]): string[] {\n    return intersection(keys.map((key: string | number) => key.toString()), this.keys);\n  }\n\n  /**\n   * Group collection items by passed path into items object\n   */\n  private groupItems(collection: Collection, path: string): void {\n    // Create empty initial object for groups\n    const obj: { [key: string]: Array<object> } = {};\n\n    // Group collection items\n    collection.items.forEach((item: object): void => {\n      let key: string = get(item, path);\n\n      // if object does not have property for grouping it will be grouped as 'other'\n      if (key === undefined) {\n        key = 'other';\n      }\n\n      if (!obj[key]) {\n        obj[key] = [];\n      }\n\n      obj[key].push(item);\n    });\n\n    // Create collections from groups\n    Object.keys(obj).forEach((key: string): void => {\n      this.items[key] = this.createCollection(obj[key]);\n    });\n  }\n}\n"]}
@@ -1,5 +1,5 @@
1
1
  import { Collection } from '../collection';
2
- import { FinancialYear } from '../../models/financial-year/financial-year';
2
+ import { FinancialYear } from '../../models';
3
3
  import uniqBy from 'lodash/uniqBy';
4
4
  export class PropertyCollection extends Collection {
5
5
  /**
@@ -108,5 +108,8 @@ export class PropertyCollection extends Collection {
108
108
  newItems.unshift(...bestProperties);
109
109
  return this.create(newItems);
110
110
  }
111
+ get netIncome() {
112
+ return this.sumBy('currentYearForecast.income') + this.sumBy('currentYearForecast.expense');
113
+ }
111
114
  }
112
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property.collection.js","sourceRoot":"","sources":["../../../../../../projects/tt-core/src/lib/collections/property/property.collection.ts"],"names":[],"mappings":"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","sourcesContent":["import { Collection } from '../collection';\nimport { Property } from '../../models/property/property';\nimport { FinancialYear } from '../../models/financial-year/financial-year';\nimport { PropertyCategory } from '../../models/property/property-category';\nimport uniqBy from 'lodash/uniqBy';\nimport { DepreciationCollection } from '../depreciation.collection';\nimport { TransactionCollection } from '../transaction/transaction.collection';\nimport { CollectionDictionary } from '../collection-dictionary';\nimport { PropertySale } from '../../models/property/property-sale/property-sale';\n\nexport class PropertyCollection extends Collection<Property> {\n  /**\n   * Get new property collection filtered by category id\n   * @param id id of category for filter\n   */\n  getByCategoryId(id: number): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.category.id === id)\n    );\n  }\n\n  /**\n   * Get new property collection filtered by active status\n   */\n  getActiveProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.isActive)\n    );\n  }\n\n  getCreatedProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.isOwn())\n    );\n  }\n\n  /**\n   * Get new property collection filtered by shared\n   */\n  getSharedProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => !property.isOwn())\n    );\n  }\n\n  /**\n   * Properties that are taxed and will be included in reports (Tax summary, My tax report, e.t.c.)\n   */\n  getTaxInclusive(): this {\n    return this.create(\n      this.items.filter((property: Property) => property.category.isTaxInclusive)\n    );\n  }\n\n  getUnsold(): this {\n    return this.create(\n      this.items.filter((property: Property) => !property.isSold())\n    );\n  }\n\n  /**\n   * Get total purchase price for all properties in the collection\n   */\n  get purchasePrice(): number {\n    return this.sumBy('purchasePrice');\n  }\n\n  get growthPercent(): number {\n    return this.sumBy('growthPercent');\n  }\n\n  get marketValue(): number {\n    return this.sumBy('marketValue');\n  }\n\n  get firstForecastYear(): number {\n    return this.items.reduce((min: number, property: Property): number => {\n      const current: number = property.firstForecastYear;\n      return min > current ? current : min;\n    }, new FinancialYear().year);\n  }\n\n  get marketValueGrowth(): number {\n    return (this.marketValue - this.purchasePrice) / this.purchasePrice;\n  }\n\n  /**\n   * list of properties\n   */\n  getCGTApplicable(): this {\n    return this.create(\n      this.items.filter((property: Property) => property.isCGTApplicable())\n    );\n  }\n\n  getOwnerOccupiedProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.category.isOwnerOccupied())\n    );\n  }\n\n  get earliestContractDate(): Date {\n    return this.items.reduce((min: Date, property: Property): Date => min < property.contractDate ? min : property.contractDate, new FinancialYear(new Date()).startDate);\n  }\n\n  /**\n   * Get list of unique property categories from collection\n   */\n  getCategories(): PropertyCategory[] {\n    return uniqBy(this.items.map((property: Property): PropertyCategory => property.category), 'id');\n  }\n\n  /**\n   * Get property with the highest growth percent\n   */\n  getBestPerformanceGrowthProperty(): Property {\n    return this.items.reduce((max: Property, current: Property): Property => max.growthPercent < current.growthPercent ? current : max, this.first);\n  }\n\n  /**\n   * Get property with the lowest tax position\n   */\n  getBestPerformanceTaxProperty(transactions: TransactionCollection, depreciations: DepreciationCollection): Property {\n    const transactionsByProperty: CollectionDictionary<TransactionCollection> = transactions.groupBy('property.id');\n    const depreciationsByProperty: CollectionDictionary<DepreciationCollection> = depreciations.groupBy('property.id');\n\n    return this.items.reduce((min: Property, current: Property): Property => {\n      const minTaxPosition: number = min.getTaxPosition(transactionsByProperty.get(min.id), depreciationsByProperty.get(min.id));\n      const currentTaxPosition: number = current.getTaxPosition(transactionsByProperty.get(current.id), depreciationsByProperty.get(current.id));\n\n      return minTaxPosition > currentTaxPosition ? current : min;\n    }, this.first);\n  }\n\n  /**\n   * Show best performance properties first\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/217677997/Property+Tank+Dashboard\n   */\n  sortByBestPerformance(transactions: TransactionCollection, depreciations: DepreciationCollection): this {\n    const activeProperties: PropertyCollection = this.getActiveProperties();\n\n    // nothing to sort when no active properties\n    if (!activeProperties.length) {\n      return this;\n    }\n\n    const bestProperties: Property[] = uniqBy(this.create([\n      activeProperties.getBestPerformanceGrowthProperty(),\n      activeProperties.getBestPerformanceTaxProperty(transactions, depreciations)\n    ]).toArray(), 'id');\n\n    const newItems: Property[] = this.diff(bestProperties).toArray();\n    newItems.unshift(...bestProperties);\n\n    return this.create(newItems);\n  }\n}\n"]}
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property.collection.js","sourceRoot":"","sources":["../../../../../../projects/tt-core/src/lib/collections/property/property.collection.ts"],"names":[],"mappings":"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","sourcesContent":["import { Collection } from '../collection';\nimport { Property, PropertyCategory, FinancialYear } from '../../models';\nimport uniqBy from 'lodash/uniqBy';\nimport { DepreciationCollection } from '../depreciation.collection';\nimport { TransactionCollection } from '../transaction';\nimport { CollectionDictionary } from '../collection-dictionary';\n\nexport class PropertyCollection extends Collection<Property> {\n  /**\n   * Get new property collection filtered by category id\n   * @param id id of category for filter\n   */\n  getByCategoryId(id: number): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.category.id === id)\n    );\n  }\n\n  /**\n   * Get new property collection filtered by active status\n   */\n  getActiveProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.isActive)\n    );\n  }\n\n  getCreatedProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.isOwn())\n    );\n  }\n\n  /**\n   * Get new property collection filtered by shared\n   */\n  getSharedProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => !property.isOwn())\n    );\n  }\n\n  /**\n   * Properties that are taxed and will be included in reports (Tax summary, My tax report, e.t.c.)\n   */\n  getTaxInclusive(): this {\n    return this.create(\n      this.items.filter((property: Property) => property.category.isTaxInclusive)\n    );\n  }\n\n  getUnsold(): this {\n    return this.create(\n      this.items.filter((property: Property) => !property.isSold())\n    );\n  }\n\n  /**\n   * Get total purchase price for all properties in the collection\n   */\n  get purchasePrice(): number {\n    return this.sumBy('purchasePrice');\n  }\n\n  get growthPercent(): number {\n    return this.sumBy('growthPercent');\n  }\n\n  get marketValue(): number {\n    return this.sumBy('marketValue');\n  }\n\n  get firstForecastYear(): number {\n    return this.items.reduce((min: number, property: Property): number => {\n      const current: number = property.firstForecastYear;\n      return min > current ? current : min;\n    }, new FinancialYear().year);\n  }\n\n  get marketValueGrowth(): number {\n    return (this.marketValue - this.purchasePrice) / this.purchasePrice;\n  }\n\n  /**\n   * list of properties\n   */\n  getCGTApplicable(): this {\n    return this.create(\n      this.items.filter((property: Property) => property.isCGTApplicable())\n    );\n  }\n\n  getOwnerOccupiedProperties(): PropertyCollection {\n    return new PropertyCollection(\n      this.items.filter((property: Property): boolean => property.category.isOwnerOccupied())\n    );\n  }\n\n  get earliestContractDate(): Date {\n    return this.items.reduce((min: Date, property: Property): Date => min < property.contractDate ? min : property.contractDate, new FinancialYear(new Date()).startDate);\n  }\n\n  /**\n   * Get list of unique property categories from collection\n   */\n  getCategories(): PropertyCategory[] {\n    return uniqBy(this.items.map((property: Property): PropertyCategory => property.category), 'id');\n  }\n\n  /**\n   * Get property with the highest growth percent\n   */\n  getBestPerformanceGrowthProperty(): Property {\n    return this.items.reduce((max: Property, current: Property): Property => max.growthPercent < current.growthPercent ? current : max, this.first);\n  }\n\n  /**\n   * Get property with the lowest tax position\n   */\n  getBestPerformanceTaxProperty(transactions: TransactionCollection, depreciations: DepreciationCollection): Property {\n    const transactionsByProperty: CollectionDictionary<TransactionCollection> = transactions.groupBy('property.id');\n    const depreciationsByProperty: CollectionDictionary<DepreciationCollection> = depreciations.groupBy('property.id');\n\n    return this.items.reduce((min: Property, current: Property): Property => {\n      const minTaxPosition: number = min.getTaxPosition(transactionsByProperty.get(min.id), depreciationsByProperty.get(min.id));\n      const currentTaxPosition: number = current.getTaxPosition(transactionsByProperty.get(current.id), depreciationsByProperty.get(current.id));\n\n      return minTaxPosition > currentTaxPosition ? current : min;\n    }, this.first);\n  }\n\n  /**\n   * Show best performance properties first\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/217677997/Property+Tank+Dashboard\n   */\n  sortByBestPerformance(transactions: TransactionCollection, depreciations: DepreciationCollection): this {\n    const activeProperties: PropertyCollection = this.getActiveProperties();\n\n    // nothing to sort when no active properties\n    if (!activeProperties.length) {\n      return this;\n    }\n\n    const bestProperties: Property[] = uniqBy(this.create([\n      activeProperties.getBestPerformanceGrowthProperty(),\n      activeProperties.getBestPerformanceTaxProperty(transactions, depreciations)\n    ]).toArray(), 'id');\n\n    const newItems: Property[] = this.diff(bestProperties).toArray();\n    newItems.unshift(...bestProperties);\n\n    return this.create(newItems);\n  }\n\n  get netIncome(): number {\n    return this.sumBy('currentYearForecast.income') + this.sumBy('currentYearForecast.expense');\n  }\n}\n"]}
@@ -117,6 +117,9 @@ export class BankAccount extends BankAccountBase {
117
117
  getPropertyBalanceAmount(propertyId) {
118
118
  return this.currentBalance * this.getPropertyPercentage(propertyId);
119
119
  }
120
+ getMonthlyRepaymentAmount(propertyId) {
121
+ return this.loan.monthlyRepaymentAmount * this.getPropertyPercentage(propertyId);
122
+ }
120
123
  /**
121
124
  * Check if bank account is active (has 'active' status or hasn't been paid out/refinanced)
122
125
  */
@@ -166,4 +169,4 @@ __decorate([
166
169
  __decorate([
167
170
  Transform(({ value }) => value === 4 ? BankAccountTypeEnum.LOAN : value)
168
171
  ], BankAccount.prototype, "type", void 0);
169
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bank-account.js","sourceRoot":"","sources":["../../../../../../projects/tt-core/src/lib/models/bank/bank-account.ts"],"names":[],"mappings":"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","sourcesContent":["import { BankAccount as BankAccountBase } from '../../db/Models/bank/bank-account';\nimport { BankAccountTypeEnum, BankAccountStatusEnum } from '../../db/Enums';\nimport { Loan } from '../loan';\nimport { TankTypeEnum } from '../../db/Enums/tank-type.enum';\nimport { BankAccountBalance } from '../../db/Models';\nimport { FinancialYear } from '../financial-year';\nimport { BankAccountProperty } from './bank-account-property';\nimport { BankConnection } from './bank-connection';\nimport { Transform, Type } from 'class-transformer';\nimport { SoleBusinessAllocation } from '../sole';\nimport { Bank } from './bank';\n\nexport class BankAccount extends BankAccountBase {\n  // list of properties related with this bank account\n  @Type(() => BankAccountProperty)\n  bankAccountProperties: BankAccountProperty[];\n\n  @Type(() => SoleBusinessAllocation)\n  businessAllocations: SoleBusinessAllocation[]\n\n  @Type(() => BankAccountBalance)\n  balances: BankAccountBalance[];\n\n  // bank account's loan\n  @Type(() => Loan)\n  loan: Loan;\n\n  @Type(() => BankConnection)\n  bankConnection: BankConnection;\n\n  // @TODO Alex: temp hack: remove when TT-2107 ready. Backend shoud return Mortgage type as Loan type\n  @Transform(({ value }) => value === 4 ? BankAccountTypeEnum.LOAN : value)\n  type: BankAccountTypeEnum;\n\n  get bank(): Bank {\n    return this.bankConnection.bank;\n  }\n\n  get bsb(): string {\n    return this.accountNumber.slice(0, 6);\n  }\n\n  get number(): string {\n    return this.accountNumber.trim().slice(6);\n  }\n\n  /**\n   * Get last digits of account number\n   */\n  get partialAccountNumber(): string {\n    return `xxxx${this.accountNumber.slice(-4)}`;\n  }\n\n  /**\n   * Get current opening balance amount\n   */\n  getOpeningBalance(): number {\n    return this.getCurrentYearBalance()?.openingBalance || 0;\n  }\n\n  /**\n   * Get bank account balance for current financial year\n   */\n  getCurrentYearBalance(): BankAccountBalance {\n    const currentFinYear: number = new FinancialYear().year;\n    return this.balances.find((balance: BankAccountBalance): boolean => balance.financialYear === currentFinYear);\n  }\n\n  /**\n   * check if bank account type is one of savings accounts\n   */\n  isSavingsAccount(): boolean {\n    return this.type === BankAccountTypeEnum.INVESTMENT ||\n      this.type === BankAccountTypeEnum.TERM_DEPOSIT ||\n      this.type === BankAccountTypeEnum.SAVINGS;\n  }\n\n  /**\n   * check if bank account type is Loan\n   */\n  isLoan(): boolean {\n    return this.type === BankAccountTypeEnum.LOAN;\n  }\n\n  /**\n   * check if bank account type is Credit card\n   */\n  isCreditCard(): boolean {\n    return this.type === BankAccountTypeEnum.CREDIT_CARD;\n  }\n\n  /**\n   * check if bank account related to work tank\n   */\n  isWorkTank(): boolean {\n    return !this.isPropertyTank() && !this.isSoleTank();\n  }\n\n  /**\n   * check if bank account related to work tank\n   */\n  isPropertyTank(): boolean {\n    return !!this.bankAccountProperties.length;\n  }\n\n  /**\n   * check if bank account related to sole tank\n   */\n  isSoleTank(): boolean {\n    return !!this.businessAllocations.length;\n  }\n\n  get tankType(): TankTypeEnum {\n    switch (true) {\n      case this.isPropertyTank():\n        return TankTypeEnum.PROPERTY;\n      case this.isSoleTank():\n        return TankTypeEnum.SOLE;\n      default:\n        return TankTypeEnum.WORK;\n    }\n  }\n\n  /**\n   * Get Bank account property by id\n   * @param id Id of property\n   */\n  getPropertyById(id: number): BankAccountProperty {\n    return this.bankAccountProperties.find((bankAccountProperty: BankAccountProperty): boolean => bankAccountProperty.property.id === id);\n  }\n\n  /**\n   * check if bank account is related with passed property by id\n   * @param propertyId Property id for checking\n   */\n  hasProperty(propertyId: number): boolean {\n    return !!this.getPropertyById(propertyId);\n  }\n\n  /**\n   * Get decimal percentage for property\n   * @param propertyId Id of property\n   */\n  getPropertyPercentage(propertyId: number): number {\n    if (!this.hasProperty(propertyId)) {\n      return 0;\n    }\n\n    return this.getPropertyById(propertyId).percent / 100;\n  }\n\n  /**\n   * Get balance amount for property by id\n   * @param propertyId Id of property\n   */\n  getPropertyBalanceAmount(propertyId: number): number {\n    return this.currentBalance * this.getPropertyPercentage(propertyId);\n  }\n\n  /**\n   * Check if bank account is active (has 'active' status or hasn't been paid out/refinanced)\n   */\n  isActive(): boolean {\n    return this.status === BankAccountStatusEnum.ACTIVE;\n  }\n\n  /**\n   * first import (transactions) of basiq accounts\n   */\n  isFirstImport(): boolean {\n    return !this.lastTransactionDate && !this.isManual;\n  }\n\n  /**\n   * Check if passed user id is owner of bank account\n   */\n  isOwner(userId: number): boolean {\n    return this.bankConnection.user.id === userId;\n  }\n\n  /**\n   * Loan is paid if it has no unallocated transactions and the bank balance is $0.\n   */\n  isLoanPaid(taxTankBalance: number): boolean {\n    const shouldPayoutLoanAccount: boolean = this.isActive() && this.isLoan() && taxTankBalance === 0;\n\n    // we don't check current balance for basiq accounts, because basiq doesn't provide us last transactions (can't do for closed accounts),\n    // that's why balance won't be 0 (updated by basiq)\n    if (shouldPayoutLoanAccount && this.isManual) {\n      return this.currentBalance === 0;\n    }\n\n    return shouldPayoutLoanAccount;\n  }\n}\n"]}
172
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bank-account.js","sourceRoot":"","sources":["../../../../../../projects/tt-core/src/lib/models/bank/bank-account.ts"],"names":[],"mappings":"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","sourcesContent":["import { BankAccount as BankAccountBase } from '../../db/Models/bank/bank-account';\nimport { BankAccountTypeEnum, BankAccountStatusEnum } from '../../db/Enums';\nimport { Loan } from '../loan';\nimport { TankTypeEnum } from '../../db/Enums/tank-type.enum';\nimport { BankAccountBalance } from '../../db/Models';\nimport { FinancialYear } from '../financial-year';\nimport { BankAccountProperty } from './bank-account-property';\nimport { BankConnection } from './bank-connection';\nimport { Transform, Type } from 'class-transformer';\nimport { SoleBusinessAllocation } from '../sole';\nimport { Bank } from './bank';\n\nexport class BankAccount extends BankAccountBase {\n  // list of properties related with this bank account\n  @Type(() => BankAccountProperty)\n  bankAccountProperties: BankAccountProperty[];\n\n  @Type(() => SoleBusinessAllocation)\n  businessAllocations: SoleBusinessAllocation[]\n\n  @Type(() => BankAccountBalance)\n  balances: BankAccountBalance[];\n\n  // bank account's loan\n  @Type(() => Loan)\n  loan: Loan;\n\n  @Type(() => BankConnection)\n  bankConnection: BankConnection;\n\n  // @TODO Alex: temp hack: remove when TT-2107 ready. Backend shoud return Mortgage type as Loan type\n  @Transform(({ value }) => value === 4 ? BankAccountTypeEnum.LOAN : value)\n  type: BankAccountTypeEnum;\n\n  get bank(): Bank {\n    return this.bankConnection.bank;\n  }\n\n  get bsb(): string {\n    return this.accountNumber.slice(0, 6);\n  }\n\n  get number(): string {\n    return this.accountNumber.trim().slice(6);\n  }\n\n  /**\n   * Get last digits of account number\n   */\n  get partialAccountNumber(): string {\n    return `xxxx${this.accountNumber.slice(-4)}`;\n  }\n\n  /**\n   * Get current opening balance amount\n   */\n  getOpeningBalance(): number {\n    return this.getCurrentYearBalance()?.openingBalance || 0;\n  }\n\n  /**\n   * Get bank account balance for current financial year\n   */\n  getCurrentYearBalance(): BankAccountBalance {\n    const currentFinYear: number = new FinancialYear().year;\n    return this.balances.find((balance: BankAccountBalance): boolean => balance.financialYear === currentFinYear);\n  }\n\n  /**\n   * check if bank account type is one of savings accounts\n   */\n  isSavingsAccount(): boolean {\n    return this.type === BankAccountTypeEnum.INVESTMENT ||\n      this.type === BankAccountTypeEnum.TERM_DEPOSIT ||\n      this.type === BankAccountTypeEnum.SAVINGS;\n  }\n\n  /**\n   * check if bank account type is Loan\n   */\n  isLoan(): boolean {\n    return this.type === BankAccountTypeEnum.LOAN;\n  }\n\n  /**\n   * check if bank account type is Credit card\n   */\n  isCreditCard(): boolean {\n    return this.type === BankAccountTypeEnum.CREDIT_CARD;\n  }\n\n  /**\n   * check if bank account related to work tank\n   */\n  isWorkTank(): boolean {\n    return !this.isPropertyTank() && !this.isSoleTank();\n  }\n\n  /**\n   * check if bank account related to work tank\n   */\n  isPropertyTank(): boolean {\n    return !!this.bankAccountProperties.length;\n  }\n\n  /**\n   * check if bank account related to sole tank\n   */\n  isSoleTank(): boolean {\n    return !!this.businessAllocations.length;\n  }\n\n  get tankType(): TankTypeEnum {\n    switch (true) {\n      case this.isPropertyTank():\n        return TankTypeEnum.PROPERTY;\n      case this.isSoleTank():\n        return TankTypeEnum.SOLE;\n      default:\n        return TankTypeEnum.WORK;\n    }\n  }\n\n  /**\n   * Get Bank account property by id\n   * @param id Id of property\n   */\n  getPropertyById(id: number): BankAccountProperty {\n    return this.bankAccountProperties.find((bankAccountProperty: BankAccountProperty): boolean => bankAccountProperty.property.id === id);\n  }\n\n  /**\n   * check if bank account is related with passed property by id\n   * @param propertyId Property id for checking\n   */\n  hasProperty(propertyId: number): boolean {\n    return !!this.getPropertyById(propertyId);\n  }\n\n  /**\n   * Get decimal percentage for property\n   * @param propertyId Id of property\n   */\n  getPropertyPercentage(propertyId: number): number {\n    if (!this.hasProperty(propertyId)) {\n      return 0;\n    }\n\n    return this.getPropertyById(propertyId).percent / 100;\n  }\n\n  /**\n   * Get balance amount for property by id\n   * @param propertyId Id of property\n   */\n  getPropertyBalanceAmount(propertyId: number): number {\n    return this.currentBalance * this.getPropertyPercentage(propertyId);\n  }\n\n  getMonthlyRepaymentAmount(propertyId: number): number {\n    return this.loan.monthlyRepaymentAmount * this.getPropertyPercentage(propertyId);\n  }\n\n  /**\n   * Check if bank account is active (has 'active' status or hasn't been paid out/refinanced)\n   */\n  isActive(): boolean {\n    return this.status === BankAccountStatusEnum.ACTIVE;\n  }\n\n  /**\n   * first import (transactions) of basiq accounts\n   */\n  isFirstImport(): boolean {\n    return !this.lastTransactionDate && !this.isManual;\n  }\n\n  /**\n   * Check if passed user id is owner of bank account\n   */\n  isOwner(userId: number): boolean {\n    return this.bankConnection.user.id === userId;\n  }\n\n  /**\n   * Loan is paid if it has no unallocated transactions and the bank balance is $0.\n   */\n  isLoanPaid(taxTankBalance: number): boolean {\n    const shouldPayoutLoanAccount: boolean = this.isActive() && this.isLoan() && taxTankBalance === 0;\n\n    // we don't check current balance for basiq accounts, because basiq doesn't provide us last transactions (can't do for closed accounts),\n    // that's why balance won't be 0 (updated by basiq)\n    if (shouldPayoutLoanAccount && this.isManual) {\n      return this.currentBalance === 0;\n    }\n\n    return shouldPayoutLoanAccount;\n  }\n}\n"]}
@@ -2,6 +2,13 @@ import { __decorate } from "tslib";
2
2
  import { BorrowingReport as BorrowingReportBase } from '../../db/Models/property/borrowing-report';
3
3
  import { Type } from 'class-transformer';
4
4
  import { User } from '../user';
5
+ import { plainToClass } from 'class-transformer';
6
+ import { Collection, IncomeSourceForecastCollection } from '../../collections';
7
+ import { AnnualClientDetails } from '../client';
8
+ /**
9
+ * Borrowing power report, shows if u can afford new property loan
10
+ * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/4717608961/Borrowing+Power+Report+-+Property+Tank
11
+ */
5
12
  export class BorrowingReport extends BorrowingReportBase {
6
13
  constructor() {
7
14
  super(...arguments);
@@ -10,8 +17,35 @@ export class BorrowingReport extends BorrowingReportBase {
10
17
  this.otherLiabilities = 0;
11
18
  this.livingExpenses = 0;
12
19
  }
20
+ /**
21
+ * some fields calculated automatically until user filled them manually
22
+ */
23
+ static calculate(salaryForecasts, incomeSourceForecasts, clientDetails = plainToClass(AnnualClientDetails, {}), report = plainToClass(BorrowingReport, {})) {
24
+ if (!report.netSalary) {
25
+ report.netSalary = new Collection(salaryForecasts).sumBy('monthlyAmount')
26
+ + new IncomeSourceForecastCollection(incomeSourceForecasts).bonuses.sumBy('monthlyAmount');
27
+ }
28
+ if (!report.spouseNetSalary) {
29
+ report.spouseNetSalary = clientDetails.spouseMonthlyIncome;
30
+ }
31
+ return report;
32
+ }
33
+ /**
34
+ * free money after expenses
35
+ */
36
+ getNetIncomeSurplus(profit, repaymentAmount) {
37
+ return profit + this.netRentIncome + this.netSalary + this.spouseNetSalary - repaymentAmount
38
+ - this.loanRepaymentAmount - this.otherLiabilities - this.livingExpenses;
39
+ }
40
+ /**
41
+ * surplus income to cover your debt obligations
42
+ */
43
+ getNetSurplusRatio(profit, repaymentAmount) {
44
+ return (profit + this.netRentIncome + this.netSalary + this.spouseNetSalary - this.livingExpenses)
45
+ / (repaymentAmount + this.loanRepaymentAmount + this.otherLiabilities);
46
+ }
13
47
  }
14
48
  __decorate([
15
49
  Type(() => User)
16
50
  ], BorrowingReport.prototype, "user", void 0);
17
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYm9ycm93aW5nLXJlcG9ydC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R0LWNvcmUvc3JjL2xpYi9tb2RlbHMvcHJvcGVydHkvYm9ycm93aW5nLXJlcG9ydC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxFQUFFLGVBQWUsSUFBSSxtQkFBbUIsRUFBRSxNQUFNLDJDQUEyQyxDQUFDO0FBQ25HLE9BQU8sRUFBRSxJQUFJLEVBQWEsTUFBTSxtQkFBbUIsQ0FBQztBQUNwRCxPQUFPLEVBQUUsSUFBSSxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBRS9CLE1BQU0sT0FBTyxlQUFnQixTQUFRLG1CQUFtQjtJQUF4RDs7UUFJRSxrQkFBYSxHQUFXLENBQUMsQ0FBQztRQUMxQix3QkFBbUIsR0FBVyxDQUFDLENBQUM7UUFDaEMscUJBQWdCLEdBQVcsQ0FBQyxDQUFDO1FBQzdCLG1CQUFjLEdBQVcsQ0FBQyxDQUFDO0lBQzdCLENBQUM7Q0FBQTtBQU5DO0lBREMsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQzs2Q0FDTiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJvcnJvd2luZ1JlcG9ydCBhcyBCb3Jyb3dpbmdSZXBvcnRCYXNlIH0gZnJvbSAnLi4vLi4vZGIvTW9kZWxzL3Byb3BlcnR5L2JvcnJvd2luZy1yZXBvcnQnO1xuaW1wb3J0IHsgVHlwZSwgVHJhbnNmb3JtIH0gZnJvbSAnY2xhc3MtdHJhbnNmb3JtZXInO1xuaW1wb3J0IHsgVXNlciB9IGZyb20gJy4uL3VzZXInO1xuXG5leHBvcnQgY2xhc3MgQm9ycm93aW5nUmVwb3J0IGV4dGVuZHMgQm9ycm93aW5nUmVwb3J0QmFzZSB7XG4gIEBUeXBlKCgpID0+IFVzZXIpXG4gIHVzZXI6IFVzZXI7XG5cbiAgbmV0UmVudEluY29tZTogbnVtYmVyID0gMDtcbiAgbG9hblJlcGF5bWVudEFtb3VudDogbnVtYmVyID0gMDtcbiAgb3RoZXJMaWFiaWxpdGllczogbnVtYmVyID0gMDtcbiAgbGl2aW5nRXhwZW5zZXM6IG51bWJlciA9IDA7XG59XG4iXX0=
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYm9ycm93aW5nLXJlcG9ydC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R0LWNvcmUvc3JjL2xpYi9tb2RlbHMvcHJvcGVydHkvYm9ycm93aW5nLXJlcG9ydC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxFQUFFLGVBQWUsSUFBSSxtQkFBbUIsRUFBRSxNQUFNLDJDQUEyQyxDQUFDO0FBQ25HLE9BQU8sRUFBRSxJQUFJLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUN6QyxPQUFPLEVBQUUsSUFBSSxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBQy9CLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUNqRCxPQUFPLEVBQUUsVUFBVSxFQUFFLDhCQUE4QixFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFDL0UsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sV0FBVyxDQUFDO0FBR2hEOzs7R0FHRztBQUNILE1BQU0sT0FBTyxlQUFnQixTQUFRLG1CQUFtQjtJQUF4RDs7UUFJRSxrQkFBYSxHQUFXLENBQUMsQ0FBQztRQUMxQix3QkFBbUIsR0FBVyxDQUFDLENBQUM7UUFDaEMscUJBQWdCLEdBQVcsQ0FBQyxDQUFDO1FBQzdCLG1CQUFjLEdBQVcsQ0FBQyxDQUFDO0lBcUM3QixDQUFDO0lBbkNDOztPQUVHO0lBQ0gsTUFBTSxDQUFDLFNBQVMsQ0FDZCxlQUFpQyxFQUNqQyxxQkFBNkMsRUFDN0MsZ0JBQXFDLFlBQVksQ0FBQyxtQkFBbUIsRUFBRSxFQUFFLENBQUMsRUFDMUUsU0FBMEIsWUFBWSxDQUFDLGVBQWUsRUFBRSxFQUFFLENBQUM7UUFFM0QsSUFBSSxDQUFDLE1BQU0sQ0FBQyxTQUFTLEVBQUU7WUFDckIsTUFBTSxDQUFDLFNBQVMsR0FBRyxJQUFJLFVBQVUsQ0FBQyxlQUFlLENBQUMsQ0FBQyxLQUFLLENBQUMsZUFBZSxDQUFDO2tCQUNyRSxJQUFJLDhCQUE4QixDQUFDLHFCQUFxQixDQUFDLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxlQUFlLENBQUMsQ0FBQztTQUM5RjtRQUNELElBQUksQ0FBQyxNQUFNLENBQUMsZUFBZSxFQUFFO1lBQzNCLE1BQU0sQ0FBQyxlQUFlLEdBQUcsYUFBYSxDQUFDLG1CQUFtQixDQUFDO1NBQzVEO1FBRUQsT0FBTyxNQUFNLENBQUM7SUFDaEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsbUJBQW1CLENBQUMsTUFBYyxFQUFFLGVBQXVCO1FBQ3pELE9BQU8sTUFBTSxHQUFHLElBQUksQ0FBQyxhQUFhLEdBQUcsSUFBSSxDQUFDLFNBQVMsR0FBRyxJQUFJLENBQUMsZUFBZSxHQUFHLGVBQWU7Y0FDeEYsSUFBSSxDQUFDLG1CQUFtQixHQUFHLElBQUksQ0FBQyxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFBO0lBQzVFLENBQUM7SUFFRDs7T0FFRztJQUNILGtCQUFrQixDQUFDLE1BQWMsRUFBRSxlQUF1QjtRQUN4RCxPQUFPLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxhQUFhLEdBQUcsSUFBSSxDQUFDLFNBQVMsR0FBRyxJQUFJLENBQUMsZUFBZSxHQUFHLElBQUksQ0FBQyxjQUFjLENBQUM7Y0FDOUYsQ0FBQyxlQUFlLEdBQUcsSUFBSSxDQUFDLG1CQUFtQixHQUFHLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFBO0lBQzFFLENBQUM7Q0FDRjtBQTFDQztJQURDLElBQUksQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUM7NkNBQ04iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBCb3Jyb3dpbmdSZXBvcnQgYXMgQm9ycm93aW5nUmVwb3J0QmFzZSB9IGZyb20gJy4uLy4uL2RiL01vZGVscy9wcm9wZXJ0eS9ib3Jyb3dpbmctcmVwb3J0JztcbmltcG9ydCB7IFR5cGUgfSBmcm9tICdjbGFzcy10cmFuc2Zvcm1lcic7XG5pbXBvcnQgeyBVc2VyIH0gZnJvbSAnLi4vdXNlcic7XG5pbXBvcnQgeyBwbGFpblRvQ2xhc3MgfSBmcm9tICdjbGFzcy10cmFuc2Zvcm1lcic7XG5pbXBvcnQgeyBDb2xsZWN0aW9uLCBJbmNvbWVTb3VyY2VGb3JlY2FzdENvbGxlY3Rpb24gfSBmcm9tICcuLi8uLi9jb2xsZWN0aW9ucyc7XG5pbXBvcnQgeyBBbm51YWxDbGllbnREZXRhaWxzIH0gZnJvbSAnLi4vY2xpZW50JztcbmltcG9ydCB7IEluY29tZVNvdXJjZUZvcmVjYXN0LCBTYWxhcnlGb3JlY2FzdCB9IGZyb20gJy4uL2luY29tZS1zb3VyY2UnO1xuXG4vKipcbiAqIEJvcnJvd2luZyBwb3dlciByZXBvcnQsIHNob3dzIGlmIHUgY2FuIGFmZm9yZCBuZXcgcHJvcGVydHkgbG9hblxuICogaHR0cHM6Ly90YXh0YW5rLmF0bGFzc2lhbi5uZXQvd2lraS9zcGFjZXMvVEFYVEFOSy9wYWdlcy80NzE3NjA4OTYxL0JvcnJvd2luZytQb3dlcitSZXBvcnQrLStQcm9wZXJ0eStUYW5rXG4gKi9cbmV4cG9ydCBjbGFzcyBCb3Jyb3dpbmdSZXBvcnQgZXh0ZW5kcyBCb3Jyb3dpbmdSZXBvcnRCYXNlIHtcbiAgQFR5cGUoKCkgPT4gVXNlcilcbiAgdXNlcjogVXNlcjtcblxuICBuZXRSZW50SW5jb21lOiBudW1iZXIgPSAwO1xuICBsb2FuUmVwYXltZW50QW1vdW50OiBudW1iZXIgPSAwO1xuICBvdGhlckxpYWJpbGl0aWVzOiBudW1iZXIgPSAwO1xuICBsaXZpbmdFeHBlbnNlczogbnVtYmVyID0gMDtcblxuICAvKipcbiAgICogc29tZSBmaWVsZHMgY2FsY3VsYXRlZCBhdXRvbWF0aWNhbGx5IHVudGlsIHVzZXIgZmlsbGVkIHRoZW0gbWFudWFsbHlcbiAgICovXG4gIHN0YXRpYyBjYWxjdWxhdGUoXG4gICAgc2FsYXJ5Rm9yZWNhc3RzOiBTYWxhcnlGb3JlY2FzdFtdLFxuICAgIGluY29tZVNvdXJjZUZvcmVjYXN0czogSW5jb21lU291cmNlRm9yZWNhc3RbXSxcbiAgICBjbGllbnREZXRhaWxzOiBBbm51YWxDbGllbnREZXRhaWxzID0gcGxhaW5Ub0NsYXNzKEFubnVhbENsaWVudERldGFpbHMsIHt9KSxcbiAgICByZXBvcnQ6IEJvcnJvd2luZ1JlcG9ydCA9IHBsYWluVG9DbGFzcyhCb3Jyb3dpbmdSZXBvcnQsIHt9KVxuICApOiBCb3Jyb3dpbmdSZXBvcnQge1xuICAgIGlmICghcmVwb3J0Lm5ldFNhbGFyeSkge1xuICAgICAgcmVwb3J0Lm5ldFNhbGFyeSA9IG5ldyBDb2xsZWN0aW9uKHNhbGFyeUZvcmVjYXN0cykuc3VtQnkoJ21vbnRobHlBbW91bnQnKVxuICAgICAgICArIG5ldyBJbmNvbWVTb3VyY2VGb3JlY2FzdENvbGxlY3Rpb24oaW5jb21lU291cmNlRm9yZWNhc3RzKS5ib251c2VzLnN1bUJ5KCdtb250aGx5QW1vdW50Jyk7XG4gICAgfVxuICAgIGlmICghcmVwb3J0LnNwb3VzZU5ldFNhbGFyeSkge1xuICAgICAgcmVwb3J0LnNwb3VzZU5ldFNhbGFyeSA9IGNsaWVudERldGFpbHMuc3BvdXNlTW9udGhseUluY29tZTtcbiAgICB9XG5cbiAgICByZXR1cm4gcmVwb3J0O1xuICB9XG5cbiAgLyoqXG4gICAqIGZyZWUgbW9uZXkgYWZ0ZXIgZXhwZW5zZXNcbiAgICovXG4gIGdldE5ldEluY29tZVN1cnBsdXMocHJvZml0OiBudW1iZXIsIHJlcGF5bWVudEFtb3VudDogbnVtYmVyKTogbnVtYmVyIHtcbiAgICByZXR1cm4gcHJvZml0ICsgdGhpcy5uZXRSZW50SW5jb21lICsgdGhpcy5uZXRTYWxhcnkgKyB0aGlzLnNwb3VzZU5ldFNhbGFyeSAtIHJlcGF5bWVudEFtb3VudFxuICAgICAgLSB0aGlzLmxvYW5SZXBheW1lbnRBbW91bnQgLSB0aGlzLm90aGVyTGlhYmlsaXRpZXMgLSB0aGlzLmxpdmluZ0V4cGVuc2VzXG4gIH1cblxuICAvKipcbiAgICogc3VycGx1cyBpbmNvbWUgdG8gY292ZXIgeW91ciBkZWJ0IG9ibGlnYXRpb25zXG4gICAqL1xuICBnZXROZXRTdXJwbHVzUmF0aW8ocHJvZml0OiBudW1iZXIsIHJlcGF5bWVudEFtb3VudDogbnVtYmVyKTogbnVtYmVyIHtcbiAgICByZXR1cm4gKHByb2ZpdCArIHRoaXMubmV0UmVudEluY29tZSArIHRoaXMubmV0U2FsYXJ5ICsgdGhpcy5zcG91c2VOZXRTYWxhcnkgLSB0aGlzLmxpdmluZ0V4cGVuc2VzKVxuICAgICAgLyAocmVwYXltZW50QW1vdW50ICsgdGhpcy5sb2FuUmVwYXltZW50QW1vdW50ICsgdGhpcy5vdGhlckxpYWJpbGl0aWVzKVxuICB9XG59XG4iXX0=
@@ -14,10 +14,10 @@ import round from 'lodash/round';
14
14
  import flatten from 'lodash/flatten';
15
15
  import hasIn from 'lodash/hasIn';
16
16
  import intersection from 'lodash/intersection';
17
+ import uniqBy from 'lodash/uniqBy';
17
18
  import first from 'lodash/first';
18
19
  import last from 'lodash/last';
19
20
  import orderBy from 'lodash/orderBy';
20
- import uniqBy from 'lodash/uniqBy';
21
21
  import differenceBy from 'lodash/differenceBy';
22
22
  import uniq from 'lodash/uniq';
23
23
  import moment from 'moment';
@@ -1649,13 +1649,13 @@ class CollectionDictionary {
1649
1649
  return this.items[key] ? this.items[key] : this.createCollection([]);
1650
1650
  }
1651
1651
  /**
1652
- * Join several collections by ids
1652
+ * Join several collections by ids, return collection of uniq models (skip duplicates)
1653
1653
  */
1654
1654
  merge(keys) {
1655
1655
  if (!this.length) {
1656
1656
  return this.createCollection([]);
1657
1657
  }
1658
- return this.createCollection(flatten(keys.map((key) => this.get(key.toString()).items)));
1658
+ return this.createCollection(uniqBy(flatten(keys.map((key) => this.get(key.toString()).items)), 'id'));
1659
1659
  }
1660
1660
  /**
1661
1661
  * Create instance of collection
@@ -2187,6 +2187,9 @@ class PropertyCollection extends Collection {
2187
2187
  newItems.unshift(...bestProperties);
2188
2188
  return this.create(newItems);
2189
2189
  }
2190
+ get netIncome() {
2191
+ return this.sumBy('currentYearForecast.income') + this.sumBy('currentYearForecast.expense');
2192
+ }
2190
2193
  }
2191
2194
 
2192
2195
  class PropertyCategoryMovementCollection extends Collection {
@@ -5977,6 +5980,10 @@ __decorate([
5977
5980
  let BorrowingReport$1 = class BorrowingReport extends AbstractModel {
5978
5981
  };
5979
5982
 
5983
+ /**
5984
+ * Borrowing power report, shows if u can afford new property loan
5985
+ * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/4717608961/Borrowing+Power+Report+-+Property+Tank
5986
+ */
5980
5987
  class BorrowingReport extends BorrowingReport$1 {
5981
5988
  constructor() {
5982
5989
  super(...arguments);
@@ -5985,6 +5992,33 @@ class BorrowingReport extends BorrowingReport$1 {
5985
5992
  this.otherLiabilities = 0;
5986
5993
  this.livingExpenses = 0;
5987
5994
  }
5995
+ /**
5996
+ * some fields calculated automatically until user filled them manually
5997
+ */
5998
+ static calculate(salaryForecasts, incomeSourceForecasts, clientDetails = plainToClass(AnnualClientDetails, {}), report = plainToClass(BorrowingReport, {})) {
5999
+ if (!report.netSalary) {
6000
+ report.netSalary = new Collection(salaryForecasts).sumBy('monthlyAmount')
6001
+ + new IncomeSourceForecastCollection(incomeSourceForecasts).bonuses.sumBy('monthlyAmount');
6002
+ }
6003
+ if (!report.spouseNetSalary) {
6004
+ report.spouseNetSalary = clientDetails.spouseMonthlyIncome;
6005
+ }
6006
+ return report;
6007
+ }
6008
+ /**
6009
+ * free money after expenses
6010
+ */
6011
+ getNetIncomeSurplus(profit, repaymentAmount) {
6012
+ return profit + this.netRentIncome + this.netSalary + this.spouseNetSalary - repaymentAmount
6013
+ - this.loanRepaymentAmount - this.otherLiabilities - this.livingExpenses;
6014
+ }
6015
+ /**
6016
+ * surplus income to cover your debt obligations
6017
+ */
6018
+ getNetSurplusRatio(profit, repaymentAmount) {
6019
+ return (profit + this.netRentIncome + this.netSalary + this.spouseNetSalary - this.livingExpenses)
6020
+ / (repaymentAmount + this.loanRepaymentAmount + this.otherLiabilities);
6021
+ }
5988
6022
  }
5989
6023
  __decorate([
5990
6024
  Type(() => User)
@@ -7797,8 +7831,26 @@ class BankAccountCollection extends Collection {
7797
7831
  getPropertyBalanceAmount(propertyId) {
7798
7832
  return this.items.reduce((sum, bankAccount) => sum + bankAccount.getPropertyBalanceAmount(propertyId), 0);
7799
7833
  }
7800
- getMonthlyRepaymentAmount(propertyId) {
7801
- return this.getLoanAccounts().reduce((sum, bankAccount) => sum + bankAccount.loan.monthlyRepaymentAmount * bankAccount.getPropertyPercentage(propertyId), 0);
7834
+ get propertiesBalanceAmount() {
7835
+ let balanceAmount = 0;
7836
+ this.items.forEach((bankAccount) => {
7837
+ bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {
7838
+ balanceAmount += Math.abs(bankAccount.getPropertyBalanceAmount(bankAccountProperty.property.id));
7839
+ });
7840
+ });
7841
+ return balanceAmount;
7842
+ }
7843
+ get propertiesMonthlyRepaymentAmount() {
7844
+ let repaymentAmount = 0;
7845
+ this.getLoanAccounts().items.forEach((bankAccount) => {
7846
+ bankAccount.bankAccountProperties.forEach((bankAccountProperty) => {
7847
+ repaymentAmount += bankAccount.getMonthlyRepaymentAmount(bankAccountProperty.property.id);
7848
+ });
7849
+ });
7850
+ return repaymentAmount;
7851
+ }
7852
+ getPropertyMonthlyRepaymentAmount(propertyId) {
7853
+ return this.getLoanAccounts().reduce((sum, bankAccount) => sum + bankAccount.getMonthlyRepaymentAmount(propertyId), 0);
7802
7854
  }
7803
7855
  getLVR(property) {
7804
7856
  return this.getPropertyBalanceAmount(property.id) / property.currentYearForecast.marketValue;
@@ -7807,6 +7859,9 @@ class BankAccountCollection extends Collection {
7807
7859
  // TODO Alex/Vik: maybe we should get it from jwtToken or think about some localStorageService?
7808
7860
  return this.filter((bankAccount) => bankAccount.isOwner(+localStorage.getItem('userId')));
7809
7861
  }
7862
+ getActiveLoanAccountsByProperties(ids) {
7863
+ return this.getActiveBankAccounts().getLoanAccounts().getByPropertiesIds(ids);
7864
+ }
7810
7865
  }
7811
7866
 
7812
7867
  /**
@@ -8593,6 +8648,9 @@ class BankAccount extends BankAccount$1 {
8593
8648
  getPropertyBalanceAmount(propertyId) {
8594
8649
  return this.currentBalance * this.getPropertyPercentage(propertyId);
8595
8650
  }
8651
+ getMonthlyRepaymentAmount(propertyId) {
8652
+ return this.loan.monthlyRepaymentAmount * this.getPropertyPercentage(propertyId);
8653
+ }
8596
8654
  /**
8597
8655
  * Check if bank account is active (has 'active' status or hasn't been paid out/refinanced)
8598
8656
  */