taxtank-core 0.33.12 → 0.33.15

Sign up to get free protection for your applications and to get access to all the features.
@@ -71,5 +71,8 @@ export class HoldingCollection extends ExportableCollection {
71
71
  getUnsold() {
72
72
  return this.removeBy('currentQuantity', 0);
73
73
  }
74
+ getUnrealisedProfit() {
75
+ return this.marketValue - this.getPurchaseValue();
76
+ }
74
77
  }
75
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaG9sZGluZy5jb2xsZWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHQtY29yZS9zcmMvbGliL2NvbGxlY3Rpb25zL2hvbGRpbmcvaG9sZGluZy5jb2xsZWN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUNqRCxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUVoRSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDN0QsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sMkNBQTJDLENBQUM7QUFDL0UsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFekQsTUFBTSxPQUFPLGlCQUFrQixTQUFRLG9CQUE2QjtJQUNsRSxlQUFlO1FBQ2IsT0FBTyxDQUFDLE1BQU0sRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLFlBQVksRUFBRSxZQUFZLEVBQUUsYUFBYSxFQUFFLGdCQUFnQixFQUFFLG9CQUFvQixFQUFFLFVBQVUsRUFBRSxhQUFhLENBQUMsQ0FBQztJQUNsSixDQUFDO0lBRUQsZUFBZTtRQUNiLE9BQU87WUFDTCxZQUFZLENBQUMsVUFBVSxFQUFFLEVBQUUsS0FBSyxFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUUsa0JBQWtCLENBQUMsTUFBTSxFQUFFLENBQUM7U0FDekUsQ0FBQztJQUNKLENBQUM7SUFFRCxhQUFhO1FBQ1gsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLE9BQWdCLEVBQUUsRUFBRSxDQUFDO1lBQzFDLFlBQVksQ0FBQyxVQUFVLEVBQUUsRUFBRSxLQUFLLEVBQUUsT0FBTyxDQUFDLElBQUksRUFBRSxJQUFJLEVBQUUsa0JBQWtCLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDaEYsWUFBWSxD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RUFBRSxDQUFDLENBQUMsQ0FBQztJQUM3QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBwbGFpblRvQ2xhc3MgfSBmcm9tICdjbGFzcy10cmFuc2Zvcm1lcic7XHJcbmltcG9ydCB7IEV4cG9ydGFibGVDb2xsZWN0aW9uIH0gZnJvbSAnLi4vZXhwb3J0YWJsZS5jb2xsZWN0aW9uJztcclxuaW1wb3J0IHsgSG9sZGluZyB9IGZyb20gJy4uLy4uL21vZGVscyc7XHJcbmltcG9ydCB7IEV4cG9ydENlbGwgfSBmcm9tICcuLi8uLi9tb2RlbHMvZXhwb3J0L2V4cG9ydC1jZWxsJztcclxuaW1wb3J0IHsgRXhwb3J0Q2VsbFR5cGVFbnVtIH0gZnJvbSAnLi4vLi4vbW9kZWxzL2V4cG9ydC9leHBvcnQtY2VsbC10eXBlLmVudW0nO1xyXG5pbXBvcnQgeyBIb2xkaW5nVHlwZUNhdGVnb3J5RW51bSB9IGZyb20gJy4uLy4uL2RiL0VudW1zJztcclxuXHJcbmV4cG9ydCBjbGFzcyBIb2xkaW5nQ29sbGVjdGlvbiBleHRlbmRzIEV4cG9ydGFibGVDb2xsZWN0aW9uPEhvbGRpbmc+IHtcclxuICBnZXRFeHBvcnRIZWFkZXIoKTogc3RyaW5nW10ge1xyXG4gICAgcmV0dXJuIFsnRGF0ZScsICdUaWNrZXInLCAnTmFtZScsICdBc3NldCB0eXBlJywgJ1F0eSBib3VnaHQnLCAnUXR5IGN1cnJlbnQnLCAnQnV5IHByaWNlL1VuaXQnLCAnQ3VycmVudCBwcmljZS9Vbml0JywgJ1RvdGFsIE1WJywgJ1Byb2ZpdC9Mb3NzJ107XHJcbiAgfVxyXG5cclxuICBnZXRFeHBvcnRGb290ZXIoKTogRXhwb3J0Q2VsbFtdIHtcclxuICAgIHJldHVybiBbXHJcbiAgICAgIHBsYWluVG9DbGFzcyhFeHBvcnRDZWxsLCB7IHZhbHVlOiAnJywgdHlwZTogRXhwb3J0Q2VsbFR5cGVFbnVtLlNUUklORyB9KSxcclxuICAgIF07XHJcbiAgfVxyXG5cclxuICBnZXRFeHBvcnRCb2R5KCk6IEV4cG9ydENlbGxbXVtdIHtcclxuICAgIHJldHVybiB0aGlzLml0ZW1zLm1hcCgoaG9sZGluZzogSG9sZGluZykgPT4gW1xyXG4gICAgICBwbGFpblRvQ2xhc3MoRXhwb3J0Q2VsbCwgeyB2YWx1ZTogaG9sZGluZy5kYXRlLCB0eXBlOiBFeHBvcnRDZWxsVHlwZUVudW0uREFURSB9KSxcclxuICAgICAgcGxhaW5Ub0NsYXNzKEV4cG9ydENlbGwsIHsgdmFsdWU6IGhvbGRpbmcudHlwZS50aWNrZXIsIHR5cGU6IEV4cG9ydENlbGxUeXBlRW51bS5TVFJJTkcgfSksXHJcbiAgICAgIHBsYWluVG9DbGFzcyhFeHBvcnRDZWxsLCB7XHJcbiAgICAgICAgdmFsdWU6IGhvbGRpbmcudHlwZS5uYW1lLFxyXG4gICAgICAgIHR5cGU6IEV4cG9ydENlbGxUeXBlRW51bS5TVFJJTkdcclxuICAgICAgfSksXHJcbiAgICAgIHBsYWluVG9DbGFzcyhFeHBvcnRDZWxsLCB7IHZhbHVlOiBob2xkaW5nLnR5cGUuY2F0ZWdvcnlMYWJlbCwgdHlwZTogRXhwb3J0Q2VsbFR5cGVFbnVtLlNUUklORyB9KSxcclxuICAgICAgcGxhaW5Ub0NsYXNzKEV4cG9ydENlbGwsIHsgdmFsdWU6IGhvbGRpbmcucXVhbnRpdHksIHR5cGU6IEV4cG9ydENlbGxUeXBlRW51bS5TVFJJTkcgfSksXHJcbiAgICAgIHBsYWluVG9DbGFzcyhFeHBvcnRDZWxsLCB7IHZhbHVlOiBob2xkaW5nLmN1cnJlbnRRdWFudGl0eSwgdHlwZTogRXhwb3J0Q2VsbFR5cGVFbnVtLlNUUklORyB9KSxcclxuICAgICAgcGxhaW5Ub0NsYXNzKEV4cG9ydENlbGwsIHsgdmFsdWU6IGhvbGRpbmcucHJpY2UsIHR5cGU6IEV4cG9ydENlbGxUeXBlRW51bS5DVVJSRU5DWSB9KSxcclxuICAgICAgcGxhaW5Ub0NsYXNzKEV4cG9ydENlbGwsIHtcclxuICAgICAgICB2YWx1ZTogaG9sZGluZy50eXBlLnByaWNlLFxyXG4gICAgICAgIHR5cGU6IEV4cG9ydENlbGxUeXBlRW51bS5DVVJSRU5DWVxyXG4gICAgICB9KSxcclxuICAgICAgcGxhaW5Ub0NsYXNzKEV4cG9ydENlbGwsIHtcclxuICAgICAgICB2YWx1ZTogaG9sZGluZy5tYXJrZXRWYWx1ZSxcclxuICAgICAgICB0eXBlOiBFeHBvcnRDZWxsVHlwZUVudW0uQ1VSUkVOQ1lcclxuICAgICAgfSksXHJcbiAgICAgIHBsYWluVG9DbGFzcyhFeHBvcnRDZWxsLCB7XHJcbiAgICAgICAgdmFsdWU6IGhvbGRpbmcuZ3Jvd3RoQ29lZmZpY2llbnQsXHJcbiAgICAgICAgdHlwZTogRXhwb3J0Q2VsbFR5cGVFbnVtLlBFUkNFTlRcclxuICAgICAgfSksXHJcbiAgICBdKTtcclxuICB9XHJcblxyXG4gIGdldFNoYXJlcygpOiB0aGlzIHtcclxuICAgIHJldHVybiB0aGlzLmZpbHRlckJ5KCd0eXBlLmNhdGVnb3J5JywgSG9sZGluZ1R5cGVDYXRlZ29yeUVudW0uU1RPQ0spO1xyXG4gIH1cclxuXHJcbiAgZ2V0Q3J5cHRvcygpOiB0aGlzIHtcclxuICAgIHJldHVybiB0aGlzLmZpbHRlckJ5KCd0eXBlLmNhdGVnb3J5JywgSG9sZGluZ1R5cGVDYXRlZ29yeUVudW0uQ1JZUFRPKTtcclxuICB9XHJcblxyXG4gIGdldE90aGVycygpOiB0aGlzIHtcclxuICAgIHJldHVybiB0aGlzLmZpbHRlckJ5KCd0eXBlLmNhdGVnb3J5JywgW1xyXG4gICAgICBIb2xkaW5nVHlwZUNhdGVnb3J5RW51bS5DT0xMRUNUSUJMRSxcclxuICAgICAgSG9sZGluZ1R5cGVDYXRlZ29yeUVudW0uVU5MSVNURURfU1RPQ0ssXHJcbiAgICAgIEhvbGRpbmdUeXBlQ2F0ZWdvcnlFbnVtLlVOTElTVEVEX1VOSVQsXHJcbiAgICAgIEhvbGRpbmdUeXBlQ2F0ZWdvcnlFbnVtLk9USEVSXHJcbiAgICBdKTtcclxuICB9XHJcblxyXG4gIGdldFB1cmNoYXNlVmFsdWUoKTogbnVtYmVyIHtcclxuICAgIHJldHVybiB0aGlzLnN1bUJ5KCdwdXJjaGFzZVZhbHVlJyk7XHJcbiAgfVxyXG5cclxuICBnZXQgbWFya2V0VmFsdWUoKTogbnVtYmVyIHtcclxuICAgIHJldHVybiB0aGlzLnN1bUJ5KCdtYXJrZXRWYWx1ZScpO1xyXG4gIH1cclxuXHJcbiAgZ2V0IHNoYXJlZE1hcmtldFZhbHVlKCk6IG51bWJlciB7XHJcbiAgICBsZXQgc2hhcmVkTWFya2V0VmFsdWUgPSAwO1xyXG4gICAgdGhpcy5pdGVtcy5mb3JFYWNoKGhvbGRpbmcgPT4ge1xyXG4gICAgICBzaGFyZWRNYXJrZXRWYWx1ZSArPSBob2xkaW5nLm93bmVyc2hpcFBlcmNlbnQgLyAxMDAgKiBob2xkaW5nLm1hcmtldFZhbHVlXHJcbiAgICB9KTtcclxuXHJcbiAgICByZXR1cm4gc2hhcmVkTWFya2V0VmFsdWU7XHJcbiAgfVxyXG5cclxuICBnZXRHcm93dGhDb2VmZmljaWVudCgpOiBudW1iZXIge1xyXG4gICAgcmV0dXJuIHRoaXMubWFya2V0VmFsdWUgLyB0aGlzLmdldFB1cmNoYXNlVmFsdWUoKSAtIDE7XHJcbiAgfVxyXG5cclxuICBnZXRVbnNvbGQoKTogdGhpcyB7XHJcbiAgICByZXR1cm4gdGhpcy5yZW1vdmVCeSgnY3VycmVudFF1YW50aXR5JywgMCk7XHJcbiAgfVxyXG59XHJcbiJdfQ==
78
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"holding.collection.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/collections/holding/holding.collection.ts"],"names":[],"mappings":"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","sourcesContent":["import { plainToClass } from 'class-transformer';\r\nimport { ExportableCollection } from '../exportable.collection';\r\nimport { Holding } from '../../models';\r\nimport { ExportCell } from '../../models/export/export-cell';\r\nimport { ExportCellTypeEnum } from '../../models/export/export-cell-type.enum';\r\nimport { HoldingTypeCategoryEnum } from '../../db/Enums';\r\n\r\nexport class HoldingCollection extends ExportableCollection<Holding> {\r\n  getExportHeader(): string[] {\r\n    return ['Date', 'Ticker', 'Name', 'Asset type', 'Qty bought', 'Qty current', 'Buy price/Unit', 'Current price/Unit', 'Total MV', 'Profit/Loss'];\r\n  }\r\n\r\n  getExportFooter(): ExportCell[] {\r\n    return [\r\n      plainToClass(ExportCell, { value: '', type: ExportCellTypeEnum.STRING }),\r\n    ];\r\n  }\r\n\r\n  getExportBody(): ExportCell[][] {\r\n    return this.items.map((holding: Holding) => [\r\n      plainToClass(ExportCell, { value: holding.date, type: ExportCellTypeEnum.DATE }),\r\n      plainToClass(ExportCell, { value: holding.type.ticker, type: ExportCellTypeEnum.STRING }),\r\n      plainToClass(ExportCell, {\r\n        value: holding.type.name,\r\n        type: ExportCellTypeEnum.STRING\r\n      }),\r\n      plainToClass(ExportCell, { value: holding.type.categoryLabel, type: ExportCellTypeEnum.STRING }),\r\n      plainToClass(ExportCell, { value: holding.quantity, type: ExportCellTypeEnum.STRING }),\r\n      plainToClass(ExportCell, { value: holding.currentQuantity, type: ExportCellTypeEnum.STRING }),\r\n      plainToClass(ExportCell, { value: holding.price, type: ExportCellTypeEnum.CURRENCY }),\r\n      plainToClass(ExportCell, {\r\n        value: holding.type.price,\r\n        type: ExportCellTypeEnum.CURRENCY\r\n      }),\r\n      plainToClass(ExportCell, {\r\n        value: holding.marketValue,\r\n        type: ExportCellTypeEnum.CURRENCY\r\n      }),\r\n      plainToClass(ExportCell, {\r\n        value: holding.growthCoefficient,\r\n        type: ExportCellTypeEnum.PERCENT\r\n      }),\r\n    ]);\r\n  }\r\n\r\n  getShares(): this {\r\n    return this.filterBy('type.category', HoldingTypeCategoryEnum.STOCK);\r\n  }\r\n\r\n  getCryptos(): this {\r\n    return this.filterBy('type.category', HoldingTypeCategoryEnum.CRYPTO);\r\n  }\r\n\r\n  getOthers(): this {\r\n    return this.filterBy('type.category', [\r\n      HoldingTypeCategoryEnum.COLLECTIBLE,\r\n      HoldingTypeCategoryEnum.UNLISTED_STOCK,\r\n      HoldingTypeCategoryEnum.UNLISTED_UNIT,\r\n      HoldingTypeCategoryEnum.OTHER\r\n    ]);\r\n  }\r\n\r\n  getPurchaseValue(): number {\r\n    return this.sumBy('purchaseValue');\r\n  }\r\n\r\n  get marketValue(): number {\r\n    return this.sumBy('marketValue');\r\n  }\r\n\r\n  get sharedMarketValue(): number {\r\n    let sharedMarketValue = 0;\r\n    this.items.forEach(holding => {\r\n      sharedMarketValue += holding.ownershipPercent / 100 * holding.marketValue\r\n    });\r\n\r\n    return sharedMarketValue;\r\n  }\r\n\r\n  getGrowthCoefficient(): number {\r\n    return this.marketValue / this.getPurchaseValue() - 1;\r\n  }\r\n\r\n  getUnsold(): this {\r\n    return this.removeBy('currentQuantity', 0);\r\n  }\r\n\r\n  getUnrealisedProfit() {\r\n    return this.marketValue - this.getPurchaseValue();\r\n  }\r\n}\r\n"]}
@@ -29,14 +29,10 @@ export class PropertyCollection extends Collection {
29
29
  return this.sumBy('growthPercent');
30
30
  }
