Fix tax calculation (#14635)

* Salary Detail - field is_additional_component

* TDS fix - exclude additional component from monthly salary, but add to annual income to calculate tax

* fix indentation
diff --git a/erpnext/hr/doctype/additional_salary/additional_salary.py b/erpnext/hr/doctype/additional_salary/additional_salary.py
index ac872ae..90f6c0b 100644
--- a/erpnext/hr/doctype/additional_salary/additional_salary.py
+++ b/erpnext/hr/doctype/additional_salary/additional_salary.py
@@ -66,6 +66,7 @@
 			struct_row['do_not_include_in_total'] = salary_component.do_not_include_in_total
 			struct_row['is_tax_applicable'] = salary_component.is_tax_applicable
 			struct_row['variable_based_on_taxable_salary'] = salary_component.variable_based_on_taxable_salary
+			struct_row['is_additional_component'] = salary_component.is_additional_component
 			additional_components_dict['amount'] = amount
 			additional_components_dict['struct_row'] = struct_row
 			additional_components_dict['type'] = salary_component.type
diff --git a/erpnext/hr/doctype/salary_detail/salary_detail.json b/erpnext/hr/doctype/salary_detail/salary_detail.json
index 4212f4a..cca3a87 100644
--- a/erpnext/hr/doctype/salary_detail/salary_detail.json
+++ b/erpnext/hr/doctype/salary_detail/salary_detail.json
@@ -219,6 +219,40 @@
    "collapsible": 0, 
    "columns": 0, 
    "default": "", 
+   "fetch_from": "salary_component.is_additional_component", 
+   "fieldname": "is_additional_component", 
+   "fieldtype": "Check", 
+   "hidden": 1, 
+   "ignore_user_permissions": 0, 
+   "ignore_xss_filter": 0, 
+   "in_filter": 0, 
+   "in_global_search": 0, 
+   "in_list_view": 0, 
+   "in_standard_filter": 0, 
+   "label": "Is Additional Component", 
+   "length": 0, 
+   "no_copy": 0, 
+   "permlevel": 0, 
+   "precision": "", 
+   "print_hide": 0, 
+   "print_hide_if_no_value": 0, 
+   "read_only": 1, 
+   "remember_last_selected_value": 0, 
+   "report_hide": 0, 
+   "reqd": 0, 
+   "search_index": 0, 
+   "set_only_once": 0, 
+   "translatable": 0, 
+   "unique": 0
+  }, 
+  {
+   "allow_bulk_edit": 0, 
+   "allow_in_quick_entry": 0, 
+   "allow_on_submit": 0, 
+   "bold": 0, 
+   "collapsible": 0, 
+   "columns": 0, 
+   "default": "", 
    "fetch_from": "salary_component.variable_based_on_taxable_salary", 
    "fieldname": "variable_based_on_taxable_salary", 
    "fieldtype": "Check", 
@@ -591,7 +625,7 @@
  "issingle": 0, 
  "istable": 1, 
  "max_attachments": 0, 
- "modified": "2018-06-05 13:11:04.078930", 
+ "modified": "2018-06-22 12:31:55.516982", 
  "modified_by": "Administrator", 
  "module": "HR", 
  "name": "Salary Detail", 
@@ -606,4 +640,4 @@
  "sort_order": "DESC", 
  "track_changes": 0, 
  "track_seen": 0
-}
+}
\ No newline at end of file
diff --git a/erpnext/hr/doctype/salary_slip/salary_slip.js b/erpnext/hr/doctype/salary_slip/salary_slip.js
index bee670c..755bb40 100644
--- a/erpnext/hr/doctype/salary_slip/salary_slip.js
+++ b/erpnext/hr/doctype/salary_slip/salary_slip.js
@@ -12,7 +12,7 @@
 				{fieldname: 'amount', columns: 4}
 			];
 		})
