@novha/calc-engines 1.4.0 → 1.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -2,10 +2,12 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.SouthAfricaIncomeTaxServiceImpl = void 0;
4
4
  class SouthAfricaIncomeTaxServiceImpl {
5
- constructor(income, age, rules) {
5
+ constructor(income, age, rules, medicalAidMembers = 0) {
6
+ this._medicalAidMembers = 0;
6
7
  this._income = income;
7
8
  this._age = age;
8
9
  this._rules = rules;
10
+ this._medicalAidMembers = medicalAidMembers;
9
11
  }
10
12
  calculateNetIncome() {
11
13
  const grossIncome = this._income;
@@ -22,7 +24,8 @@ class SouthAfricaIncomeTaxServiceImpl {
22
24
  }
23
25
  const grossTax = this.calculateBracketTax(this._income);
24
26
  const rebate = this.getRebate(this._age);
25
- const incomeTax = Math.max(0, grossTax - rebate);
27
+ const medicalAidCredit = this.calculateMedicalAidCredit(this._medicalAidMembers);
28
+ const incomeTax = Math.max(0, grossTax - rebate - medicalAidCredit);
26
29
  const uif = this.calculateUif(this._income);
27
30
  const totalDeductions = incomeTax + uif;
28
31
  const netIncome = grossIncome - totalDeductions;
@@ -71,9 +74,24 @@ class SouthAfricaIncomeTaxServiceImpl {
71
74
  const cappedIncome = Math.min(income, this._rules.uif.annualIncomeCap);
72
75
  return Math.min(cappedIncome * this._rules.uif.rate, this._rules.uif.maxAnnualContribution);
73
76
  }
77
+ calculateMedicalAidCredit(members) {
78
+ if (members <= 0)
79
+ return 0;
80
+ const monthly = this._rules.medicalAidTaxCredit.monthly;
81
+ let monthlyCredit = monthly.taxpayer;
82
+ if (members >= 2) {
83
+ monthlyCredit += monthly.firstDependant;
84
+ }
85
+ if (members > 2) {
86
+ monthlyCredit +=
87
+ (members - 2) * monthly.additionalDependant;
88
+ }
89
+ return (monthlyCredit *
90
+ this._rules.medicalAidTaxCredit.annualMultiplier);
91
+ }
74
92
  round(value, decimals = 2) {
75
93
  return Number(value.toFixed(decimals));
76
94
  }
77
95
  }
78
96
  exports.SouthAfricaIncomeTaxServiceImpl = SouthAfricaIncomeTaxServiceImpl;
79
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SouthAfricaIncomeTaxServiceImpl.js","sourceRoot":"","sources":["../../../src/income-tax/south-africa/SouthAfricaIncomeTaxServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComputedIncomeTaxValues, IncomeTaxRules } from \"./domain/types\";\nimport { SouthAfricaIncomeTaxService } from \"./SouthAfricaIncomeTaxService\";\n\nexport class SouthAfricaIncomeTaxServiceImpl implements SouthAfricaIncomeTaxService {\n    private _income: number;\n    private _age: number;\n    private _rules: IncomeTaxRules;\n\n    constructor(income: number, age: number, rules: IncomeTaxRules) {\n        this._income = income;\n        this._age = age;\n        this._rules = rules;\n    }\n\n    calculateNetIncome(): ComputedIncomeTaxValues {\n        const grossIncome = this._income;\n\n        const threshold = this.getTaxThreshold(this._age);\n\n        if (this._income <= threshold) {\n            return {\n                grossIncome,\n                incomeTax: 0,\n                uif: 0,\n                totalDeductions: 0,\n                netIncome: grossIncome,\n                effectiveTaxRate: 0,\n            };\n        }\n\n        const grossTax = this.calculateBracketTax(this._income);\n\n        const rebate = this.getRebate(this._age);\n        const incomeTax = Math.max(0, grossTax - rebate);\n\n        const uif = this.calculateUif(this._income);\n\n        const totalDeductions = incomeTax + uif;\n        const netIncome = grossIncome - totalDeductions;\n        const effectiveTaxRate = incomeTax / grossIncome;\n\n        return {\n            grossIncome,\n            incomeTax: this.round(incomeTax),\n            uif: this.round(uif),\n            totalDeductions: this.round(totalDeductions),\n            netIncome: this.round(netIncome),\n            effectiveTaxRate: this.round(effectiveTaxRate, 4),\n        };\n    }\n\n    private calculateBracketTax(income: number): number {\n        let tax = 0;\n\n        for (const bracket of this._rules.taxBrackets) {\n            if (income <= bracket.from) break;\n\n            const upper = bracket.to ?? income;\n            const taxableAmount = Math.min(upper, income) - bracket.from;\n\n            if (taxableAmount > 0) {\n                tax += taxableAmount * bracket.rate;\n            }\n        }\n\n        return tax;\n    }\n\n    private getTaxThreshold(age: number): number {\n        const thresholds = this._rules.taxThresholds;\n\n        if (age >= 75) return thresholds.age75Plus;\n        if (age >= 65) return thresholds.age65To74;\n\n        return thresholds.under65;\n    }\n\n    private getRebate(age: number): number {\n        let rebate = this._rules.rebates.primary.amount;\n\n        if (age >= this._rules.rebates.secondary.ageMin) {\n            rebate += this._rules.rebates.secondary.amount;\n        }\n\n        if (age >= this._rules.rebates.tertiary.ageMin) {\n            rebate += this._rules.rebates.tertiary.amount;\n        }\n\n        return rebate;\n    }\n\n    private calculateUif(income: number): number {\n        const cappedIncome = Math.min(\n            income,\n            this._rules.uif.annualIncomeCap,\n        );\n\n        return Math.min(\n            cappedIncome * this._rules.uif.rate,\n            this._rules.uif.maxAnnualContribution,\n        );\n    }\n\n    private round(value: number, decimals = 2): number {\n        return Number(value.toFixed(decimals));\n    }\n}"]}
97
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SouthAfricaIncomeTaxServiceImpl.js","sourceRoot":"","sources":["../../../src/income-tax/south-africa/SouthAfricaIncomeTaxServiceImpl.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComputedIncomeTaxValues, IncomeTaxRules } from \"./domain/types\";\nimport { SouthAfricaIncomeTaxService } from \"./SouthAfricaIncomeTaxService\";\n\nexport class SouthAfricaIncomeTaxServiceImpl implements SouthAfricaIncomeTaxService {\n    private _income: number;\n    private _age: number;\n    private _rules: IncomeTaxRules;\n    private _medicalAidMembers: number = 0;\n\n    constructor(income: number, age: number, rules: IncomeTaxRules, medicalAidMembers: number = 0) {\n        this._income = income;\n        this._age = age;\n        this._rules = rules;\n        this._medicalAidMembers = medicalAidMembers;\n    }\n\n    calculateNetIncome(): ComputedIncomeTaxValues {\n        const grossIncome = this._income;\n\n        const threshold = this.getTaxThreshold(this._age);\n\n        if (this._income <= threshold) {\n            return {\n                grossIncome,\n                incomeTax: 0,\n                uif: 0,\n                totalDeductions: 0,\n                netIncome: grossIncome,\n                effectiveTaxRate: 0,\n            };\n        }\n\n        const grossTax = this.calculateBracketTax(this._income);\n\n        const rebate = this.getRebate(this._age);\n        const medicalAidCredit =\n            this.calculateMedicalAidCredit(this._medicalAidMembers);\n        const incomeTax = Math.max(0, grossTax - rebate - medicalAidCredit);\n\n        const uif = this.calculateUif(this._income);\n\n        const totalDeductions = incomeTax + uif;\n        const netIncome = grossIncome - totalDeductions;\n        const effectiveTaxRate = incomeTax / grossIncome;\n\n        return {\n            grossIncome,\n            incomeTax: this.round(incomeTax),\n            uif: this.round(uif),\n            totalDeductions: this.round(totalDeductions),\n            netIncome: this.round(netIncome),\n            effectiveTaxRate: this.round(effectiveTaxRate, 4),\n        };\n    }\n\n    private calculateBracketTax(income: number): number {\n        let tax = 0;\n\n        for (const bracket of this._rules.taxBrackets) {\n            if (income <= bracket.from) break;\n\n            const upper = bracket.to ?? income;\n            const taxableAmount = Math.min(upper, income) - bracket.from;\n\n            if (taxableAmount > 0) {\n                tax += taxableAmount * bracket.rate;\n            }\n        }\n\n        return tax;\n    }\n\n    private getTaxThreshold(age: number): number {\n        const thresholds = this._rules.taxThresholds;\n\n        if (age >= 75) return thresholds.age75Plus;\n        if (age >= 65) return thresholds.age65To74;\n\n        return thresholds.under65;\n    }\n\n    private getRebate(age: number): number {\n        let rebate = this._rules.rebates.primary.amount;\n\n        if (age >= this._rules.rebates.secondary.ageMin) {\n            rebate += this._rules.rebates.secondary.amount;\n        }\n\n        if (age >= this._rules.rebates.tertiary.ageMin) {\n            rebate += this._rules.rebates.tertiary.amount;\n        }\n\n        return rebate;\n    }\n\n    private calculateUif(income: number): number {\n        const cappedIncome = Math.min(\n            income,\n            this._rules.uif.annualIncomeCap,\n        );\n\n        return Math.min(\n            cappedIncome * this._rules.uif.rate,\n            this._rules.uif.maxAnnualContribution,\n        );\n    }\n\n    private calculateMedicalAidCredit(members: number): number {\n        if (members <= 0) return 0;\n\n        const monthly = this._rules.medicalAidTaxCredit.monthly;\n\n        let monthlyCredit = monthly.taxpayer;\n\n        if (members >= 2) {\n            monthlyCredit += monthly.firstDependant;\n        }\n\n        if (members > 2) {\n            monthlyCredit +=\n                (members - 2) * monthly.additionalDependant;\n        }\n\n        return (\n            monthlyCredit *\n            this._rules.medicalAidTaxCredit.annualMultiplier\n        );\n    }\n\n    private round(value: number, decimals = 2): number {\n        return Number(value.toFixed(decimals));\n    }\n}"]}
@@ -4,11 +4,13 @@ export declare class SouthAfricaIncomeTaxServiceImpl implements SouthAfricaIncom
4
4
  private _income;