31
31
  get marketValue() {
32
- return this.sumBy('valuation.marketValue');
32
+ return this.sumBy('marketValue');
33
33
  }
34
34
  get sharedMarketValue() {
35
- let sharedMarketValue = 0;
36
- this.items.forEach(property => {
37
- sharedMarketValue += property.shareRatio * property.marketValue;
38
- });
39
- return sharedMarketValue;
35
+ return this.sumBy('sharedMarketValue');
40
36
  }
41
37
  /**
42
38
  * Get list of unique property categories from collection
@@ -114,4 +110,4 @@ export class PropertyCollection extends Collection {
114
110
  return propertiesByCategory;
115
111
  }
116
112
  }
117
- //# 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';\r\nimport { Property, PropertyCategory } from '../../models';\r\nimport uniqBy from 'lodash/uniqBy';\r\nimport { DepreciationCollection } from '../depreciation.collection';\r\nimport { TransactionCollection } from '../transaction';\r\nimport { CollectionDictionary } from '../collection-dictionary';\r\nimport { PropertyCategoryMovementCollection } from './property-category-movement.collection';\r\n\r\nexport class PropertyCollection extends Collection<Property> {\r\n  /**\r\n   * Get new property collection filtered by active status\r\n   */\r\n  getActiveProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => property.isActive)\r\n    );\r\n  }\r\n\r\n  getCreatedProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => property.isOwn())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get new property collection filtered by shared\r\n   */\r\n  getSharedProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => !property.isOwn())\r\n    );\r\n  }\r\n\r\n  getUnsold(): this {\r\n    return this.create(\r\n      this.items.filter((property: Property) => !property.isSold())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get total purchase price for all properties in the collection\r\n   */\r\n  get purchasePrice(): number {\r\n    return this.sumBy('purchasePrice');\r\n  }\r\n\r\n  get growthPercent(): number {\r\n    return this.sumBy('growthPercent');\r\n  }\r\n\r\n  get marketValue(): number {\r\n    return this.sumBy('valuation.marketValue');\r\n  }\r\n\r\n  get sharedMarketValue(): number {\r\n    let sharedMarketValue = 0;\r\n    this.items.forEach(property => {\r\n      sharedMarketValue += property.shareRatio * property.marketValue\r\n    });\r\n\r\n    return sharedMarketValue;\r\n  }\r\n\r\n  /**\r\n   * Get list of unique property categories from collection\r\n   */\r\n  getCategories(): PropertyCategory[] {\r\n    return uniqBy(this.items.map((property: Property): PropertyCategory => property.category), 'id');\r\n  }\r\n\r\n  /**\r\n   * list of properties\r\n   */\r\n  getCGTApplicable(): this {\r\n    return this.create(\r\n      this.items.filter((property: Property) => property.isCGTApplicable())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get property with the highest growth percent\r\n   */\r\n  getBestPerformanceGrowthProperty(): Property {\r\n    return this.items.reduce((max: Property, current: Property): Property => max.growthPercent < current.growthPercent ? current : max, this.first);\r\n  }\r\n\r\n  /**\r\n   * Get property with the lowest tax position\r\n   */\r\n  getBestPerformanceTaxProperty(transactions: TransactionCollection, depreciations: DepreciationCollection): Property {\r\n    const transactionsByProperty: CollectionDictionary<TransactionCollection> = transactions.groupBy('property.id');\r\n    const depreciationsByProperty: CollectionDictionary<DepreciationCollection> = depreciations.groupBy('property.id');\r\n\r\n    return this.items.reduce((min: Property, current: Property): Property => {\r\n      const minTaxPosition: number = min.getTaxPosition(transactionsByProperty.get(min.id), depreciationsByProperty.get(min.id));\r\n      const currentTaxPosition: number = current.getTaxPosition(transactionsByProperty.get(current.id), depreciationsByProperty.get(current.id));\r\n\r\n      return minTaxPosition > currentTaxPosition ? current : min;\r\n    }, this.first);\r\n  }\r\n\r\n  /**\r\n   * @TODO Nicole not needed anymore?\r\n   * Show best performance properties first\r\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/217677997/Property+Tank+Dashboard\r\n   */\r\n  sortByBestPerformance(transactions: TransactionCollection, depreciations: DepreciationCollection): this {\r\n    const activeProperties: PropertyCollection = this.getActiveProperties();\r\n\r\n    // nothing to sort when no active properties\r\n    if (!activeProperties.length) {\r\n      return this;\r\n    }\r\n\r\n    const bestProperties: Property[] = uniqBy(this.create([\r\n      activeProperties.getBestPerformanceGrowthProperty(),\r\n      activeProperties.getBestPerformanceTaxProperty(transactions, depreciations)\r\n    ]).toArray(), 'id');\r\n\r\n    const newItems: Property[] = this.diff(bestProperties).toArray();\r\n    newItems.unshift(...bestProperties);\r\n\r\n    return this.create(newItems);\r\n  }\r\n\r\n  get netMonthlyIncome(): number {\r\n    return this.sumBy('monthlyIncome') + this.sumBy('monthlyExpense');\r\n  }\r\n\r\n  getOwnerOccupiedProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => property.category.isOwnerOccupied())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Properties that are taxed and will be included in reports (Tax summary, My tax report, e.t.c.)\r\n   */\r\n  getTaxInclusive(): this {\r\n    return this.create(\r\n      this.items.filter((property: Property) => property.category.isTaxInclusive)\r\n    );\r\n  }\r\n\r\n  groupByCategory(categories: Collection<PropertyCategory>, movements: PropertyCategoryMovementCollection, allowEmpty = false): Map<PropertyCategory, PropertyCollection> {\r\n    const categoryById = categories.indexBy('id');\r\n    const movementsByCategory = movements.getCurrent().groupBy('propertyCategory.id');\r\n\r\n    const propertiesByCategory = new Map<PropertyCategory, PropertyCollection>;\r\n    movementsByCategory.keys.forEach(key => {\r\n      propertiesByCategory.set(\r\n        categoryById.get(key),\r\n        this.filterBy('id', movementsByCategory.get(key).map(movement => movement.property.id))\r\n      );\r\n    });\r\n\r\n    if (allowEmpty) {\r\n      categories.remove(Array.from(propertiesByCategory.keys())).toArray().forEach(category => {\r\n        propertiesByCategory.set(category, new PropertyCollection());\r\n      });\r\n    }\r\n\r\n    return propertiesByCategory;\r\n  }\r\n}\r\n"]}
113
+ //# 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';\r\nimport { Property, PropertyCategory } from '../../models';\r\nimport uniqBy from 'lodash/uniqBy';\r\nimport { DepreciationCollection } from '../depreciation.collection';\r\nimport { TransactionCollection } from '../transaction';\r\nimport { CollectionDictionary } from '../collection-dictionary';\r\nimport { PropertyCategoryMovementCollection } from './property-category-movement.collection';\r\n\r\nexport class PropertyCollection extends Collection<Property> {\r\n  /**\r\n   * Get new property collection filtered by active status\r\n   */\r\n  getActiveProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => property.isActive)\r\n    );\r\n  }\r\n\r\n  getCreatedProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => property.isOwn())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get new property collection filtered by shared\r\n   */\r\n  getSharedProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => !property.isOwn())\r\n    );\r\n  }\r\n\r\n  getUnsold(): this {\r\n    return this.create(\r\n      this.items.filter((property: Property) => !property.isSold())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get total purchase price for all properties in the collection\r\n   */\r\n  get purchasePrice(): number {\r\n    return this.sumBy('purchasePrice');\r\n  }\r\n\r\n  get growthPercent(): number {\r\n    return this.sumBy('growthPercent');\r\n  }\r\n\r\n  get marketValue(): number {\r\n    return this.sumBy('marketValue');\r\n  }\r\n\r\n  get sharedMarketValue(): number {\r\n    return this.sumBy('sharedMarketValue');\r\n  }\r\n\r\n  /**\r\n   * Get list of unique property categories from collection\r\n   */\r\n  getCategories(): PropertyCategory[] {\r\n    return uniqBy(this.items.map((property: Property): PropertyCategory => property.category), 'id');\r\n  }\r\n\r\n  /**\r\n   * list of properties\r\n   */\r\n  getCGTApplicable(): this {\r\n    return this.create(\r\n      this.items.filter((property: Property) => property.isCGTApplicable())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get property with the highest growth percent\r\n   */\r\n  getBestPerformanceGrowthProperty(): Property {\r\n    return this.items.reduce((max: Property, current: Property): Property => max.growthPercent < current.growthPercent ? current : max, this.first);\r\n  }\r\n\r\n  /**\r\n   * Get property with the lowest tax position\r\n   */\r\n  getBestPerformanceTaxProperty(transactions: TransactionCollection, depreciations: DepreciationCollection): Property {\r\n    const transactionsByProperty: CollectionDictionary<TransactionCollection> = transactions.groupBy('property.id');\r\n    const depreciationsByProperty: CollectionDictionary<DepreciationCollection> = depreciations.groupBy('property.id');\r\n\r\n    return this.items.reduce((min: Property, current: Property): Property => {\r\n      const minTaxPosition: number = min.getTaxPosition(transactionsByProperty.get(min.id), depreciationsByProperty.get(min.id));\r\n      const currentTaxPosition: number = current.getTaxPosition(transactionsByProperty.get(current.id), depreciationsByProperty.get(current.id));\r\n\r\n      return minTaxPosition > currentTaxPosition ? current : min;\r\n    }, this.first);\r\n  }\r\n\r\n  /**\r\n   * @TODO Nicole not needed anymore?\r\n   * Show best performance properties first\r\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/217677997/Property+Tank+Dashboard\r\n   */\r\n  sortByBestPerformance(transactions: TransactionCollection, depreciations: DepreciationCollection): this {\r\n    const activeProperties: PropertyCollection = this.getActiveProperties();\r\n\r\n    // nothing to sort when no active properties\r\n    if (!activeProperties.length) {\r\n      return this;\r\n    }\r\n\r\n    const bestProperties: Property[] = uniqBy(this.create([\r\n      activeProperties.getBestPerformanceGrowthProperty(),\r\n      activeProperties.getBestPerformanceTaxProperty(transactions, depreciations)\r\n    ]).toArray(), 'id');\r\n\r\n    const newItems: Property[] = this.diff(bestProperties).toArray();\r\n    newItems.unshift(...bestProperties);\r\n\r\n    return this.create(newItems);\r\n  }\r\n\r\n  get netMonthlyIncome(): number {\r\n    return this.sumBy('monthlyIncome') + this.sumBy('monthlyExpense');\r\n  }\r\n\r\n  getOwnerOccupiedProperties(): PropertyCollection {\r\n    return new PropertyCollection(\r\n      this.items.filter((property: Property): boolean => property.category.isOwnerOccupied())\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Properties that are taxed and will be included in reports (Tax summary, My tax report, e.t.c.)\r\n   */\r\n  getTaxInclusive(): this {\r\n    return this.create(\r\n      this.items.filter((property: Property) => property.category.isTaxInclusive)\r\n    );\r\n  }\r\n\r\n  groupByCategory(categories: Collection<PropertyCategory>, movements: PropertyCategoryMovementCollection, allowEmpty = false): Map<PropertyCategory, PropertyCollection> {\r\n    const categoryById = categories.indexBy('id');\r\n    const movementsByCategory = movements.getCurrent().groupBy('propertyCategory.id');\r\n\r\n    const propertiesByCategory = new Map<PropertyCategory, PropertyCollection>;\r\n    movementsByCategory.keys.forEach(key => {\r\n      propertiesByCategory.set(\r\n        categoryById.get(key),\r\n        this.filterBy('id', movementsByCategory.get(key).map(movement => movement.property.id))\r\n      );\r\n    });\r\n\r\n    if (allowEmpty) {\r\n      categories.remove(Array.from(propertiesByCategory.keys())).toArray().forEach(category => {\r\n        propertiesByCategory.set(category, new PropertyCollection());\r\n      });\r\n    }\r\n\r\n    return propertiesByCategory;\r\n  }\r\n}\r\n"]}
@@ -18,6 +18,9 @@ export class TransactionBaseFilterForm extends FormGroup {
18
18
  });
19
19
  }
20
20
  filter(collection) {
21
+ if (!collection.length) {
22
+ return collection;
23
+ }
21
24
  const value = this.value;
22
25
  if (value.dateFrom) {
23
26
  collection = collection.filter(transaction => transaction.date >= value.dateFrom);
@@ -37,4 +40,4 @@ export class TransactionBaseFilterForm extends FormGroup {
37
40
  return collection;
38
41
  }
39
42
  }
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhbnNhY3Rpb24tYmFzZS1maWx0ZXIuZm9ybS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R0LWNvcmUvc3JjL2xpYi9mb3Jtcy90cmFuc2FjdGlvbi90cmFuc2FjdGlvbi1iYXNlLWZpbHRlci5mb3JtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxXQUFXLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFHeEQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLCtCQUErQixDQUFDO0FBYTdELE1BQU0sT0FBTyx5QkFBMEIsU0FBUSxTQUF5QztJQUN0RixZQUFZLFFBQXVCLEVBQUUsUUFBdUIsRUFBRSxVQUF1QjtRQUNuRixLQUFLLENBQUM7WUFDSixRQUFRLEVBQUUsSUFBSSxXQUFXLENBQUMsUUFBUSxDQUFDO1lBQ25DLFFBQVEsRUFBRSxJQUFJLFdBQVcsQ0FBQyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDO1lBQzlELFVBQVUsRUFBRSxJQUFJLFdBQVcsQ0FBQyxFQUFFLEtBQUssRUFBRSxVQUFVLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDO1lBQ2xFLFFBQVEsRUFBRSxJQUFJLFdBQVcsRUFBRTtZQUMzQixNQUFNLEVBQUUsSUFBSSxXQUFXLEVBQUU7U0FDMUIsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUFFRCxZQUFZO1FBQ1YsSUFBSSxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUMsQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxFQUFFO1lBQ3JELFFBQVEsS0FBSyxZQUFZLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ2hHLFFBQVEsS0FBSyxZQUFZLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQzFHLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVELE1BQU0sQ0FBcUYsVUFBMkI7UUFDcEgsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQztRQUV6QixJQUFJLEtBQUssQ0FBQyxRQUFRLEVBQUU7WUFDbEIsVUFBVSxHQUFHLFVBQVUsQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLEVBQUUsQ0FBQyxXQUFXLENBQUMsSUFBSSxJQUFJLEtBQUssQ0FBQyxRQUFRLENBQUMsQ0FBQztTQUNuRjtRQUNELElBQUksS0FBSyxDQUFDLE1BQU0sRUFBRTtZQUNoQixVQUFVLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQUMsRUFBRSxDQUFDLFdBQVcsQ0FBQyxJQUFJLElBQUksS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1NBQ2pGO1FBQ0QsSUFBSSxLQUFLLENBQUMsUUFBUSxFQUFFO1lBQ2xCLFVBQVUsR0FBRyxVQUFVLENBQUMsUUFBUSxDQUFDLFVBQVUsRUFBRSxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUM7U0FDOUQ7UUFDRCxJQUFJLEtBQUssQ0FBQyxRQUFRLEVBQUU7WUFDbEIsVUFBVSxHQUFHLFVBQVUsQ0FBQyxRQUFRLENBQUMsYUFBYSxFQUFFLEtBQUssQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDLENBQUM7U0FDcEU7UUFDRCxJQUFJLEtBQUssQ0FBQyxVQUFVLEVBQUU7WUFDcEIsVUFBVSxHQUFHLFVBQVUsQ0FBQyxRQUFRLENBQUMsYUFBYSxFQUFFLEtBQUssQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7U0FDaEc7UUFFRCxPQUFPLFVBQVUsQ0FBQztJQUNwQixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGb3JtQ29udHJvbCwgRm9ybUdyb3VwIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xyXG5pbXBvcnQgeyBDb250cm9sc0ludGVyZmFjZSB9IGZyb20gJy4uL2Fic3RyYWN0LmZvcm0nO1xyXG5pbXBvcnQgeyBDb2xsZWN0aW9uIH0gZnJvbSAnLi4vLi4vY29sbGVjdGlvbnMnO1xyXG5pbXBvcnQgeyBUYW5rVHlwZUVudW0gfSBmcm9tICcuLi8uLi9kYi9FbnVtcy90YW5rLXR5cGUuZW51bSc7XHJcbmltcG9ydCB7IFByb3BlcnR5LCBTb2xlQnVzaW5lc3MgfSBmcm9tICcuLi8uLi9tb2RlbHMnO1xyXG5pbXBvcnQgeyBJRXZlbnRMaXN0ZW5lciB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMnO1xyXG5pbXBvcnQgeyBUcmFuc2FjdGlvbkJhc2UgfSBmcm9tICcuLi8uLi9kYi9Nb2RlbHMnO1xyXG5cclxuaW50ZXJmYWNlIElUcmFuc2FjdGlvbkJhc2VGaWx0ZXJDb250cm9scyBleHRlbmRzIENvbnRyb2xzSW50ZXJmYWNlIHtcclxuICB0YW5rVHlwZTogRm9ybUNvbnRyb2w8VGFua1R5cGVFbnVtPjtcclxuICBidXNpbmVzczogRm9ybUNvbnRyb2w8U29sZUJ1c2luZXNzPjtcclxuICBwcm9wZXJ0aWVzOiBGb3JtQ29udHJvbDxQcm9wZXJ0eVtdPlxyXG4gIGRhdGVGcm9tOiBGb3JtQ29udHJvbDxEYXRlPjtcclxuICBkYXRlVG86IEZvcm1Db250cm9sPERhdGU+O1xyXG59XHJcblxyXG5leHBvcnQgY2xhc3MgVHJhbnNhY3Rpb25CYXNlRmlsdGVyRm9ybSBleHRlbmRzIEZvcm1Hcm91cDxJVHJhbnNhY3Rpb25CYXNlRmlsdGVyQ29udHJvbHM+IGltcGxlbWVudHMgSUV2ZW50TGlzdGVuZXIge1xyXG4gIGNvbnN0cnVjdG9yKHRhbmtUeXBlPzogVGFua1R5cGVFbnVtLCBidXNpbmVzcz86IFNvbGVCdXNpbmVzcywgcHJvcGVydGllcz86IFByb3BlcnR5W10pIHtcclxuICAgIHN1cGVyKHtcclxuICAgICAgdGFua1R5cGU6IG5ldyBGb3JtQ29udHJvbCh0YW5rVHlwZSksXHJcbiAgICAgIGJ1c2luZXNzOiBuZXcgRm9ybUNvbnRyb2woeyB2YWx1ZTogYnVzaW5lc3MsIGRpc2FibGVkOiB0cnVlIH0pLFxyXG4gICAgICBwcm9wZXJ0aWVzOiBuZXcgRm9ybUNvbnRyb2woeyB2YWx1ZTogcHJvcGVydGllcywgZGlzYWJsZWQ6IHRydWUgfSksXHJcbiAgICAgIGRhdGVGcm9tOiBuZXcgRm9ybUNvbnRyb2woKSxcclxuICAgICAgZGF0ZVRvOiBuZXcgRm9ybUNvbnRyb2woKSxcclxuICAgIH0pO1xyXG5cclxuICAgIHRoaXMubGlzdGVuRXZlbnRzKCk7XHJcbiAgfVxyXG5cclxuICBsaXN0ZW5FdmVudHMoKTogdm9pZCB7XHJcbiAgICB0aGlzLmdldCgndGFua1R5cGUnKS52YWx1ZUNoYW5nZXMuc3Vic2NyaWJlKHRhbmtUeXBlID0+IHtcclxuICAgICAgdGFua1R5cGUgPT09IFRhbmtUeXBlRW51bS5TT0xFID8gdGhpcy5nZXQoJ2J1c2luZXNzJykuZW5hYmxlKCkgOiB0aGlzLmdldCgnYnVzaW5lc3MnKS5kaXNhYmxlKCk7XHJcbiAgICAgIHRhbmtUeXBlID09PSBUYW5rVHlwZUVudW0uUFJPUEVSVFkgPyB0aGlzLmdldCgncHJvcGVydGllcycpLmVuYWJsZSgpIDogdGhpcy5nZXQoJ3Byb3BlcnRpZXMnKS5kaXNhYmxlKCk7XHJcbiAgICB9KTtcclxuICB9XHJcblxyXG4gIGZpbHRlcjxNb2RlbENsYXNzIGV4dGVuZHMgVHJhbnNhY3Rpb25CYXNlLCBDb2xsZWN0aW9uQ2xhc3MgZXh0ZW5kcyBDb2xsZWN0aW9uPE1vZGVsQ2xhc3M+Pihjb2xsZWN0aW9uOiBDb2xsZWN0aW9uQ2xhc3MpOiBDb2xsZWN0aW9uQ2xhc3Mge1xyXG4gICAgY29uc3QgdmFsdWUgPSB0aGlzLnZhbHVlO1xyXG5cclxuICAgIGlmICh2YWx1ZS5kYXRlRnJvbSkge1xyXG4gICAgICBjb2xsZWN0aW9uID0gY29sbGVjdGlvbi5maWx0ZXIodHJhbnNhY3Rpb24gPT4gdHJhbnNhY3Rpb24uZGF0ZSA+PSB2YWx1ZS5kYXRlRnJvbSk7XHJcbiAgICB9XHJcbiAgICBpZiAodmFsdWUuZGF0ZVRvKSB7XHJcbiAgICAgIGNvbGxlY3Rpb24gPSBjb2xsZWN0aW9uLmZpbHRlcih0cmFuc2FjdGlvbiA9PiB0cmFuc2FjdGlvbi5kYXRlIDw9IHZhbHVlLmRhdGVUbyk7XHJcbiAgICB9XHJcbiAgICBpZiAodmFsdWUudGFua1R5cGUpIHtcclxuICAgICAgY29sbGVjdGlvbiA9IGNvbGxlY3Rpb24uZmlsdGVyQnkoJ3RhbmtUeXBlJywgdmFsdWUudGFua1R5cGUpO1xyXG4gICAgfVxyXG4gICAgaWYgKHZhbHVlLmJ1c2luZXNzKSB7XHJcbiAgICAgIGNvbGxlY3Rpb24gPSBjb2xsZWN0aW9uLmZpbHRlckJ5KCdidXNpbmVzcy5pZCcsIHZhbHVlLmJ1c2luZXNzLmlkKTtcclxuICAgIH1cclxuICAgIGlmICh2YWx1ZS5wcm9wZXJ0aWVzKSB7XHJcbiAgICAgIGNvbGxlY3Rpb24gPSBjb2xsZWN0aW9uLmZpbHRlckJ5KCdwcm9wZXJ0eS5pZCcsIHZhbHVlLnByb3BlcnRpZXMubWFwKHByb3BlcnR5ID0+IHByb3BlcnR5LmlkKSk7XHJcbiAgICB9XHJcblxyXG4gICAgcmV0dXJuIGNvbGxlY3Rpb247XHJcbiAgfVxyXG59XHJcbiJdfQ==
43
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhbnNhY3Rpb24tYmFzZS1maWx0ZXIuZm9ybS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R0LWNvcmUvc3JjL2xpYi9mb3Jtcy90cmFuc2FjdGlvbi90cmFuc2FjdGlvbi1iYXNlLWZpbHRlci5mb3JtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxXQUFXLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFHeEQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLCtCQUErQixDQUFDO0FBYTdELE1BQU0sT0FBTyx5QkFBMEIsU0FBUSxTQUF5QztJQUN0RixZQUFZLFFBQXVCLEVBQUUsUUFBdUIsRUFBRSxVQUF1QjtRQUNuRixLQUFLLENBQUM7WUFDSixRQUFRLEVBQUUsSUFBSSxXQUFXLENBQUMsUUFBUSxDQUFDO1lBQ25DLFFBQVEsRUFBRSxJQUFJLFdBQVcsQ0FBQyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDO1lBQzlELFVBQVUsRUFBRSxJQUFJLFdBQVcsQ0FBQyxFQUFFLEtBQUssRUFBRSxVQUFVLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDO1lBQ2xFLFFBQVEsRUFBRSxJQUFJLFdBQVcsRUFBRTtZQUMzQixNQUFNLEVBQUUsSUFBSSxXQUFXLEVBQUU7U0FDMUIsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUFFRCxZQUFZO1FBQ1YsSUFBSSxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUMsQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxFQUFFO1lBQ3JELFFBQVEsS0FBSyxZQUFZLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ2hHLFFBQVEsS0FBSyxZQUFZLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQzFHLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVELE1BQU0sQ0FBcUYsVUFBMkI7UUFDcEgsSUFBSSxDQUFDLFVBQVUsQ0FBQyxNQUFNLEVBQUU7WUFDdEIsT0FBTyxVQUFVLENBQUM7U0FDbkI7UUFDRCxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDO1FBRXpCLElBQUksS0FBSyxDQUFDLFFBQVEsRUFBRTtZQUNsQixVQUFVLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQUMsRUFBRSxDQUFDLFdBQVcsQ0FBQyxJQUFJLElBQUksS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1NBQ25GO1FBQ0QsSUFBSSxLQUFLLENBQUMsTUFBTSxFQUFFO1lBQ2hCLFVBQVUsR0FBRyxVQUFVLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxFQUFFLENBQUMsV0FBVyxDQUFDLElBQUksSUFBSSxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDakY7UUFDRCxJQUFJLEtBQUssQ0FBQyxRQUFRLEVBQUU7WUFDbEIsVUFBVSxHQUFHLFVBQVUsQ0FBQyxRQUFRLENBQUMsVUFBVSxFQUFFLEtBQUssQ0FBQyxRQUFRLENBQUMsQ0FBQztTQUM5RDtRQUNELElBQUksS0FBSyxDQUFDLFFBQVEsRUFBRTtZQUNsQixVQUFVLEdBQUcsVUFBVSxDQUFDLFFBQVEsQ0FBQyxhQUFhLEVBQUUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsQ0FBQztTQUNwRTtRQUNELElBQUksS0FBSyxDQUFDLFVBQVUsRUFBRTtZQUNwQixVQUFVLEdBQUcsVUFBVSxDQUFDLFFBQVEsQ0FBQyxhQUFhLEVBQUUsS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztTQUNoRztRQUVELE9BQU8sVUFBVSxDQUFDO0lBQ3BCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEZvcm1Db250cm9sLCBGb3JtR3JvdXAgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XHJcbmltcG9ydCB7IENvbnRyb2xzSW50ZXJmYWNlIH0gZnJvbSAnLi4vYWJzdHJhY3QuZm9ybSc7XHJcbmltcG9ydCB7IENvbGxlY3Rpb24gfSBmcm9tICcuLi8uLi9jb2xsZWN0aW9ucyc7XHJcbmltcG9ydCB7IFRhbmtUeXBlRW51bSB9IGZyb20gJy4uLy4uL2RiL0VudW1zL3RhbmstdHlwZS5lbnVtJztcclxuaW1wb3J0IHsgUHJvcGVydHksIFNvbGVCdXNpbmVzcyB9IGZyb20gJy4uLy4uL21vZGVscyc7XHJcbmltcG9ydCB7IElFdmVudExpc3RlbmVyIH0gZnJvbSAnLi4vLi4vaW50ZXJmYWNlcyc7XHJcbmltcG9ydCB7IFRyYW5zYWN0aW9uQmFzZSB9IGZyb20gJy4uLy4uL2RiL01vZGVscyc7XHJcblxyXG5pbnRlcmZhY2UgSVRyYW5zYWN0aW9uQmFzZUZpbHRlckNvbnRyb2xzIGV4dGVuZHMgQ29udHJvbHNJbnRlcmZhY2Uge1xyXG4gIHRhbmtUeXBlOiBGb3JtQ29udHJvbDxUYW5rVHlwZUVudW0+O1xyXG4gIGJ1c2luZXNzOiBGb3JtQ29udHJvbDxTb2xlQnVzaW5lc3M+O1xyXG4gIHByb3BlcnRpZXM6IEZvcm1Db250cm9sPFByb3BlcnR5W10+XHJcbiAgZGF0ZUZyb206IEZvcm1Db250cm9sPERhdGU+O1xyXG4gIGRhdGVUbzogRm9ybUNvbnRyb2w8RGF0ZT47XHJcbn1cclxuXHJcbmV4cG9ydCBjbGFzcyBUcmFuc2FjdGlvbkJhc2VGaWx0ZXJGb3JtIGV4dGVuZHMgRm9ybUdyb3VwPElUcmFuc2FjdGlvbkJhc2VGaWx0ZXJDb250cm9scz4gaW1wbGVtZW50cyBJRXZlbnRMaXN0ZW5lciB7XHJcbiAgY29uc3RydWN0b3IodGFua1R5cGU/OiBUYW5rVHlwZUVudW0sIGJ1c2luZXNzPzogU29sZUJ1c2luZXNzLCBwcm9wZXJ0aWVzPzogUHJvcGVydHlbXSkge1xyXG4gICAgc3VwZXIoe1xyXG4gICAgICB0YW5rVHlwZTogbmV3IEZvcm1Db250cm9sKHRhbmtUeXBlKSxcclxuICAgICAgYnVzaW5lc3M6IG5ldyBGb3JtQ29udHJvbCh7IHZhbHVlOiBidXNpbmVzcywgZGlzYWJsZWQ6IHRydWUgfSksXHJcbiAgICAgIHByb3BlcnRpZXM6IG5ldyBGb3JtQ29udHJvbCh7IHZhbHVlOiBwcm9wZXJ0aWVzLCBkaXNhYmxlZDogdHJ1ZSB9KSxcclxuICAgICAgZGF0ZUZyb206IG5ldyBGb3JtQ29udHJvbCgpLFxyXG4gICAgICBkYXRlVG86IG5ldyBGb3JtQ29udHJvbCgpLFxyXG4gICAgfSk7XHJcblxyXG4gICAgdGhpcy5saXN0ZW5FdmVudHMoKTtcclxuICB9XHJcblxyXG4gIGxpc3RlbkV2ZW50cygpOiB2b2lkIHtcclxuICAgIHRoaXMuZ2V0KCd0YW5rVHlwZScpLnZhbHVlQ2hhbmdlcy5zdWJzY3JpYmUodGFua1R5cGUgPT4ge1xyXG4gICAgICB0YW5rVHlwZSA9PT0gVGFua1R5cGVFbnVtLlNPTEUgPyB0aGlzLmdldCgnYnVzaW5lc3MnKS5lbmFibGUoKSA6IHRoaXMuZ2V0KCdidXNpbmVzcycpLmRpc2FibGUoKTtcclxuICAgICAgdGFua1R5cGUgPT09IFRhbmtUeXBlRW51bS5QUk9QRVJUWSA/IHRoaXMuZ2V0KCdwcm9wZXJ0aWVzJykuZW5hYmxlKCkgOiB0aGlzLmdldCgncHJvcGVydGllcycpLmRpc2FibGUoKTtcclxuICAgIH0pO1xyXG4gIH1cclxuXHJcbiAgZmlsdGVyPE1vZGVsQ2xhc3MgZXh0ZW5kcyBUcmFuc2FjdGlvbkJhc2UsIENvbGxlY3Rpb25DbGFzcyBleHRlbmRzIENvbGxlY3Rpb248TW9kZWxDbGFzcz4+KGNvbGxlY3Rpb246IENvbGxlY3Rpb25DbGFzcyk6IENvbGxlY3Rpb25DbGFzcyB7XHJcbiAgICBpZiAoIWNvbGxlY3Rpb24ubGVuZ3RoKSB7XHJcbiAgICAgIHJldHVybiBjb2xsZWN0aW9uO1xyXG4gICAgfVxyXG4gICAgY29uc3QgdmFsdWUgPSB0aGlzLnZhbHVlO1xyXG5cclxuICAgIGlmICh2YWx1ZS5kYXRlRnJvbSkge1xyXG4gICAgICBjb2xsZWN0aW9uID0gY29sbGVjdGlvbi5maWx0ZXIodHJhbnNhY3Rpb24gPT4gdHJhbnNhY3Rpb24uZGF0ZSA+PSB2YWx1ZS5kYXRlRnJvbSk7XHJcbiAgICB9XHJcbiAgICBpZiAodmFsdWUuZGF0ZVRvKSB7XHJcbiAgICAgIGNvbGxlY3Rpb24gPSBjb2xsZWN0aW9uLmZpbHRlcih0cmFuc2FjdGlvbiA9PiB0cmFuc2FjdGlvbi5kYXRlIDw9IHZhbHVlLmRhdGVUbyk7XHJcbiAgICB9XHJcbiAgICBpZiAodmFsdWUudGFua1R5cGUpIHtcclxuICAgICAgY29sbGVjdGlvbiA9IGNvbGxlY3Rpb24uZmlsdGVyQnkoJ3RhbmtUeXBlJywgdmFsdWUudGFua1R5cGUpO1xyXG4gICAgfVxyXG4gICAgaWYgKHZhbHVlLmJ1c2luZXNzKSB7XHJcbiAgICAgIGNvbGxlY3Rpb24gPSBjb2xsZWN0aW9uLmZpbHRlckJ5KCdidXNpbmVzcy5pZCcsIHZhbHVlLmJ1c2luZXNzLmlkKTtcclxuICAgIH1cclxuICAgIGlmICh2YWx1ZS5wcm9wZXJ0aWVzKSB7XHJcbiAgICAgIGNvbGxlY3Rpb24gPSBjb2xsZWN0aW9uLmZpbHRlckJ5KCdwcm9wZXJ0eS5pZCcsIHZhbHVlLnByb3BlcnRpZXMubWFwKHByb3BlcnR5ID0+IHByb3BlcnR5LmlkKSk7XHJcbiAgICB9XHJcblxyXG4gICAgcmV0dXJuIGNvbGxlY3Rpb247XHJcbiAgfVxyXG59XHJcbiJdfQ==
@@ -47,6 +47,12 @@ export class Property extends PropertyBase {
47
47
  }
48
48
  return this.valuation?.marketValue || 0;
49
49
  }
50
+ get sharedMarketValue() {
51
+ if (this.isSold()) {
52
+ return 0;
53
+ }
54
+ return this.shareRatio * this.marketValue;
55
+ }
50
56
  get currentYearForecast() {
51
57
  const year = new FinancialYear().year;
52
58
  return this.forecasts.find((forecast) => forecast.financialYear === year)
@@ -247,4 +253,4 @@ __decorate([
247
253
  __decorate([
248
254
  Exclude()
249
255
  ], Property.prototype, "documentFile", void 0);
250
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/models/property/property.ts"],"names":[],"mappings":"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","sourcesContent":["import { Property as PropertyBase } from '../../db/Models/property/property';\r\nimport { PropertyShare } from '../../db/Models';\r\nimport { PropertySubscription } from './property-subscription';\r\nimport { Exclude, Type } from 'class-transformer';\r\nimport { Address } from '../address';\r\nimport { PropertyCategory } from './property-category';\r\nimport { PropertyValuation } from './property-valuation';\r\nimport { PropertyForecast } from './property-forecast';\r\nimport { User } from '../user';\r\nimport { TaxExemptionEnum } from '../../db/Enums/property/property-sale/tax-exemption.enum';\r\nimport { PropertyCategoryListEnum } from '../../db/Enums/property/property-category-list.enum';\r\nimport moment from 'moment';\r\nimport { PropertySale } from './property-sale';\r\nimport { unitOfTime } from 'moment';\r\nimport {\r\n  DepreciationCollection, PropertySaleTaxExemptionMetaFieldCollection, TransactionCollection\r\n} from '../../collections';\r\nimport round from 'lodash/round';\r\nimport { AppFile } from '../file';\r\nimport { PropertyCategoryMovement } from './property-category-movement';\r\nimport { FinancialYear } from '../financial-year';\r\nimport { plainToClass } from 'class-transformer';\r\n\r\n/**\r\n * propertySale docs - https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/4209508353/Property+Sold+button\r\n */\r\nexport class Property extends PropertyBase {\r\n  /**\r\n   * Property acquired before 20 September 1985 will generally be treated as pre-Capital Gains Tax (CGT) assets.\r\n   * Any assets acquired before this day are CGT exempt (because the tax didn't exist before this date).\r\n   */\r\n  static preCGTAssetDate: Date = new Date(1985, 9, 20);\r\n\r\n  @Type(() => Number)\r\n  purchasePrice: number;\r\n\r\n  @Type(() => Number)\r\n  growthPercent: number\r\n\r\n  @Type(() => Date)\r\n  contractDate: Date;\r\n\r\n  @Type(() => Date)\r\n  settlementDate: Date;\r\n\r\n  @Type(() => Address)\r\n  address: Address;\r\n\r\n  @Type(() => PropertyCategory)\r\n  category: PropertyCategory;\r\n\r\n  @Type(() => PropertyCategoryMovement)\r\n  categoryMovements: PropertyCategoryMovement[];\r\n\r\n  @Type(() => Number)\r\n  stampDuty?: number;\r\n\r\n  @Type(() => Number)\r\n  legalFees?: number;\r\n\r\n  @Type(() => Number)\r\n  otherCapitalCosts?: number;\r\n\r\n  @Type(() => PropertyForecast)\r\n  forecasts: PropertyForecast[];\r\n\r\n  @Type(() => PropertyShare)\r\n  shares: PropertyShare[];\r\n\r\n  @Type(() => PropertyValuation)\r\n  valuations: PropertyValuation[];\r\n\r\n  @Type(() => PropertyValuation)\r\n  valuation: PropertyValuation;\r\n\r\n  @Type(() => User)\r\n  user: User;\r\n\r\n  @Type(() => PropertySubscription)\r\n  subscriptions: PropertySubscription[];\r\n\r\n  @Type(() => AppFile)\r\n  file: AppFile;\r\n\r\n  @Exclude()\r\n  documentFile: File;\r\n\r\n  get name(): string {\r\n    return this.address.name;\r\n  }\r\n\r\n  get isActive(): boolean {\r\n    return !!this.subscriptions?.length;\r\n  }\r\n\r\n  isOwn(): boolean {\r\n    return this.user.id === +localStorage.getItem('userId');\r\n  }\r\n\r\n  isSold(): boolean {\r\n    return !!this.myShare?.sale;\r\n  }\r\n\r\n  get capitalCostsTotalAmount(): number {\r\n    return this.stampDuty + this.legalFees + this.otherCapitalCosts;\r\n  }\r\n\r\n  get marketValue(): number {\r\n    if (this.isSold()) {\r\n      return 0;\r\n    }\r\n\r\n    return this.valuation?.marketValue || 0;\r\n  }\r\n\r\n  get currentYearForecast(): PropertyForecast {\r\n    const year = new FinancialYear().year;\r\n\r\n    return this.forecasts.find((forecast: PropertyForecast): boolean => forecast.financialYear === year)\r\n      ?? plainToClass(PropertyForecast, {});\r\n  }\r\n\r\n  get forecastedRentalReturn(): number {\r\n    return this.currentYearForecast?.rentalReturn || 0;\r\n  }\r\n\r\n  get forecastedTaxPosition(): number {\r\n    return this.currentYearForecast?.taxPosition || 0;\r\n  }\r\n\r\n  get forecastedCashPosition(): number {\r\n    return this.currentYearForecast?.cashPosition || 0;\r\n  }\r\n\r\n  get myShare(): PropertyShare {\r\n    return this.shares.find((share: PropertyShare): boolean => share.user.id === +localStorage.getItem('userId'));\r\n  }\r\n\r\n  get claimPercent(): number {\r\n    return this.currentYearForecast.claimPercent;\r\n  }\r\n\r\n  get claimCoefficient(): number {\r\n    return this.claimPercent / 100;\r\n  }\r\n\r\n  get sharePercent(): number {\r\n    return this.myShare.percent;\r\n  }\r\n\r\n  get shareRatio(): number {\r\n    return this.myShare.percent / 100;\r\n  }\r\n\r\n  getCurrentSubscription(): PropertySubscription {\r\n    // always return first element because subscriptions array contains only one element\r\n    return this.subscriptions[0];\r\n  }\r\n\r\n  getForecastByYear(year: number): PropertyForecast {\r\n    return this.forecasts.find((forecast: PropertyForecast): boolean => forecast.financialYear === year);\r\n  }\r\n\r\n  get isShared(): boolean {\r\n    return this.shares.length > 1;\r\n  }\r\n\r\n  /**\r\n   * @TODO consider to move methods related with propertySale to separated class, since used just in one module yet\r\n   * purchase costs - claimed costs, except `ppr to investment` exemption,\r\n   * in that case it's equal to market value, when property became an investment property\r\n   */\r\n  calculateCostBase(sale: PropertySale): number {\r\n    const marketValue: number = new PropertySaleTaxExemptionMetaFieldCollection(sale.taxExemptionMetaFields).getMarketValue();\r\n    if (marketValue) {\r\n      return marketValue;\r\n    }\r\n\r\n    return this.purchasePrice + this.capitalCostsTotalAmount + sale.holdingCosts + sale.structuralImprovementsWDV - sale.buildingAtCostClaimed;\r\n  }\r\n\r\n  calculateGrossCapitalGain(sale: PropertySale): number {\r\n    return round((sale.netPrice - this.calculateCostBase(sale)) * this.getPartialCGTExemptionRatio(sale) * this.shareRatio, 2);\r\n  }\r\n\r\n  /**\r\n   * net capital gain includes tax exemption\r\n   */\r\n  calculateNetCapitalGain(sale: PropertySale): number {\r\n    return this.getCGTExemptionRatio(sale) * sale.grossCapitalGain;\r\n  }\r\n\r\n  /**\r\n   * guess tax exemption based on property details\r\n   */\r\n  getCGTExemption(sale: PropertySale): TaxExemptionEnum {\r\n    if (sale.taxExemption) {\r\n      return sale.taxExemption.id;\r\n    }\r\n\r\n    switch (true) {\r\n      // exemption for main residence properties\r\n      case this.category.id === PropertyCategoryListEnum.OWNER_OCCUPIED:\r\n        return TaxExemptionEnum.PPR;\r\n      // exemption for investment properties owned for at least one year\r\n      case this.isOneYearExemptionApplicable(sale):\r\n        return TaxExemptionEnum.ONE_YEAR_RULE;\r\n      default:\r\n        return null;\r\n    }\r\n  }\r\n\r\n  getPartialCGTExemptionRatio(sale: PropertySale): number {\r\n    const metaFields: PropertySaleTaxExemptionMetaFieldCollection = new PropertySaleTaxExemptionMetaFieldCollection(sale.taxExemptionMetaFields);\r\n\r\n    switch (sale.taxExemption?.id) {\r\n      case TaxExemptionEnum.INVESTMENT_TO_PPR:\r\n        const ownershipDays = this.getOwnershipDuration(sale);\r\n        return (ownershipDays - metaFields.getPPRDays()) / ownershipDays;\r\n      case TaxExemptionEnum.PPR_TO_INVESTMENT:\r\n        return metaFields.getClaimPercent() / 100;\r\n      default:\r\n        return 1;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * tax exemption can reduce cgt from 100% to less (up to zero)\r\n   */\r\n  getCGTExemptionRatio(sale: PropertySale): number {\r\n    switch (sale.taxExemption?.id) {\r\n      // 50% exemption for investments owned for at least one year\r\n      case TaxExemptionEnum.ONE_YEAR_RULE:\r\n        return 0.5;\r\n      case TaxExemptionEnum.INVESTMENT_TO_PPR:\r\n      case TaxExemptionEnum.PPR_TO_INVESTMENT:\r\n        return this.isOneYearExemptionApplicable(sale) ? 0.5 : 1;\r\n      // full exemption\r\n      case TaxExemptionEnum.PPR:\r\n      case TaxExemptionEnum.SIX_YEARS_RULE:\r\n      case TaxExemptionEnum.TRANSFER:\r\n      case TaxExemptionEnum.OTHER:\r\n        return 0;\r\n      // no exemption\r\n      default:\r\n        return 1;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * in cgt report we apply losses, which can reset 1 year rule (in case if there is no profit after loss)\r\n   */\r\n  isOneYearExemptionApplicable(sale: PropertySale): boolean {\r\n    return sale.grossCapitalGain > 0 && this.getOwnershipDuration(sale, 'years') > 0;\r\n  }\r\n\r\n  /**\r\n   * CGT is not applicable for properties acquired before 20.09.1985 (tax didn't exist before this date).\r\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/4644110466/Tax+Return+MyTax+-+Online+Form (\"Capital gains or losses\" section)\r\n   */\r\n  isCGTApplicable(): boolean {\r\n    return this.contractDate >= Property.preCGTAssetDate;\r\n  }\r\n\r\n  /**\r\n   * ownership duration from purchase till sale\r\n   */\r\n  getOwnershipDuration(sale: PropertySale, unitOfTime: unitOfTime.Diff = 'days'): number {\r\n    return moment(sale.date).diff(moment(this.contractDate), unitOfTime);\r\n  }\r\n\r\n  /**\r\n   * Tax Position = Income - Expense - Interest - Depreciation\r\n   * Where (Income - Expense - Interest) is cash position.\r\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/217415928/Dashboard+Property\r\n   */\r\n  getTaxPosition(transactions: TransactionCollection, depreciations: DepreciationCollection): number {\r\n    return transactions.grossClaimAmount - Math.abs(depreciations.claimAmount);\r\n  }\r\n\r\n  get monthlyIncome(): number {\r\n    return this.currentYearForecast.income / 12;\r\n  }\r\n\r\n  get monthlyExpense(): number {\r\n    return this.currentYearForecast.expense / 12;\r\n  }\r\n}\r\n"]}
256
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"property.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/models/property/property.ts"],"names":[],"mappings":"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","sourcesContent":["import { Property as PropertyBase } from '../../db/Models/property/property';\r\nimport { PropertyShare } from '../../db/Models';\r\nimport { PropertySubscription } from './property-subscription';\r\nimport { Exclude, Type } from 'class-transformer';\r\nimport { Address } from '../address';\r\nimport { PropertyCategory } from './property-category';\r\nimport { PropertyValuation } from './property-valuation';\r\nimport { PropertyForecast } from './property-forecast';\r\nimport { User } from '../user';\r\nimport { TaxExemptionEnum } from '../../db/Enums/property/property-sale/tax-exemption.enum';\r\nimport { PropertyCategoryListEnum } from '../../db/Enums/property/property-category-list.enum';\r\nimport moment from 'moment';\r\nimport { PropertySale } from './property-sale';\r\nimport { unitOfTime } from 'moment';\r\nimport {\r\n  DepreciationCollection, PropertySaleTaxExemptionMetaFieldCollection, TransactionCollection\r\n} from '../../collections';\r\nimport round from 'lodash/round';\r\nimport { AppFile } from '../file';\r\nimport { PropertyCategoryMovement } from './property-category-movement';\r\nimport { FinancialYear } from '../financial-year';\r\nimport { plainToClass } from 'class-transformer';\r\n\r\n/**\r\n * propertySale docs - https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/4209508353/Property+Sold+button\r\n */\r\nexport class Property extends PropertyBase {\r\n  /**\r\n   * Property acquired before 20 September 1985 will generally be treated as pre-Capital Gains Tax (CGT) assets.\r\n   * Any assets acquired before this day are CGT exempt (because the tax didn't exist before this date).\r\n   */\r\n  static preCGTAssetDate: Date = new Date(1985, 9, 20);\r\n\r\n  @Type(() => Number)\r\n  purchasePrice: number;\r\n\r\n  @Type(() => Number)\r\n  growthPercent: number\r\n\r\n  @Type(() => Date)\r\n  contractDate: Date;\r\n\r\n  @Type(() => Date)\r\n  settlementDate: Date;\r\n\r\n  @Type(() => Address)\r\n  address: Address;\r\n\r\n  @Type(() => PropertyCategory)\r\n  category: PropertyCategory;\r\n\r\n  @Type(() => PropertyCategoryMovement)\r\n  categoryMovements: PropertyCategoryMovement[];\r\n\r\n  @Type(() => Number)\r\n  stampDuty?: number;\r\n\r\n  @Type(() => Number)\r\n  legalFees?: number;\r\n\r\n  @Type(() => Number)\r\n  otherCapitalCosts?: number;\r\n\r\n  @Type(() => PropertyForecast)\r\n  forecasts: PropertyForecast[];\r\n\r\n  @Type(() => PropertyShare)\r\n  shares: PropertyShare[];\r\n\r\n  @Type(() => PropertyValuation)\r\n  valuations: PropertyValuation[];\r\n\r\n  @Type(() => PropertyValuation)\r\n  valuation: PropertyValuation;\r\n\r\n  @Type(() => User)\r\n  user: User;\r\n\r\n  @Type(() => PropertySubscription)\r\n  subscriptions: PropertySubscription[];\r\n\r\n  @Type(() => AppFile)\r\n  file: AppFile;\r\n\r\n  @Exclude()\r\n  documentFile: File;\r\n\r\n  get name(): string {\r\n    return this.address.name;\r\n  }\r\n\r\n  get isActive(): boolean {\r\n    return !!this.subscriptions?.length;\r\n  }\r\n\r\n  isOwn(): boolean {\r\n    return this.user.id === +localStorage.getItem('userId');\r\n  }\r\n\r\n  isSold(): boolean {\r\n    return !!this.myShare?.sale;\r\n  }\r\n\r\n  get capitalCostsTotalAmount(): number {\r\n    return this.stampDuty + this.legalFees + this.otherCapitalCosts;\r\n  }\r\n\r\n  get marketValue(): number {\r\n    if (this.isSold()) {\r\n      return 0;\r\n    }\r\n\r\n    return this.valuation?.marketValue || 0;\r\n  }\r\n\r\n  get sharedMarketValue(): number {\r\n    if (this.isSold()) {\r\n      return 0;\r\n    }\r\n\r\n    return this.shareRatio * this.marketValue;\r\n  }\r\n\r\n  get currentYearForecast(): PropertyForecast {\r\n    const year = new FinancialYear().year;\r\n\r\n    return this.forecasts.find((forecast: PropertyForecast): boolean => forecast.financialYear === year)\r\n      ?? plainToClass(PropertyForecast, {});\r\n  }\r\n\r\n  get forecastedRentalReturn(): number {\r\n    return this.currentYearForecast?.rentalReturn || 0;\r\n  }\r\n\r\n  get forecastedTaxPosition(): number {\r\n    return this.currentYearForecast?.taxPosition || 0;\r\n  }\r\n\r\n  get forecastedCashPosition(): number {\r\n    return this.currentYearForecast?.cashPosition || 0;\r\n  }\r\n\r\n  get myShare(): PropertyShare {\r\n    return this.shares.find((share: PropertyShare): boolean => share.user.id === +localStorage.getItem('userId'));\r\n  }\r\n\r\n  get claimPercent(): number {\r\n    return this.currentYearForecast.claimPercent;\r\n  }\r\n\r\n  get claimCoefficient(): number {\r\n    return this.claimPercent / 100;\r\n  }\r\n\r\n  get sharePercent(): number {\r\n    return this.myShare.percent;\r\n  }\r\n\r\n  get shareRatio(): number {\r\n    return this.myShare.percent / 100;\r\n  }\r\n\r\n  getCurrentSubscription(): PropertySubscription {\r\n    // always return first element because subscriptions array contains only one element\r\n    return this.subscriptions[0];\r\n  }\r\n\r\n  getForecastByYear(year: number): PropertyForecast {\r\n    return this.forecasts.find((forecast: PropertyForecast): boolean => forecast.financialYear === year);\r\n  }\r\n\r\n  get isShared(): boolean {\r\n    return this.shares.length > 1;\r\n  }\r\n\r\n  /**\r\n   * @TODO consider to move methods related with propertySale to separated class, since used just in one module yet\r\n   * purchase costs - claimed costs, except `ppr to investment` exemption,\r\n   * in that case it's equal to market value, when property became an investment property\r\n   */\r\n  calculateCostBase(sale: PropertySale): number {\r\n    const marketValue: number = new PropertySaleTaxExemptionMetaFieldCollection(sale.taxExemptionMetaFields).getMarketValue();\r\n    if (marketValue) {\r\n      return marketValue;\r\n    }\r\n\r\n    return this.purchasePrice + this.capitalCostsTotalAmount + sale.holdingCosts + sale.structuralImprovementsWDV - sale.buildingAtCostClaimed;\r\n  }\r\n\r\n  calculateGrossCapitalGain(sale: PropertySale): number {\r\n    return round((sale.netPrice - this.calculateCostBase(sale)) * this.getPartialCGTExemptionRatio(sale) * this.shareRatio, 2);\r\n  }\r\n\r\n  /**\r\n   * net capital gain includes tax exemption\r\n   */\r\n  calculateNetCapitalGain(sale: PropertySale): number {\r\n    return this.getCGTExemptionRatio(sale) * sale.grossCapitalGain;\r\n  }\r\n\r\n  /**\r\n   * guess tax exemption based on property details\r\n   */\r\n  getCGTExemption(sale: PropertySale): TaxExemptionEnum {\r\n    if (sale.taxExemption) {\r\n      return sale.taxExemption.id;\r\n    }\r\n\r\n    switch (true) {\r\n      // exemption for main residence properties\r\n      case this.category.id === PropertyCategoryListEnum.OWNER_OCCUPIED:\r\n        return TaxExemptionEnum.PPR;\r\n      // exemption for investment properties owned for at least one year\r\n      case this.isOneYearExemptionApplicable(sale):\r\n        return TaxExemptionEnum.ONE_YEAR_RULE;\r\n      default:\r\n        return null;\r\n    }\r\n  }\r\n\r\n  getPartialCGTExemptionRatio(sale: PropertySale): number {\r\n    const metaFields: PropertySaleTaxExemptionMetaFieldCollection = new PropertySaleTaxExemptionMetaFieldCollection(sale.taxExemptionMetaFields);\r\n\r\n    switch (sale.taxExemption?.id) {\r\n      case TaxExemptionEnum.INVESTMENT_TO_PPR:\r\n        const ownershipDays = this.getOwnershipDuration(sale);\r\n        return (ownershipDays - metaFields.getPPRDays()) / ownershipDays;\r\n      case TaxExemptionEnum.PPR_TO_INVESTMENT:\r\n        return metaFields.getClaimPercent() / 100;\r\n      default:\r\n        return 1;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * tax exemption can reduce cgt from 100% to less (up to zero)\r\n   */\r\n  getCGTExemptionRatio(sale: PropertySale): number {\r\n    switch (sale.taxExemption?.id) {\r\n      // 50% exemption for investments owned for at least one year\r\n      case TaxExemptionEnum.ONE_YEAR_RULE:\r\n        return 0.5;\r\n      case TaxExemptionEnum.INVESTMENT_TO_PPR:\r\n      case TaxExemptionEnum.PPR_TO_INVESTMENT:\r\n        return this.isOneYearExemptionApplicable(sale) ? 0.5 : 1;\r\n      // full exemption\r\n      case TaxExemptionEnum.PPR:\r\n      case TaxExemptionEnum.SIX_YEARS_RULE:\r\n      case TaxExemptionEnum.TRANSFER:\r\n      case TaxExemptionEnum.OTHER:\r\n        return 0;\r\n      // no exemption\r\n      default:\r\n        return 1;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * in cgt report we apply losses, which can reset 1 year rule (in case if there is no profit after loss)\r\n   */\r\n  isOneYearExemptionApplicable(sale: PropertySale): boolean {\r\n    return sale.grossCapitalGain > 0 && this.getOwnershipDuration(sale, 'years') > 0;\r\n  }\r\n\r\n  /**\r\n   * CGT is not applicable for properties acquired before 20.09.1985 (tax didn't exist before this date).\r\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/4644110466/Tax+Return+MyTax+-+Online+Form (\"Capital gains or losses\" section)\r\n   */\r\n  isCGTApplicable(): boolean {\r\n    return this.contractDate >= Property.preCGTAssetDate;\r\n  }\r\n\r\n  /**\r\n   * ownership duration from purchase till sale\r\n   */\r\n  getOwnershipDuration(sale: PropertySale, unitOfTime: unitOfTime.Diff = 'days'): number {\r\n    return moment(sale.date).diff(moment(this.contractDate), unitOfTime);\r\n  }\r\n\r\n  /**\r\n   * Tax Position = Income - Expense - Interest - Depreciation\r\n   * Where (Income - Expense - Interest) is cash position.\r\n   * https://taxtank.atlassian.net/wiki/spaces/TAXTANK/pages/217415928/Dashboard+Property\r\n   */\r\n  getTaxPosition(transactions: TransactionCollection, depreciations: DepreciationCollection): number {\r\n    return transactions.grossClaimAmount - Math.abs(depreciations.claimAmount);\r\n  }\r\n\r\n  get monthlyIncome(): number {\r\n    return this.currentYearForecast.income / 12;\r\n  }\r\n\r\n  get monthlyExpense(): number {\r\n    return this.currentYearForecast.expense / 12;\r\n  }\r\n}\r\n"]}
@@ -2339,14 +2339,10 @@ class PropertyCollection extends Collection {
2339
2339
  return this.sumBy('growthPercent');
2340
2340
  }
2341
2341
  get marketValue() {
2342
- return this.sumBy('valuation.marketValue');
2342
+ return this.sumBy('marketValue');
2343
2343
  }
2344
2344
  get sharedMarketValue() {
2345
- let sharedMarketValue = 0;
2346
- this.items.forEach(property => {
2347
- sharedMarketValue += property.shareRatio * property.marketValue;
2348
- });
2349
- return sharedMarketValue;
2345
+ return this.sumBy('sharedMarketValue');
2350
2346
  }
2351
2347
  /**
2352
2348
  * Get list of unique property categories from collection
@@ -5800,6 +5796,12 @@ class Property extends Property$1 {
5800
5796
  }
5801
5797
  return this.valuation?.marketValue || 0;
5802
5798
  }
5799
+ get sharedMarketValue() {
5800
+ if (this.isSold()) {
5801
+ return 0;
5802
+ }
5803
+ return this.shareRatio * this.marketValue;
5804
+ }
5803
5805
  get currentYearForecast() {
5804
5806
  const year = new FinancialYear().year;
5805
5807
  return this.forecasts.find((forecast) => forecast.financialYear === year)
@@ -9504,6 +9506,9 @@ class HoldingCollection extends ExportableCollection {
9504
9506
  getUnsold() {
9505
9507
  return this.removeBy('currentQuantity', 0);
9506
9508
  }
9509
+ getUnrealisedProfit() {
9510
+ return this.marketValue - this.getPurchaseValue();
9511
+ }
9507
9512
  }
9508
9513
 
9509
9514
  class HoldingSaleCollection extends ExportableCollection {
@@ -23861,6 +23866,9 @@ class TransactionBaseFilterForm extends FormGroup {
23861
23866
  });
23862
23867
  }
23863
23868
  filter(collection) {
23869
+ if (!collection.length) {
23870
+ return collection;
23871
+ }
23864
23872
  const value = this.value;
23865
23873
  if (value.dateFrom) {
23866
23874
  collection = collection.filter(transaction => transaction.date >= value.dateFrom);