-		
+
 		frm.fields_dict["timesheets"].grid.get_field("time_sheet").get_query = function(){
 			return {
 				filters: {
@@ -69,7 +69,7 @@
 		frm.trigger("toggle_fields")
 		frm.trigger("toggle_reqd_fields")
 		var salary_detail_fields = ["formula", "abbr", "statistical_component", "is_tax_applicable",
-			"is_flexible_benefit", "variable_based_on_taxable_salary"]
+			"is_flexible_benefit", "variable_based_on_taxable_salary", "is_additional_component"]
 		cur_frm.fields_dict['earnings'].grid.set_column_disp(salary_detail_fields,false);
 		cur_frm.fields_dict['deductions'].grid.set_column_disp(salary_detail_fields,false);
 	},
diff --git a/erpnext/hr/doctype/salary_slip/salary_slip.py b/erpnext/hr/doctype/salary_slip/salary_slip.py
index 8f58fe5..e0aef9d 100644
--- a/erpnext/hr/doctype/salary_slip/salary_slip.py
+++ b/erpnext/hr/doctype/salary_slip/salary_slip.py
@@ -114,7 +114,8 @@
 				'do_not_include_in_total' : struct_row.do_not_include_in_total,
 				'is_tax_applicable': struct_row.is_tax_applicable,
 				'is_flexible_benefit': struct_row.is_flexible_benefit,
-				'variable_based_on_taxable_salary': struct_row.variable_based_on_taxable_salary
+				'variable_based_on_taxable_salary': struct_row.variable_based_on_taxable_salary,
+				'is_additional_component': struct_row.is_additional_component
 			})
 		else:
 			component_row.amount = amount
@@ -526,18 +527,18 @@
 			# get all untaxed benefits till date, pass amount to be taxed by later methods
 			benefit_amount_to_tax = self.calculate_unclaimed_taxable_benefit(payroll_period)
 			# flexi's excluded from monthly tax, add flexis in this slip to total_taxable_benefit
-			benefit_amount_to_tax += self.get_taxable_earnings(only_flexi=True)
+			benefit_amount_to_tax += self.get_taxable_earnings(only_flexi=True)["taxable_earning"]
 		if self.deduct_tax_for_unsubmitted_tax_exemption_proof:
 			# calc tax to be paid for the period till date considering prorata taxes paid and proofs submitted
 			return self.calculate_unclaimed_taxable_earning(payroll_period, tax_component, benefit_amount_to_tax)
 
 		# calc prorata tax to be applied
-		return self.calculate_variable_tax(tax_component, payroll_period, benefit_amount_to_tax=benefit_amount_to_tax)
+		return self.calculate_variable_tax(tax_component, payroll_period, benefit_amount_to_tax)
 
 	def calculate_variable_tax(self, tax_component, payroll_period, benefit_amount_to_tax=0):
-		total_taxable_earning = self.get_taxable_earnings()
+		taxable_earnings = self.get_taxable_earnings()
 		period_factor = self.get_period_factor(payroll_period.start_date, payroll_period.end_date)
-		annual_earning = total_taxable_earning * period_factor
+		annual_earning = taxable_earnings["taxable_earning"] * period_factor
 
 		# Calculate total exemption declaration
 		exemption_amount = 0
@@ -547,8 +548,8 @@
 				{"employee": self.employee, "payroll_period": payroll_period.name, "docstatus": 1},
 				"total_exemption_amount")
 		annual_taxable_earning = annual_earning - exemption_amount
-
-		return self.calculate_tax(payroll_period, tax_component, annual_taxable_earning, period_factor, 0, benefit_amount_to_tax)
+		additional_income = benefit_amount_to_tax + taxable_earnings["additional_income"]
+		return self.calculate_tax(payroll_period, tax_component, annual_taxable_earning, period_factor, 0, additional_income)
 
 	def calculate_tax_for_payroll_period(self, tax_component, payroll_period):
 		# get total taxable income, total tax paid in payroll period
@@ -570,7 +571,8 @@
 		total_taxable_earning = taxable_income - total_tax_exemption_proof - total_benefit_claim
 
 		# add taxable earnings of current salary_slip, include flexi
-		total_taxable_earning += self.get_taxable_earnings(include_flexi=1)
+		taxable_earnings = self.get_taxable_earnings(include_flexi=1)
+		total_taxable_earning += taxable_earnings["taxable_earning"] + taxable_earnings["additional_income"]
 		return self.calculate_tax(payroll_period, tax_component, total_taxable_earning, 1, tax_paid, 0)
 
 	def calculate_unclaimed_taxable_benefit(self, payroll_period):
@@ -606,7 +608,6 @@
 		return total_benefit - total_benefit_claim
 
 	def calculate_unclaimed_taxable_earning(self, payroll_period, tax_component, benefit_amount_to_tax):
-		total_taxable_earning, total_tax_paid = 0, 0
 		start_date = payroll_period.start_date
 
 		# if tax deducted earlier set the start date
@@ -618,19 +619,54 @@
 		if last_deducted and last_deducted[0][0]:
 			start_date = getdate(last_deducted[0][0])
 