5
5
  private _age;
6
6
  private _rules;
7
- constructor(income: number, age: number, rules: IncomeTaxRules);
7
+ private _medicalAidMembers;
8
+ constructor(income: number, age: number, rules: IncomeTaxRules, medicalAidMembers?: number);
8
9
  calculateNetIncome(): ComputedIncomeTaxValues;
9
10
  private calculateBracketTax;
10
11
  private getTaxThreshold;
11
12
  private getRebate;
12
13
  private calculateUif;
14
+ private calculateMedicalAidCredit;
13
15
  private round;
14
16
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@novha/calc-engines",
3
- "version": "1.4.0",
3
+ "version": "1.4.1",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/types/index.d.ts",
6
6
  "scripts": {
@@ -5,11 +5,13 @@ export class SouthAfricaIncomeTaxServiceImpl implements SouthAfricaIncomeTaxServ
5
5
  private _income: number;
6
6
  private _age: number;
7
7
  private _rules: IncomeTaxRules;
8
+ private _medicalAidMembers: number = 0;
8
9
 
9
- constructor(income: number, age: number, rules: IncomeTaxRules) {
10
+ constructor(income: number, age: number, rules: IncomeTaxRules, medicalAidMembers: number = 0) {
10
11
  this._income = income;
11
12
  this._age = age;
12
13
  this._rules = rules;
14
+ this._medicalAidMembers = medicalAidMembers;
13
15
  }
14
16
 
15
17
  calculateNetIncome(): ComputedIncomeTaxValues {
@@ -31,7 +33,9 @@ export class SouthAfricaIncomeTaxServiceImpl implements SouthAfricaIncomeTaxServ
31
33
  const grossTax = this.calculateBracketTax(this._income);
32
34
 
33
35
  const rebate = this.getRebate(this._age);
34
- const incomeTax = Math.max(0, grossTax - rebate);
36
+ const medicalAidCredit =
37
+ this.calculateMedicalAidCredit(this._medicalAidMembers);
38
+ const incomeTax = Math.max(0, grossTax - rebate - medicalAidCredit);
35
39
 
36
40
  const uif = this.calculateUif(this._income);
37
41
 
@@ -101,6 +105,28 @@ export class SouthAfricaIncomeTaxServiceImpl implements SouthAfricaIncomeTaxServ
101
105
  );
102
106
  }
103
107
 
108
+ private calculateMedicalAidCredit(members: number): number {
109
+ if (members <= 0) return 0;
110
+
111
+ const monthly = this._rules.medicalAidTaxCredit.monthly;
112
+
113
+ let monthlyCredit = monthly.taxpayer;
114
+
115
+ if (members >= 2) {
116
+ monthlyCredit += monthly.firstDependant;
117
+ }
118
+
119
+ if (members > 2) {
120
+ monthlyCredit +=
121
+ (members - 2) * monthly.additionalDependant;
122
+ }
123
+
124
+ return (
125
+ monthlyCredit *
126
+ this._rules.medicalAidTaxCredit.annualMultiplier
127
+ );
128
+ }
129
+
104
130
  private round(value: number, decimals = 2): number {
105
131
  return Number(value.toFixed(decimals));
106
132
  }