+		total_taxable_earning, total_additional_pay = self.get_taxable_earnings_after(start_date)
+		total_tax_paid = self.get_tax_paid_after(start_date, payroll_period, tax_component)
+
+		total_exemption_amount = 0
+		# add up total Proof Submission
+		sum_exemption = frappe.db.sql("""select sum(total_amount) from
+		`tabEmployee Tax Exemption Proof Submission` where docstatus=1 and employee='{0}' and
+		payroll_period='{1}' and processed_in_payroll=0""".format(self.employee, payroll_period.name))
+		if sum_exemption and sum_exemption[0][0]:
+			total_exemption_amount = sum_exemption[0][0]
+		total_taxable_earning -= total_exemption_amount
+
+		total_additional_pay += benefit_amount_to_tax
+		# recalc annual tax slab by start date and end date
+		period_factor = self.get_period_factor(payroll_period.start_date, payroll_period.end_date, start_date, self.end_date)
+		annual_taxable_earning = total_taxable_earning * period_factor
+		return self.calculate_tax(payroll_period, tax_component, annual_taxable_earning, period_factor, total_tax_paid, total_additional_pay)
+
+	def get_taxable_earnings_after(self, start_date):
+		total_taxable_earning, total_additional_pay = 0, 0
 		# calc total taxable amount in period
 		sum_taxable_earning = frappe.db.sql("""select sum(sd.amount) from `tabSalary Detail` sd join
 					`tabSalary Slip` ss on sd.parent=ss.name where sd.parentfield='earnings'
-					and sd.is_tax_applicable=1 and is_flexible_benefit=0 and ss.docstatus=1
-					and ss.employee='{0}' and ss.start_date between '{1}' and '{2}' and
-					ss.end_date between '{1}' and '{2}'""".format(self.employee,
+					and sd.is_tax_applicable=1 and is_additional_component=0 and is_flexible_benefit=0
+					and ss.docstatus=1 and ss.employee='{0}' and ss.start_date between '{1}' and '{2}'
+					and ss.end_date between '{1}' and '{2}'""".format(self.employee,
 					start_date, self.start_date))
 		if sum_taxable_earning and sum_taxable_earning[0][0]:
 			total_taxable_earning = sum_taxable_earning[0][0]
 
-		# add taxable earning in this salary slip
-		total_taxable_earning += self.get_taxable_earnings()
+		sum_additional_earning = frappe.db.sql("""select sum(sd.amount) from `tabSalary Detail` sd join
+					`tabSalary Slip` ss on sd.parent=ss.name where sd.parentfield='earnings'
+					and sd.is_tax_applicable=1 and is_additional_component=1 and is_flexible_benefit=0
+					and ss.docstatus=1 and ss.employee='{0}' and ss.start_date between '{1}' and '{2}'
+					and ss.end_date between '{1}' and '{2}'""".format(self.employee,
+					start_date, self.start_date))
+		if sum_additional_earning and sum_additional_earning[0][0]:
+			total_additional_pay = sum_additional_earning[0][0]
 
+		# add taxable earning, additional_income in this salary slip
+		taxable_earnings = self.get_taxable_earnings()
+		total_taxable_earning += taxable_earnings["taxable_earning"]
+		total_additional_pay += taxable_earnings["additional_income"]
+
+		return total_taxable_earning, total_additional_pay
+
+	def get_tax_paid_after(self, start_date, payroll_period, tax_component):
+		total_tax_paid = 0
 		# find total_tax_paid from salary slip where benefit is not taxed
 		sum_tax_paid = frappe.db.sql("""select sum(sd.amount) from `tabSalary Detail` sd join
 					`tabSalary Slip` ss on sd.parent=ss.name where sd.parentfield='deductions'
@@ -654,49 +690,42 @@
 				struct_row, pro_rata_tax = ss_obj.calculate_variable_tax(tax_component, payroll_period)
 				if pro_rata_tax:
 					total_tax_paid += pro_rata_tax
-		total_exemption_amount = 0
-
-		# add up total Proof Submission
-		sum_exemption = frappe.db.sql("""select sum(total_amount) from
-		`tabEmployee Tax Exemption Proof Submission` where docstatus=1 and employee='{0}' and
-		payroll_period='{1}' and processed_in_payroll=0""".format(self.employee, payroll_period.name))
-		if sum_exemption and sum_exemption[0][0]:
-			total_exemption_amount = sum_exemption[0][0]
-		total_taxable_earning -= total_exemption_amount
-
-		# recalc annual tax slab by start date and end date
-		period_factor = self.get_period_factor(payroll_period.start_date, payroll_period.end_date, start_date, self.end_date)
-		annual_taxable_earning = total_taxable_earning * period_factor
-		return self.calculate_tax(payroll_period, tax_component, annual_taxable_earning, period_factor, total_tax_paid, benefit_amount_to_tax)
+		return total_tax_paid
 
 	def get_taxable_earnings(self, include_flexi=0, only_flexi=0):
 		taxable_earning = 0
+		additional_income = 0
 		for earning in self.earnings:
-			if only_flexi:
-				if earning.is_tax_applicable and earning.is_flexible_benefit:
-					taxable_earning += earning.amount
-				continue
-			if include_flexi:
-				if earning.is_tax_applicable or (earning.is_tax_applicable and earning.is_flexible_benefit):
-					taxable_earning += earning.amount
-			else:
-				if earning.is_tax_applicable and not earning.is_flexible_benefit:
-					taxable_earning += earning.amount
-		return taxable_earning
+			if earning.is_tax_applicable:
+				if earning.is_additional_component:
+					additional_income += earning.amount
+					continue
+				if only_flexi:
+					if earning.is_tax_applicable and earning.is_flexible_benefit:
+						taxable_earning += earning.amount
+					continue
+				if include_flexi:
+					if earning.is_tax_applicable or (earning.is_tax_applicable and earning.is_flexible_benefit):
+						taxable_earning += earning.amount
+				else:
+					if earning.is_tax_applicable and not earning.is_flexible_benefit:
+						taxable_earning += earning.amount
+		return {"taxable_earning": taxable_earning, "additional_income": additional_income}
 
-	def calculate_tax(self, payroll_period, tax_component, annual_taxable_earning, period_factor, tax_paid=0, benefit_amount_to_tax=0):
+	def calculate_tax(self, payroll_period, tax_component, annual_taxable_earning, period_factor, tax_paid=0, additional_income=0):
 		# Get tax calc by period
 		annual_tax = self.calculate_tax_by_tax_slab(payroll_period.name, annual_taxable_earning)
 
 		# Calc prorata tax
 		tax_amount = annual_tax / period_factor
+
+		# find the annual tax diff caused by additional_income, add to tax_amount
+		if additional_income > 0:
+			annual_tax_with_additional_income = self.calculate_tax_by_tax_slab(payroll_period.name, annual_taxable_earning + additional_income)
+			tax_amount += annual_tax_with_additional_income - annual_tax
+		#less paid taxes
 		if tax_paid:
 			tax_amount -= tax_paid
-
-		# find the annual tax diff caused by benefit_amount_to_tax, add to tax_amount
-		if benefit_amount_to_tax > 0:
-			annual_tax_with_benefit_amt = self.calculate_tax_by_tax_slab(payroll_period.name, annual_taxable_earning + benefit_amount_to_tax)
-			tax_amount += annual_tax_with_benefit_amt - annual_tax
 		struct_row = self.get_salary_slip_row(tax_component)
 		return struct_row, tax_amount
 
diff --git a/erpnext/manufacturing/report/bom_stock_calculated/bom_stock_calculated.py b/erpnext/manufacturing/report/bom_stock_calculated/bom_stock_calculated.py
index aac459e..79da23b 100644
--- a/erpnext/manufacturing/report/bom_stock_calculated/bom_stock_calculated.py
+++ b/erpnext/manufacturing/report/bom_stock_calculated/bom_stock_calculated.py
@@ -14,7 +14,7 @@
 	qty_to_make = filters.get("qty_to_make")
 
 	for rows in data:
-	        item_map = get_item_details(rows[0])
+		item_map = get_item_details(rows[0])
 		reqd_qty = qty_to_make * rows[3]
 		last_pur_price = frappe.db.get_value("Item", rows[0], "last_purchase_rate")
 		if rows[4] > 0:
@@ -79,7 +79,7 @@
 				LEFT JOIN `tabBin` AS ledger
 				ON bom_item.item_code = ledger.item_code
 				{conditions}
-				
+
 			WHERE
 				bom_item.parent = '{bom}' and bom_item.parenttype='BOM'
 
@@ -89,5 +89,3 @@
         items = frappe.db.sql("""select it.item_group, it.item_name, it.stock_uom, it.name, it.brand, it.description, it.manufacturer_part_no, it.manufacturer from tabItem it where it.item_code = %s""", item_code, as_dict=1)
 
         return dict((d.name, d) for d in items)
-
-