Merge branch 'develop' into asset-capitalization
This commit is contained in:
		
						commit
						bb00d38dd7
					
				
							
								
								
									
										2
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
								
							| @ -23,7 +23,7 @@ If your issue is not clear or does not meet the guidelines, then it will be clos | ||||
| 1. **Steps to Reproduce:** The bug report must have a list of steps needed to reproduce a bug. If we cannot reproduce it, then we cannot solve it. | ||||
| 1. **Version Number:** Please add the version number in your report. Often a bug is fixed in the latest version | ||||
| 1. **Clear Title:** Add a clear subject to your bug report like "Unable to submit Purchase Order without Basic Rate" instead of just "Cannot Submit" | ||||
| 1. **Screenshots:** Screenshots are a great way of communicating the issues. Try adding annotations or using LiceCAP to take a screencast in `gif`. | ||||
| 1. **Screenshots:** Screenshots are a great way of communicating issues. Try adding annotations or using LiceCAP to take a screencast in `gif`. | ||||
| 
 | ||||
| ### Feature Request Guidelines | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/patch.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/patch.yml
									
									
									
									
										vendored
									
									
								
							| @ -93,7 +93,7 @@ jobs: | ||||
|           for version in $(seq 12 13) | ||||
|           do | ||||
|               echo "Updating to v$version" | ||||
|               branch_name="version-$version" | ||||
|               branch_name="version-$version-hotfix" | ||||
| 
 | ||||
|               git -C "apps/frappe" fetch --depth 1 upstream $branch_name:$branch_name | ||||
|               git -C "apps/erpnext" fetch --depth 1 upstream $branch_name:$branch_name | ||||
|  | ||||
| @ -8,6 +8,7 @@ | ||||
| [](https://github.com/frappe/erpnext/actions/workflows/server-tests.yml) | ||||
| [](https://www.codetriage.com/frappe/erpnext) | ||||
| [](https://codecov.io/gh/frappe/erpnext) | ||||
| [](https://hub.docker.com/r/frappe/erpnext-worker) | ||||
| 
 | ||||
| [https://erpnext.com](https://erpnext.com) | ||||
| 
 | ||||
|  | ||||
| @ -4,7 +4,7 @@ import frappe | ||||
| 
 | ||||
| from erpnext.hooks import regional_overrides | ||||
| 
 | ||||
| __version__ = '13.9.0' | ||||
| __version__ = '14.0.0-dev' | ||||
| 
 | ||||
| def get_default_company(user=None): | ||||
| 	'''Get default company for user''' | ||||
|  | ||||
| @ -78,6 +78,7 @@ frappe.treeview_settings["Account"] = { | ||||
| 				const format = (value, currency) => format_currency(Math.abs(value), currency); | ||||
| 
 | ||||
| 				if (account.balance!==undefined) { | ||||
| 					node.parent && node.parent.find('.balance-area').remove(); | ||||
| 					$('<span class="balance-area pull-right">' | ||||
| 						+ (account.balance_in_account_currency ? | ||||
| 							(format(account.balance_in_account_currency, account.account_currency) + " / ") : "") | ||||
| @ -175,7 +176,7 @@ frappe.treeview_settings["Account"] = { | ||||
| 					&& node.expandable && !node.hide_add; | ||||
| 			}, | ||||
| 			click: function() { | ||||
| 				var me = frappe.treeview_settings['Account'].treeview; | ||||
| 				var me = frappe.views.trees['Account']; | ||||
| 				me.new_node(); | ||||
| 			}, | ||||
| 			btnClass: "hidden-xs" | ||||
|  | ||||
| @ -8,7 +8,7 @@ frappe.ui.form.on('Accounting Dimension Filter', { | ||||
| 		} | ||||
| 
 | ||||
| 		let help_content = | ||||
| 			`<table class="table table-bordered" style="background-color: #f9f9f9;">
 | ||||
| 			`<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
 | ||||
| 				<tr><td> | ||||
| 					<p> | ||||
| 						<i class="fa fa-hand-right"></i> | ||||
|  | ||||
| @ -19,6 +19,9 @@ class AccountsSettings(Document): | ||||
| 		frappe.db.set_default("add_taxes_from_item_tax_template", | ||||
| 			self.get("add_taxes_from_item_tax_template", 0)) | ||||
| 
 | ||||
| 		frappe.db.set_default("enable_common_party_accounting", | ||||
| 			self.get("enable_common_party_accounting", 0)) | ||||
| 
 | ||||
| 		self.validate_stale_days() | ||||
| 		self.enable_payment_schedule_in_print() | ||||
| 		self.toggle_discount_accounting_fields() | ||||
|  | ||||
							
								
								
									
										0
									
								
								erpnext/accounts/doctype/advance_tax/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								erpnext/accounts/doctype/advance_tax/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										56
									
								
								erpnext/accounts/doctype/advance_tax/advance_tax.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								erpnext/accounts/doctype/advance_tax/advance_tax.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,56 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "allow_rename": 1, | ||||
|  "creation": "2021-11-25 10:24:39.836195", | ||||
|  "doctype": "DocType", | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "reference_type", | ||||
|   "reference_name", | ||||
|   "reference_detail", | ||||
|   "account_head", | ||||
|   "allocated_amount" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "fieldname": "reference_type", | ||||
|    "fieldtype": "Link", | ||||
|    "label": "Reference Type", | ||||
|    "options": "DocType" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "reference_name", | ||||
|    "fieldtype": "Dynamic Link", | ||||
|    "label": "Reference Name", | ||||
|    "options": "reference_type" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "reference_detail", | ||||
|    "fieldtype": "Data", | ||||
|    "label": "Reference Detail" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "account_head", | ||||
|    "fieldtype": "Link", | ||||
|    "label": "Account Head", | ||||
|    "options": "Account" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "allocated_amount", | ||||
|    "fieldtype": "Currency", | ||||
|    "label": "Allocated Amount", | ||||
|    "options": "party_account_currency" | ||||
|   } | ||||
|  ], | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-11-25 10:27:51.712286", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Advance Tax", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [], | ||||
|  "sort_field": "modified", | ||||
|  "sort_order": "DESC" | ||||
| } | ||||
							
								
								
									
										9
									
								
								erpnext/accounts/doctype/advance_tax/advance_tax.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								erpnext/accounts/doctype/advance_tax/advance_tax.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | ||||
| # Copyright (c) 2021, Frappe Technologies Pvt. Ltd. and contributors | ||||
| # For license information, please see license.txt | ||||
| 
 | ||||
| # import frappe | ||||
| from frappe.model.document import Document | ||||
| 
 | ||||
| 
 | ||||
| class AdvanceTax(Document): | ||||
| 	pass | ||||
| @ -25,8 +25,7 @@ | ||||
|   "allocated_amount", | ||||
|   "column_break_13", | ||||
|   "base_tax_amount", | ||||
|   "base_total", | ||||
|   "base_allocated_amount" | ||||
|   "base_total" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
| @ -168,12 +167,6 @@ | ||||
|    "label": "Allocated Amount", | ||||
|    "options": "currency" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "base_allocated_amount", | ||||
|    "fieldtype": "Currency", | ||||
|    "label": "Allocated Amount (Company Currency)", | ||||
|    "options": "Company:company:default_currency" | ||||
|   }, | ||||
|   { | ||||
|    "fetch_from": "account_head.account_currency", | ||||
|    "fieldname": "currency", | ||||
| @ -186,7 +179,7 @@ | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-06-09 11:46:58.373170", | ||||
|  "modified": "2021-11-25 11:10:10.945027", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Advance Taxes and Charges", | ||||
|  | ||||
| @ -342,7 +342,15 @@ def get_pe_matching_query(amount_condition, account_from_to, transaction): | ||||
| 
 | ||||
| def get_je_matching_query(amount_condition, transaction): | ||||
| 	# get matching journal entry query | ||||
| 
 | ||||
| 	company_account = frappe.get_value("Bank Account", transaction.bank_account, "account") | ||||
| 	root_type = frappe.get_value("Account", company_account, "root_type") | ||||
| 
 | ||||
| 	if root_type == "Liability": | ||||
| 		cr_or_dr = "debit" if transaction.withdrawal > 0 else "credit" | ||||
| 	else: | ||||
| 		cr_or_dr = "credit" if transaction.withdrawal > 0 else "debit" | ||||
| 
 | ||||
| 	return f""" | ||||
| 
 | ||||
| 		SELECT | ||||
| @ -426,7 +434,7 @@ def get_pi_matching_query(amount_condition): | ||||
| 
 | ||||
| def get_ec_matching_query(bank_account, company, amount_condition): | ||||
| 	# get matching Expense Claim query | ||||
| 	mode_of_payments = [x["parent"] for x in frappe.db.get_list("Mode of Payment Account", | ||||
| 	mode_of_payments = [x["parent"] for x in frappe.db.get_all("Mode of Payment Account", | ||||
| 			filters={"default_account": bank_account}, fields=["parent"])] | ||||
| 	mode_of_payments = '(\'' + '\', \''.join(mode_of_payments) + '\' )' | ||||
| 	company_currency = get_company_currency(company) | ||||
|  | ||||
| @ -6,7 +6,7 @@ frappe.provide("erpnext.accounts.dimensions"); | ||||
| frappe.ui.form.on('Loyalty Program', { | ||||
| 	setup: function(frm) { | ||||
| 		var help_content = | ||||
| 			`<table class="table table-bordered" style="background-color: #f9f9f9;">
 | ||||
| 			`<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
 | ||||
| 				<tr><td> | ||||
| 					<h4> | ||||
| 						<i class="fa fa-hand-right"></i> | ||||
|  | ||||
| @ -25,3 +25,17 @@ class PartyLink(Document): | ||||
| 		if existing_party_link: | ||||
| 			frappe.throw(_('{} {} is already linked with another {}') | ||||
| 				.format(self.primary_role, self.primary_party, existing_party_link[0])) | ||||
| 
 | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| def create_party_link(primary_role, primary_party, secondary_party): | ||||
| 	party_link = frappe.new_doc('Party Link') | ||||
| 	party_link.primary_role = primary_role | ||||
| 	party_link.primary_party = primary_party | ||||
| 	party_link.secondary_role = 'Customer' if primary_role == 'Supplier' else 'Supplier' | ||||
| 	party_link.secondary_party = secondary_party | ||||
| 
 | ||||
| 	party_link.save(ignore_permissions=True) | ||||
| 
 | ||||
| 	return party_link | ||||
| 
 | ||||
|  | ||||
| @ -61,7 +61,6 @@ | ||||
|   "taxes_and_charges_section", | ||||
|   "purchase_taxes_and_charges_template", | ||||
|   "sales_taxes_and_charges_template", | ||||
|   "advance_tax_account", | ||||
|   "column_break_55", | ||||
|   "apply_tax_withholding_amount", | ||||
|   "tax_withholding_category", | ||||
| @ -685,15 +684,6 @@ | ||||
|    "fieldtype": "Section Break", | ||||
|    "hide_border": 1 | ||||
|   }, | ||||
|   { | ||||
|    "depends_on": "eval:doc.apply_tax_withholding_amount", | ||||
|    "description": "Provisional tax account for advance tax. Taxes are parked in this account until payments are allocated to invoices", | ||||
|    "fieldname": "advance_tax_account", | ||||
|    "fieldtype": "Link", | ||||
|    "label": "Advance Tax Account", | ||||
|    "mandatory_depends_on": "eval:doc.apply_tax_withholding_amount", | ||||
|    "options": "Account" | ||||
|   }, | ||||
|   { | ||||
|    "depends_on": "eval:doc.received_amount && doc.payment_type != 'Internal Transfer'", | ||||
|    "fieldname": "received_amount_after_tax", | ||||
| @ -730,7 +720,7 @@ | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "is_submittable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-10-22 17:50:24.632806", | ||||
|  "modified": "2021-11-24 18:58:24.919764", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Payment Entry", | ||||
|  | ||||
| @ -20,7 +20,7 @@ from erpnext.accounts.doctype.journal_entry.journal_entry import get_default_ban | ||||
| from erpnext.accounts.doctype.tax_withholding_category.tax_withholding_category import ( | ||||
| 	get_party_tax_withholding_details, | ||||
| ) | ||||
| from erpnext.accounts.general_ledger import make_gl_entries | ||||
| from erpnext.accounts.general_ledger import make_gl_entries, process_gl_map | ||||
| from erpnext.accounts.party import get_party_account | ||||
| from erpnext.accounts.utils import get_account_currency, get_balance_on, get_outstanding_invoices | ||||
| from erpnext.controllers.accounts_controller import ( | ||||
| @ -339,7 +339,7 @@ class PaymentEntry(AccountsController): | ||||
| 		for k, v in no_oustanding_refs.items(): | ||||
| 			frappe.msgprint( | ||||
| 				_("{} - {} now have {} as they had no outstanding amount left before submitting the Payment Entry.") | ||||
| 					.format(k, frappe.bold(", ".join(d.reference_name for d in v)), frappe.bold("negative outstanding amount")) | ||||
| 					.format(_(k), frappe.bold(", ".join(d.reference_name for d in v)), frappe.bold(_("negative outstanding amount"))) | ||||
| 				+ "<br><br>" + _("If this is undesirable please cancel the corresponding Payment Entry."), | ||||
| 				title=_("Warning"), indicator="orange") | ||||
| 
 | ||||
| @ -433,23 +433,12 @@ class PaymentEntry(AccountsController): | ||||
| 		if not self.apply_tax_withholding_amount: | ||||
| 			return | ||||
| 
 | ||||
| 		if not self.advance_tax_account: | ||||
| 			frappe.throw(_("Advance TDS account is mandatory for advance TDS deduction")) | ||||
| 
 | ||||
| 		net_total = self.paid_amount | ||||
| 
 | ||||
| 		for reference in self.get("references"): | ||||
| 			net_total_for_tds = 0 | ||||
| 			if reference.reference_doctype == 'Purchase Order': | ||||
| 				net_total_for_tds += flt(frappe.db.get_value('Purchase Order', reference.reference_name, 'net_total')) | ||||
| 
 | ||||
| 			if net_total_for_tds: | ||||
| 				net_total = net_total_for_tds | ||||
| 
 | ||||
| 		# Adding args as purchase invoice to get TDS amount | ||||
| 		args = frappe._dict({ | ||||
| 			'company': self.company, | ||||
| 			'doctype': 'Purchase Invoice', | ||||
| 			'doctype': 'Payment Entry', | ||||
| 			'supplier': self.party, | ||||
| 			'posting_date': self.posting_date, | ||||
| 			'net_total': net_total | ||||
| @ -461,7 +450,6 @@ class PaymentEntry(AccountsController): | ||||
| 			return | ||||
| 
 | ||||
| 		tax_withholding_details.update({ | ||||
| 			'add_deduct_tax': 'Add', | ||||
| 			'cost_center': self.cost_center or erpnext.get_default_cost_center(self.company) | ||||
| 		}) | ||||
| 
 | ||||
| @ -623,7 +611,7 @@ class PaymentEntry(AccountsController): | ||||
| 
 | ||||
| 			if not total_negative_outstanding: | ||||
| 				frappe.throw(_("Cannot {0} {1} {2} without any negative outstanding invoice") | ||||
| 					.format(self.payment_type, ("to" if self.party_type=="Customer" else "from"), | ||||
| 					.format(_(self.payment_type), (_("to") if self.party_type=="Customer" else _("from")), | ||||
| 						self.party_type), InvalidPaymentEntry) | ||||
| 
 | ||||
| 			elif paid_amount - additional_charges > total_negative_outstanding: | ||||
| @ -689,6 +677,7 @@ class PaymentEntry(AccountsController): | ||||
| 		self.add_deductions_gl_entries(gl_entries) | ||||
| 		self.add_tax_gl_entries(gl_entries) | ||||
| 
 | ||||
| 		gl_entries = process_gl_map(gl_entries) | ||||
| 		make_gl_entries(gl_entries, cancel=cancel, adv_adj=adv_adj) | ||||
| 
 | ||||
| 	def add_party_gl_entries(self, gl_entries): | ||||
| @ -752,7 +741,8 @@ class PaymentEntry(AccountsController): | ||||
| 					"against": self.party if self.payment_type=="Pay" else self.paid_to, | ||||
| 					"credit_in_account_currency": self.paid_amount, | ||||
| 					"credit": self.base_paid_amount, | ||||
| 					"cost_center": self.cost_center | ||||
| 					"cost_center": self.cost_center, | ||||
| 					"post_net_value": True | ||||
| 				}, item=self) | ||||
| 			) | ||||
| 		if self.payment_type in ("Receive", "Internal Transfer"): | ||||
| @ -782,14 +772,10 @@ class PaymentEntry(AccountsController): | ||||
| 				rev_dr_or_cr = "credit" if dr_or_cr == "debit" else "debit" | ||||
| 				against = self.party or self.paid_to | ||||
| 
 | ||||
| 			payment_or_advance_account = self.get_party_account_for_taxes() | ||||
| 			payment_account = self.get_party_account_for_taxes() | ||||
| 			tax_amount = d.tax_amount | ||||
| 			base_tax_amount = d.base_tax_amount | ||||
| 
 | ||||
| 			if self.advance_tax_account: | ||||
| 				tax_amount = -1 * tax_amount | ||||
| 				base_tax_amount = -1 * base_tax_amount | ||||
| 
 | ||||
| 			gl_entries.append( | ||||
| 				self.get_gl_dict({ | ||||
| 					"account": d.account_head, | ||||
| @ -798,19 +784,21 @@ class PaymentEntry(AccountsController): | ||||
| 					dr_or_cr + "_in_account_currency": base_tax_amount | ||||
| 					if account_currency==self.company_currency | ||||
| 					else d.tax_amount, | ||||
| 					"cost_center": d.cost_center | ||||
| 					"cost_center": d.cost_center, | ||||
| 					"post_net_value": True, | ||||
| 				}, account_currency, item=d)) | ||||
| 
 | ||||
| 			if not d.included_in_paid_amount or self.advance_tax_account: | ||||
| 			if not d.included_in_paid_amount: | ||||
| 				gl_entries.append( | ||||
| 					self.get_gl_dict({ | ||||
| 						"account": payment_or_advance_account, | ||||
| 						"account": payment_account, | ||||
| 						"against": against, | ||||
| 						rev_dr_or_cr: tax_amount, | ||||
| 						rev_dr_or_cr + "_in_account_currency": base_tax_amount | ||||
| 						if account_currency==self.company_currency | ||||
| 						else d.tax_amount, | ||||
| 						"cost_center": self.cost_center, | ||||
| 						"post_net_value": True, | ||||
| 					}, account_currency, item=d)) | ||||
| 
 | ||||
| 	def add_deductions_gl_entries(self, gl_entries): | ||||
| @ -832,9 +820,7 @@ class PaymentEntry(AccountsController): | ||||
| 				) | ||||
| 
 | ||||
| 	def get_party_account_for_taxes(self): | ||||
| 		if self.advance_tax_account: | ||||
| 			return self.advance_tax_account | ||||
| 		elif self.payment_type == 'Receive': | ||||
| 		if self.payment_type == 'Receive': | ||||
| 			return self.paid_to | ||||
| 		elif self.payment_type in ('Pay', 'Internal Transfer'): | ||||
| 			return self.paid_from | ||||
| @ -1106,7 +1092,7 @@ def get_outstanding_reference_documents(args): | ||||
| 
 | ||||
| 	if not data: | ||||
| 		frappe.msgprint(_("No outstanding invoices found for the {0} {1} which qualify the filters you have specified.") | ||||
| 			.format(args.get("party_type").lower(), frappe.bold(args.get("party")))) | ||||
| 			.format(_(args.get("party_type")).lower(), frappe.bold(args.get("party")))) | ||||
| 
 | ||||
| 	return data | ||||
| 
 | ||||
| @ -1599,13 +1585,6 @@ def get_payment_entry(dt, dn, party_amount=None, bank_account=None, bank_amount= | ||||
| 			}) | ||||
| 			pe.set_difference_amount() | ||||
| 
 | ||||
| 	if doc.doctype == 'Purchase Order' and doc.apply_tds: | ||||
| 		pe.apply_tax_withholding_amount = 1 | ||||
| 		pe.tax_withholding_category = doc.tax_withholding_category | ||||
| 
 | ||||
| 		if not pe.advance_tax_account: | ||||
| 			pe.advance_tax_account = frappe.db.get_value('Company', pe.company, 'unrealized_profit_loss_account') | ||||
| 
 | ||||
| 	return pe | ||||
| 
 | ||||
| def get_bank_cash_account(doc, bank_account): | ||||
|  | ||||
| @ -548,10 +548,14 @@ def make_payment_order(source_name, target_doc=None): | ||||
| 
 | ||||
| 	return doclist | ||||
| 
 | ||||
| def validate_payment(doc, method=""): | ||||
| 	if not frappe.db.has_column(doc.reference_doctype, 'status'): | ||||
| def validate_payment(doc, method=None): | ||||
| 	if doc.reference_doctype != "Payment Request" or ( | ||||
| 		frappe.db.get_value(doc.reference_doctype, doc.reference_docname, 'status') | ||||
| 		!= "Paid" | ||||
| 	): | ||||
| 		return | ||||
| 
 | ||||
| 	status = frappe.db.get_value(doc.reference_doctype, doc.reference_docname, 'status') | ||||
| 	if status == 'Paid': | ||||
| 		frappe.throw(_("The Payment Request {0} is already paid, cannot process payment twice").format(doc.reference_docname)) | ||||
| 	frappe.throw( | ||||
| 		_("The Payment Request {0} is already paid, cannot process payment twice") | ||||
| 		.format(doc.reference_docname) | ||||
| 	) | ||||
|  | ||||
| @ -88,9 +88,10 @@ class PeriodClosingVoucher(AccountsController): | ||||
| 
 | ||||
| 		for acc in pl_accounts: | ||||
| 			if flt(acc.bal_in_company_currency): | ||||
| 				cost_center = acc.cost_center if self.cost_center_wise_pnl else company_cost_center | ||||
| 				gl_entry = self.get_gl_dict({ | ||||
| 					"account": self.closing_account_head, | ||||
| 					"cost_center": acc.cost_center or company_cost_center, | ||||
| 					"cost_center": cost_center, | ||||
| 					"finance_book": acc.finance_book, | ||||
| 					"account_currency": acc.account_currency, | ||||
| 					"debit_in_account_currency": abs(flt(acc.bal_in_account_currency)) if flt(acc.bal_in_account_currency) > 0 else 0, | ||||
|  | ||||
| @ -66,8 +66,8 @@ class TestPeriodClosingVoucher(unittest.TestCase): | ||||
| 		company = create_company() | ||||
| 		surplus_account = create_account() | ||||
| 
 | ||||
| 		cost_center1 = create_cost_center("Test Cost Center 1") | ||||
| 		cost_center2 = create_cost_center("Test Cost Center 2") | ||||
| 		cost_center1 = create_cost_center("Main") | ||||
| 		cost_center2 = create_cost_center("Western Branch") | ||||
| 
 | ||||
| 		create_sales_invoice( | ||||
| 			company=company, | ||||
| @ -86,7 +86,10 @@ class TestPeriodClosingVoucher(unittest.TestCase): | ||||
| 			debit_to="Debtors - TPC" | ||||
| 		) | ||||
| 
 | ||||
| 		pcv = self.make_period_closing_voucher() | ||||
| 		pcv = self.make_period_closing_voucher(submit=False) | ||||
| 		pcv.cost_center_wise_pnl = 1 | ||||
| 		pcv.save() | ||||
| 		pcv.submit() | ||||
| 		surplus_account = pcv.closing_account_head | ||||
| 
 | ||||
| 		expected_gle = ( | ||||
| @ -149,7 +152,7 @@ class TestPeriodClosingVoucher(unittest.TestCase): | ||||
| 
 | ||||
| 		self.assertEqual(pcv_gle, expected_gle) | ||||
| 
 | ||||
| 	def make_period_closing_voucher(self): | ||||
| 	def make_period_closing_voucher(self, submit=True): | ||||
| 		surplus_account = create_account() | ||||
| 		cost_center = create_cost_center("Test Cost Center 1") | ||||
| 		pcv = frappe.get_doc({ | ||||
| @ -163,6 +166,7 @@ class TestPeriodClosingVoucher(unittest.TestCase): | ||||
| 			"remarks": "test" | ||||
| 		}) | ||||
| 		pcv.insert() | ||||
| 		if submit: | ||||
| 			pcv.submit() | ||||
| 
 | ||||
| 		return pcv | ||||
|  | ||||
| @ -171,6 +171,7 @@ | ||||
|   "sales_team_section_break", | ||||
|   "sales_partner", | ||||
|   "column_break10", | ||||
|   "amount_eligible_for_commission", | ||||
|   "commission_rate", | ||||
|   "total_commission", | ||||
|   "section_break2", | ||||
| @ -1561,16 +1562,23 @@ | ||||
|    "label": "Coupon Code", | ||||
|    "options": "Coupon Code", | ||||
|    "print_hide": 1 | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "amount_eligible_for_commission", | ||||
|    "fieldtype": "Currency", | ||||
|    "label": "Amount Eligible for Commission", | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "icon": "fa fa-file-text", | ||||
|  "is_submittable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-08-27 20:12:57.306772", | ||||
|  "modified": "2021-10-05 12:11:53.871828", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "POS Invoice", | ||||
|  "name_case": "Title Case", | ||||
|  "naming_rule": "By \"Naming Series\" field", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|  | ||||
| @ -46,6 +46,7 @@ | ||||
|   "base_amount", | ||||
|   "pricing_rules", | ||||
|   "is_free_item", | ||||
|   "grant_commission", | ||||
|   "section_break_21", | ||||
|   "net_rate", | ||||
|   "net_amount", | ||||
| @ -800,14 +801,22 @@ | ||||
|    "no_copy": 1, | ||||
|    "print_hide": 1, | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "default": "0", | ||||
|    "fieldname": "grant_commission", | ||||
|    "fieldtype": "Check", | ||||
|    "label": "Grant Commission", | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-01-04 17:34:49.924531", | ||||
|  "modified": "2021-10-05 12:23:47.506290", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "POS Invoice Item", | ||||
|  "naming_rule": "Random", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [], | ||||
|  "sort_field": "modified", | ||||
|  | ||||
| @ -110,9 +110,15 @@ class POSInvoiceMergeLog(Document): | ||||
| 
 | ||||
| 	def merge_pos_invoice_into(self, invoice, data): | ||||
| 		items, payments, taxes = [], [], [] | ||||
| 
 | ||||
| 		loyalty_amount_sum, loyalty_points_sum = 0, 0 | ||||
| 
 | ||||
| 		rounding_adjustment, base_rounding_adjustment = 0, 0 | ||||
| 		rounded_total, base_rounded_total = 0, 0 | ||||
| 
 | ||||
| 		loyalty_amount_sum, loyalty_points_sum, idx = 0, 0, 1 | ||||
| 
 | ||||
| 
 | ||||
| 		for doc in data: | ||||
| 			map_doc(doc, invoice, table_map={ "doctype": invoice.doctype }) | ||||
| 
 | ||||
| @ -146,6 +152,8 @@ class POSInvoiceMergeLog(Document): | ||||
| 						found = True | ||||
| 				if not found: | ||||
| 					tax.charge_type = 'Actual' | ||||
| 					tax.idx = idx | ||||
| 					idx += 1 | ||||
| 					tax.included_in_print_rate = 0 | ||||
| 					tax.tax_amount = tax.tax_amount_after_discount_amount | ||||
| 					tax.base_tax_amount = tax.base_tax_amount_after_discount_amount | ||||
| @ -163,8 +171,8 @@ class POSInvoiceMergeLog(Document): | ||||
| 					payments.append(payment) | ||||
| 			rounding_adjustment += doc.rounding_adjustment | ||||
| 			rounded_total += doc.rounded_total | ||||
| 			base_rounding_adjustment += doc.rounding_adjustment | ||||
| 			base_rounded_total += doc.rounded_total | ||||
| 			base_rounding_adjustment += doc.base_rounding_adjustment | ||||
| 			base_rounded_total += doc.base_rounded_total | ||||
| 
 | ||||
| 
 | ||||
| 		if loyalty_points_sum: | ||||
| @ -176,9 +184,9 @@ class POSInvoiceMergeLog(Document): | ||||
| 		invoice.set('payments', payments) | ||||
| 		invoice.set('taxes', taxes) | ||||
| 		invoice.set('rounding_adjustment',rounding_adjustment) | ||||
| 		invoice.set('rounding_adjustment',base_rounding_adjustment) | ||||
| 		invoice.set('base_rounded_total',base_rounded_total) | ||||
| 		invoice.set('base_rounding_adjustment',base_rounding_adjustment) | ||||
| 		invoice.set('rounded_total',rounded_total) | ||||
| 		invoice.set('base_rounded_total',base_rounded_total) | ||||
| 		invoice.additional_discount_percentage = 0 | ||||
| 		invoice.discount_amount = 0.0 | ||||
| 		invoice.taxes_and_charges = None | ||||
|  | ||||
| @ -38,7 +38,7 @@ frappe.ui.form.on('Pricing Rule', { | ||||
| 
 | ||||
| 	refresh: function(frm) { | ||||
| 		var help_content = | ||||
| 			`<table class="table table-bordered" style="background-color: #f9f9f9;">
 | ||||
| 			`<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
 | ||||
| 				<tr><td> | ||||
| 					<h4> | ||||
| 						<i class="fa fa-hand-right"></i> | ||||
|  | ||||
| @ -543,6 +543,75 @@ class TestPricingRule(unittest.TestCase): | ||||
| 		frappe.get_doc("Item Price", {"item_code": "Water Flask"}).delete() | ||||
| 		item.delete() | ||||
| 
 | ||||
| 	def test_pricing_rule_for_different_currency(self): | ||||
| 		make_item("Test Sanitizer Item") | ||||
| 
 | ||||
| 		pricing_rule_record = { | ||||
| 			"doctype": "Pricing Rule", | ||||
| 			"title": "_Test Sanitizer Rule", | ||||
| 			"apply_on": "Item Code", | ||||
| 			"items": [{ | ||||
| 				"item_code": "Test Sanitizer Item", | ||||
| 			}], | ||||
| 			"selling": 1, | ||||
| 			"currency": "INR", | ||||
| 			"rate_or_discount": "Rate", | ||||
| 			"rate": 0, | ||||
| 			"priority": 2, | ||||
| 			"margin_type": "Percentage", | ||||
| 			"margin_rate_or_amount": 0.0, | ||||
| 			"company": "_Test Company" | ||||
| 		} | ||||
| 
 | ||||
| 		rule = frappe.get_doc(pricing_rule_record) | ||||
| 		rule.rate_or_discount = 'Rate' | ||||
| 		rule.rate = 100.0 | ||||
| 		rule.insert() | ||||
| 
 | ||||
| 		rule1 = frappe.get_doc(pricing_rule_record) | ||||
| 		rule1.currency = 'USD' | ||||
| 		rule1.rate_or_discount = 'Rate' | ||||
| 		rule1.rate = 2.0 | ||||
| 		rule1.priority = 1 | ||||
| 		rule1.insert() | ||||
| 
 | ||||
| 		args = frappe._dict({ | ||||
| 			"item_code": "Test Sanitizer Item", | ||||
| 			"company": "_Test Company", | ||||
| 			"price_list": "_Test Price List", | ||||
| 			"currency": "USD", | ||||
| 			"doctype": "Sales Invoice", | ||||
| 			"conversion_rate": 1, | ||||
| 			"price_list_currency": "_Test Currency", | ||||
| 			"plc_conversion_rate": 1, | ||||
| 			"order_type": "Sales", | ||||
| 			"customer": "_Test Customer", | ||||
| 			"name": None, | ||||
| 			"transaction_date": frappe.utils.nowdate() | ||||
| 		}) | ||||
| 
 | ||||
| 		details = get_item_details(args) | ||||
| 		self.assertEqual(details.price_list_rate, 2.0) | ||||
| 
 | ||||
| 
 | ||||
| 		args = frappe._dict({ | ||||
| 			"item_code": "Test Sanitizer Item", | ||||
| 			"company": "_Test Company", | ||||
| 			"price_list": "_Test Price List", | ||||
| 			"currency": "INR", | ||||
| 			"doctype": "Sales Invoice", | ||||
| 			"conversion_rate": 1, | ||||
| 			"price_list_currency": "_Test Currency", | ||||
| 			"plc_conversion_rate": 1, | ||||
| 			"order_type": "Sales", | ||||
| 			"customer": "_Test Customer", | ||||
| 			"name": None, | ||||
| 			"transaction_date": frappe.utils.nowdate() | ||||
| 		}) | ||||
| 
 | ||||
| 		details = get_item_details(args) | ||||
| 		self.assertEqual(details.price_list_rate, 100.0) | ||||
| 
 | ||||
| 	def test_pricing_rule_for_transaction(self): | ||||
| 		make_item("Water Flask 1") | ||||
| 		frappe.delete_doc_if_exists('Pricing Rule', '_Test Pricing Rule') | ||||
|  | ||||
| @ -264,6 +264,11 @@ def filter_pricing_rules(args, pricing_rules, doc=None): | ||||
| 			else: | ||||
| 				p.variant_of = None | ||||
| 
 | ||||
| 	if len(pricing_rules) > 1: | ||||
| 		filtered_rules = list(filter(lambda x: x.currency==args.get('currency'), pricing_rules)) | ||||
| 		if filtered_rules: | ||||
| 			pricing_rules = filtered_rules | ||||
| 
 | ||||
| 	# find pricing rule with highest priority | ||||
| 	if pricing_rules: | ||||
| 		max_priority = max(cint(p.priority) for p in pricing_rules) | ||||
|  | ||||
| @ -20,6 +20,9 @@ price_discount_fields = ['rate_or_discount', 'apply_discount_on', 'apply_discoun | ||||
| product_discount_fields = ['free_item', 'free_qty', 'free_item_uom', | ||||
| 	'free_item_rate', 'same_item', 'is_recursive', 'apply_multiple_pricing_rules'] | ||||
| 
 | ||||
| class TransactionExists(frappe.ValidationError): | ||||
| 	pass | ||||
| 
 | ||||
| class PromotionalScheme(Document): | ||||
| 	def validate(self): | ||||
| 		if not self.selling and not self.buying: | ||||
| @ -28,6 +31,40 @@ class PromotionalScheme(Document): | ||||
| 			or self.product_discount_slabs): | ||||
| 			frappe.throw(_("Price or product discount slabs are required")) | ||||
| 
 | ||||
| 		self.validate_applicable_for() | ||||
| 		self.validate_pricing_rules() | ||||
| 
 | ||||
| 	def validate_applicable_for(self): | ||||
| 		if self.applicable_for: | ||||
| 			applicable_for = frappe.scrub(self.applicable_for) | ||||
| 
 | ||||
| 			if not self.get(applicable_for): | ||||
| 				msg = (f'The field {frappe.bold(self.applicable_for)} is required') | ||||
| 				frappe.throw(_(msg)) | ||||
| 
 | ||||
| 	def validate_pricing_rules(self): | ||||
| 		if self.is_new(): | ||||
| 			return | ||||
| 
 | ||||
| 		transaction_exists = False | ||||
| 		docnames = [] | ||||
| 
 | ||||
| 		# If user has changed applicable for | ||||
| 		if self._doc_before_save.applicable_for == self.applicable_for: | ||||
| 			return | ||||
| 
 | ||||
| 		docnames = frappe.get_all('Pricing Rule', | ||||
| 			filters= {'promotional_scheme': self.name}) | ||||
| 
 | ||||
| 		for docname in docnames: | ||||
| 			if frappe.db.exists('Pricing Rule Detail', | ||||
| 				{'pricing_rule': docname.name, 'docstatus': ('<', 2)}): | ||||
| 				raise_for_transaction_exists(self.name) | ||||
| 
 | ||||
| 		if docnames and not transaction_exists: | ||||
| 			for docname in docnames: | ||||
| 				frappe.delete_doc('Pricing Rule', docname.name) | ||||
| 
 | ||||
| 	def on_update(self): | ||||
| 		pricing_rules = frappe.get_all( | ||||
| 			'Pricing Rule', | ||||
| @ -67,6 +104,13 @@ class PromotionalScheme(Document): | ||||
| 			{'promotional_scheme': self.name}): | ||||
| 			frappe.delete_doc('Pricing Rule', rule.name) | ||||
| 
 | ||||
| def raise_for_transaction_exists(name): | ||||
| 	msg = (f"""You can't change the {frappe.bold(_('Applicable For'))} | ||||
| 		because transactions are present against the Promotional Scheme {frappe.bold(name)}. """) | ||||
| 	msg += 'Kindly disable this Promotional Scheme and create new for new Applicable For.' | ||||
| 
 | ||||
| 	frappe.throw(_(msg), TransactionExists) | ||||
| 
 | ||||
| def get_pricing_rules(doc, rules=None): | ||||
| 	if rules is None: | ||||
| 		rules = {} | ||||
| @ -84,45 +128,59 @@ def _get_pricing_rules(doc, child_doc, discount_fields, rules=None): | ||||
| 	new_doc = [] | ||||
| 	args = get_args_for_pricing_rule(doc) | ||||
| 	applicable_for = frappe.scrub(doc.get('applicable_for')) | ||||
| 
 | ||||
| 	for idx, d in enumerate(doc.get(child_doc)): | ||||
| 		if d.name in rules: | ||||
| 			for applicable_for_value in args.get(applicable_for): | ||||
| 				temp_args = args.copy() | ||||
| 				docname = frappe.get_all( | ||||
| 					'Pricing Rule', | ||||
| 					fields = ["promotional_scheme_id", "name", applicable_for], | ||||
| 					filters = { | ||||
| 						'promotional_scheme_id': d.name, | ||||
| 						applicable_for: applicable_for_value | ||||
| 					} | ||||
| 				) | ||||
| 
 | ||||
| 				if docname: | ||||
| 					pr = frappe.get_doc('Pricing Rule', docname[0].get('name')) | ||||
| 					temp_args[applicable_for] = applicable_for_value | ||||
| 					pr = set_args(temp_args, pr, doc, child_doc, discount_fields, d) | ||||
| 			if not args.get(applicable_for): | ||||
| 				docname = get_pricing_rule_docname(d) | ||||
| 				pr = prepare_pricing_rule(args, doc, child_doc, discount_fields, d, docname) | ||||
| 				new_doc.append(pr) | ||||
| 			else: | ||||
| 					pr = frappe.new_doc("Pricing Rule") | ||||
| 					pr.title = doc.name | ||||
| 					temp_args[applicable_for] = applicable_for_value | ||||
| 					pr = set_args(temp_args, pr, doc, child_doc, discount_fields, d) | ||||
| 
 | ||||
| 				for applicable_for_value in args.get(applicable_for): | ||||
| 					docname = get_pricing_rule_docname(d, applicable_for, applicable_for_value) | ||||
| 					pr = prepare_pricing_rule(args, doc, child_doc, discount_fields, | ||||
| 						d, docname, applicable_for, applicable_for_value) | ||||
| 					new_doc.append(pr) | ||||
| 
 | ||||
| 		else: | ||||
| 		elif args.get(applicable_for): | ||||
| 			applicable_for_values = args.get(applicable_for) or [] | ||||
| 			for applicable_for_value in applicable_for_values: | ||||
| 				pr = frappe.new_doc("Pricing Rule") | ||||
| 				pr.title = doc.name | ||||
| 				temp_args = args.copy() | ||||
| 				temp_args[applicable_for] = applicable_for_value | ||||
| 				pr = set_args(temp_args, pr, doc, child_doc, discount_fields, d) | ||||
| 				pr = prepare_pricing_rule(args, doc, child_doc, discount_fields, | ||||
| 					d, applicable_for=applicable_for, value= applicable_for_value) | ||||
| 
 | ||||
| 				new_doc.append(pr) | ||||
| 		else: | ||||
| 			pr = prepare_pricing_rule(args, doc, child_doc, discount_fields, d) | ||||
| 			new_doc.append(pr) | ||||
| 
 | ||||
| 	return new_doc | ||||
| 
 | ||||
| def get_pricing_rule_docname(row: dict, applicable_for: str = None, applicable_for_value: str = None) -> str: | ||||
| 	fields = ['promotional_scheme_id', 'name'] | ||||
| 	filters = { | ||||
| 		'promotional_scheme_id': row.name | ||||
| 	} | ||||
| 
 | ||||
| 	if applicable_for: | ||||
| 		fields.append(applicable_for) | ||||
| 		filters[applicable_for] = applicable_for_value | ||||
| 
 | ||||
| 	docname = frappe.get_all('Pricing Rule', fields = fields, filters = filters) | ||||
| 	return docname[0].name if docname else '' | ||||
| 
 | ||||
| def prepare_pricing_rule(args, doc, child_doc, discount_fields, d, docname=None, applicable_for=None, value=None): | ||||
| 	if docname: | ||||
| 		pr = frappe.get_doc("Pricing Rule", docname) | ||||
| 	else: | ||||
| 		pr = frappe.new_doc("Pricing Rule") | ||||
| 
 | ||||
| 	pr.title = doc.name | ||||
| 	temp_args = args.copy() | ||||
| 
 | ||||
| 	if value: | ||||
| 		temp_args[applicable_for] = value | ||||
| 
 | ||||
| 	return set_args(temp_args, pr, doc, child_doc, discount_fields, d) | ||||
| 
 | ||||
| def set_args(args, pr, doc, child_doc, discount_fields, child_doc_fields): | ||||
| 	pr.update(args) | ||||
| @ -145,6 +203,7 @@ def set_args(args, pr, doc, child_doc, discount_fields, child_doc_fields): | ||||
| 				apply_on: d.get(apply_on), | ||||
| 				'uom': d.uom | ||||
| 			}) | ||||
| 
 | ||||
| 	return pr | ||||
| 
 | ||||
| def get_args_for_pricing_rule(doc): | ||||
|  | ||||
| @ -5,10 +5,17 @@ import unittest | ||||
| 
 | ||||
| import frappe | ||||
| 
 | ||||
| from erpnext.accounts.doctype.promotional_scheme.promotional_scheme import TransactionExists | ||||
| from erpnext.selling.doctype.sales_order.test_sales_order import make_sales_order | ||||
| 
 | ||||
| 
 | ||||
| class TestPromotionalScheme(unittest.TestCase): | ||||
| 	def setUp(self): | ||||
| 		if frappe.db.exists('Promotional Scheme', '_Test Scheme'): | ||||
| 			frappe.delete_doc('Promotional Scheme', '_Test Scheme') | ||||
| 
 | ||||
| 	def test_promotional_scheme(self): | ||||
| 		ps = make_promotional_scheme() | ||||
| 		ps = make_promotional_scheme(applicable_for='Customer', customer='_Test Customer') | ||||
| 		price_rules = frappe.get_all('Pricing Rule', fields = ["promotional_scheme_id", "name", "creation"], | ||||
| 			filters = {'promotional_scheme': ps.name}) | ||||
| 		self.assertTrue(len(price_rules),1) | ||||
| @ -39,22 +46,62 @@ class TestPromotionalScheme(unittest.TestCase): | ||||
| 			filters = {'promotional_scheme': ps.name}) | ||||
| 		self.assertEqual(price_rules, []) | ||||
| 
 | ||||
| def make_promotional_scheme(): | ||||
| 	def test_promotional_scheme_without_applicable_for(self): | ||||
| 		ps = make_promotional_scheme() | ||||
| 		price_rules = frappe.get_all('Pricing Rule', filters = {'promotional_scheme': ps.name}) | ||||
| 
 | ||||
| 		self.assertTrue(len(price_rules), 1) | ||||
| 		frappe.delete_doc('Promotional Scheme', ps.name) | ||||
| 
 | ||||
| 		price_rules = frappe.get_all('Pricing Rule', filters = {'promotional_scheme': ps.name}) | ||||
| 		self.assertEqual(price_rules, []) | ||||
| 
 | ||||
| 	def test_change_applicable_for_in_promotional_scheme(self): | ||||
| 		ps = make_promotional_scheme() | ||||
| 		price_rules = frappe.get_all('Pricing Rule', filters = {'promotional_scheme': ps.name}) | ||||
| 		self.assertTrue(len(price_rules), 1) | ||||
| 
 | ||||
| 		so = make_sales_order(qty=5, currency='USD', do_not_save=True) | ||||
| 		so.set_missing_values() | ||||
| 		so.save() | ||||
| 		self.assertEqual(price_rules[0].name, so.pricing_rules[0].pricing_rule) | ||||
| 
 | ||||
| 		ps.applicable_for = 'Customer' | ||||
| 		ps.append('customer', { | ||||
| 			'customer': '_Test Customer' | ||||
| 		}) | ||||
| 
 | ||||
| 		self.assertRaises(TransactionExists, ps.save) | ||||
| 
 | ||||
| 		frappe.delete_doc('Sales Order', so.name) | ||||
| 		frappe.delete_doc('Promotional Scheme', ps.name) | ||||
| 		price_rules = frappe.get_all('Pricing Rule', filters = {'promotional_scheme': ps.name}) | ||||
| 		self.assertEqual(price_rules, []) | ||||
| 
 | ||||
| def make_promotional_scheme(**args): | ||||
| 	args = frappe._dict(args) | ||||
| 
 | ||||
| 	ps = frappe.new_doc('Promotional Scheme') | ||||
| 	ps.name = '_Test Scheme' | ||||
| 	ps.append('items',{ | ||||
| 		'item_code': '_Test Item' | ||||
| 	}) | ||||
| 
 | ||||
| 	ps.selling = 1 | ||||
| 	ps.append('price_discount_slabs',{ | ||||
| 		'min_qty': 4, | ||||
| 		'validate_applied_rule': 0, | ||||
| 		'discount_percentage': 20, | ||||
| 		'rule_description': 'Test' | ||||
| 	}) | ||||
| 	ps.applicable_for = 'Customer' | ||||
| 	ps.append('customer',{ | ||||
| 		'customer': "_Test Customer" | ||||
| 
 | ||||
| 	ps.company = '_Test Company' | ||||
| 	if args.applicable_for: | ||||
| 		ps.applicable_for = args.applicable_for | ||||
| 		ps.append(frappe.scrub(args.applicable_for), { | ||||
| 			frappe.scrub(args.applicable_for): args.get(frappe.scrub(args.applicable_for)) | ||||
| 		}) | ||||
| 
 | ||||
| 	ps.save() | ||||
| 
 | ||||
| 	return ps | ||||
|  | ||||
| @ -136,7 +136,7 @@ | ||||
|    "label": "Threshold for Suggestion" | ||||
|   }, | ||||
|   { | ||||
|    "default": "1", | ||||
|    "default": "0", | ||||
|    "fieldname": "validate_applied_rule", | ||||
|    "fieldtype": "Check", | ||||
|    "label": "Validate Applied Rule" | ||||
| @ -169,7 +169,7 @@ | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-08-19 15:49:29.598727", | ||||
|  "modified": "2021-11-16 00:25:33.843996", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Promotional Scheme Price Discount", | ||||
|  | ||||
| @ -592,8 +592,17 @@ frappe.ui.form.on("Purchase Invoice", { | ||||
| 		erpnext.accounts.dimensions.update_dimension(frm, frm.doctype); | ||||
| 
 | ||||
| 		if (frm.doc.company) { | ||||
| 			frappe.db.get_value('Company', frm.doc.company, 'default_payable_account', (r) => { | ||||
| 				frm.set_value('credit_to', r.default_payable_account); | ||||
| 			frappe.call({ | ||||
| 				method: | ||||
| 					"erpnext.accounts.party.get_party_account", | ||||
| 				args: { | ||||
| 					party_type: 'Supplier', | ||||
| 					party: frm.doc.supplier, | ||||
| 					company: frm.doc.company | ||||
| 				}, | ||||
| 				callback: (response) => { | ||||
| 					if (response) frm.set_value("credit_to", response.message); | ||||
| 				}, | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| @ -130,6 +130,7 @@ | ||||
|   "allocate_advances_automatically", | ||||
|   "get_advances", | ||||
|   "advances", | ||||
|   "advance_tax", | ||||
|   "payment_schedule_section", | ||||
|   "payment_terms_template", | ||||
|   "ignore_default_payment_terms_template", | ||||
| @ -1408,13 +1409,21 @@ | ||||
|   { | ||||
|    "fieldname": "column_break_147", | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "advance_tax", | ||||
|    "fieldtype": "Table", | ||||
|    "hidden": 1, | ||||
|    "label": "Advance Tax", | ||||
|    "options": "Advance Tax", | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "icon": "fa fa-file-text", | ||||
|  "idx": 204, | ||||
|  "is_submittable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-10-12 20:55:16.145651", | ||||
|  "modified": "2021-11-25 13:31:02.716727", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Purchase Invoice", | ||||
|  | ||||
| @ -427,6 +427,7 @@ class PurchaseInvoice(BuyingController): | ||||
| 
 | ||||
| 		self.update_project() | ||||
| 		update_linked_doc(self.doctype, self.name, self.inter_company_invoice_reference) | ||||
| 		self.update_advance_tax_references() | ||||
| 
 | ||||
| 		self.process_common_party_accounting() | ||||
| 
 | ||||
| @ -472,8 +473,6 @@ class PurchaseInvoice(BuyingController): | ||||
| 		self.make_exchange_gain_loss_gl_entries(gl_entries) | ||||
| 		self.make_internal_transfer_gl_entries(gl_entries) | ||||
| 
 | ||||
| 		self.allocate_advance_taxes(gl_entries) | ||||
| 
 | ||||
| 		gl_entries = make_regional_gl_entries(gl_entries, self) | ||||
| 
 | ||||
| 		gl_entries = merge_similar_entries(gl_entries) | ||||
| @ -729,7 +728,7 @@ class PurchaseInvoice(BuyingController): | ||||
| 									"account": self.stock_received_but_not_billed, | ||||
| 									"against": self.supplier, | ||||
| 									"debit": flt(item.item_tax_amount, item.precision("item_tax_amount")), | ||||
| 									"remarks": self.remarks or "Accounting Entry for Stock", | ||||
| 									"remarks": self.remarks or _("Accounting Entry for Stock"), | ||||
| 									"cost_center": self.cost_center, | ||||
| 									"project": item.project or self.project | ||||
| 								}, item=item) | ||||
| @ -937,7 +936,7 @@ class PurchaseInvoice(BuyingController): | ||||
| 							"cost_center": tax.cost_center, | ||||
| 							"against": self.supplier, | ||||
| 							"credit": valuation_tax[tax.name], | ||||
| 							"remarks": self.remarks or "Accounting Entry for Stock" | ||||
| 							"remarks": self.remarks or _("Accounting Entry for Stock") | ||||
| 						}, item=tax)) | ||||
| 
 | ||||
| 	@property | ||||
| @ -1074,6 +1073,7 @@ class PurchaseInvoice(BuyingController): | ||||
| 
 | ||||
| 		unlink_inter_company_doc(self.doctype, self.name, self.inter_company_invoice_reference) | ||||
| 		self.ignore_linked_doctypes = ('GL Entry', 'Stock Ledger Entry', 'Repost Item Valuation') | ||||
| 		self.update_advance_tax_references(cancel=1) | ||||
| 
 | ||||
| 	def update_project(self): | ||||
| 		project_list = [] | ||||
| @ -1150,7 +1150,10 @@ class PurchaseInvoice(BuyingController): | ||||
| 		if not self.tax_withholding_category: | ||||
| 			return | ||||
| 
 | ||||
| 		tax_withholding_details = get_party_tax_withholding_details(self, self.tax_withholding_category) | ||||
| 		tax_withholding_details, advance_taxes = get_party_tax_withholding_details(self, self.tax_withholding_category) | ||||
| 
 | ||||
| 		# Adjust TDS paid on advances | ||||
| 		self.allocate_advance_tds(tax_withholding_details, advance_taxes) | ||||
| 
 | ||||
| 		if not tax_withholding_details: | ||||
| 			return | ||||
| @ -1174,6 +1177,39 @@ class PurchaseInvoice(BuyingController): | ||||
| 		# calculate totals again after applying TDS | ||||
| 		self.calculate_taxes_and_totals() | ||||
| 
 | ||||
| 	def allocate_advance_tds(self, tax_withholding_details, advance_taxes): | ||||
| 		self.set('advance_tax', []) | ||||
| 		for tax in advance_taxes: | ||||
| 			allocated_amount = 0 | ||||
| 			pending_amount = flt(tax.tax_amount - tax.allocated_amount) | ||||
| 			if flt(tax_withholding_details.get('tax_amount')) >= pending_amount: | ||||
| 				tax_withholding_details['tax_amount'] -= pending_amount | ||||
| 				allocated_amount = pending_amount | ||||
| 			elif flt(tax_withholding_details.get('tax_amount')) and flt(tax_withholding_details.get('tax_amount')) < pending_amount: | ||||
| 				allocated_amount = tax_withholding_details['tax_amount'] | ||||
| 				tax_withholding_details['tax_amount'] = 0 | ||||
| 
 | ||||
| 			self.append('advance_tax', { | ||||
| 				'reference_type': 'Payment Entry', | ||||
| 				'reference_name': tax.parent, | ||||
| 				'reference_detail': tax.name, | ||||
| 				'account_head': tax.account_head, | ||||
| 				'allocated_amount': allocated_amount | ||||
| 			}) | ||||
| 
 | ||||
| 	def update_advance_tax_references(self, cancel=0): | ||||
| 		for tax in self.get('advance_tax'): | ||||
| 			at = frappe.qb.DocType("Advance Taxes and Charges").as_("at") | ||||
| 
 | ||||
| 			if cancel: | ||||
| 				frappe.qb.update(at).set( | ||||
| 					at.allocated_amount, at.allocated_amount - tax.allocated_amount | ||||
| 				).where(at.name == tax.reference_detail).run() | ||||
| 			else: | ||||
| 				frappe.qb.update(at).set( | ||||
| 					at.allocated_amount, at.allocated_amount + tax.allocated_amount | ||||
| 				).where(at.name == tax.reference_detail).run() | ||||
| 
 | ||||
| 	def set_status(self, update=False, status=None, update_modified=True): | ||||
| 		if self.is_new(): | ||||
| 			if self.get('amended_from'): | ||||
|  | ||||
| @ -13,6 +13,7 @@ from erpnext.accounts.doctype.account.test_account import create_account, get_in | ||||
| from erpnext.accounts.doctype.payment_entry.payment_entry import get_payment_entry | ||||
| from erpnext.buying.doctype.supplier.test_supplier import create_supplier | ||||
| from erpnext.controllers.accounts_controller import get_payment_terms | ||||
| from erpnext.controllers.buying_controller import QtyMismatchError | ||||
| from erpnext.exceptions import InvalidCurrency | ||||
| from erpnext.projects.doctype.project.test_project import make_project | ||||
| from erpnext.stock.doctype.item.test_item import create_item | ||||
| @ -35,6 +36,27 @@ class TestPurchaseInvoice(unittest.TestCase): | ||||
| 	def tearDownClass(self): | ||||
| 		unlink_payment_on_cancel_of_invoice(0) | ||||
| 
 | ||||
| 	def test_purchase_invoice_received_qty(self): | ||||
| 		""" | ||||
| 			1. Test if received qty is validated against accepted + rejected | ||||
| 			2. Test if received qty is auto set on save | ||||
| 		""" | ||||
| 		pi = make_purchase_invoice( | ||||
| 			qty=1, | ||||
| 			rejected_qty=1, | ||||
| 			received_qty=3, | ||||
| 			item_code="_Test Item Home Desktop 200", | ||||
| 			rejected_warehouse = "_Test Rejected Warehouse - _TC", | ||||
| 			update_stock=True, do_not_save=True) | ||||
| 		self.assertRaises(QtyMismatchError, pi.save) | ||||
| 
 | ||||
| 		pi.items[0].received_qty = 0 | ||||
| 		pi.save() | ||||
| 		self.assertEqual(pi.items[0].received_qty, 2) | ||||
| 
 | ||||
| 		# teardown | ||||
| 		pi.delete() | ||||
| 
 | ||||
| 	def test_gl_entries_without_perpetual_inventory(self): | ||||
| 		frappe.db.set_value("Company", "_Test Company", "round_off_account", "Round Off - _TC") | ||||
| 		pi = frappe.copy_doc(test_records[0]) | ||||
| @ -811,29 +833,12 @@ class TestPurchaseInvoice(unittest.TestCase): | ||||
| 
 | ||||
| 		pi.shipping_rule = shipping_rule.name | ||||
| 		pi.insert() | ||||
| 
 | ||||
| 		shipping_amount = 0.0 | ||||
| 		for condition in shipping_rule.get("conditions"): | ||||
| 			if not condition.to_value or (flt(condition.from_value) <= pi.net_total <= flt(condition.to_value)): | ||||
| 				shipping_amount = condition.shipping_amount | ||||
| 
 | ||||
| 		shipping_charge = { | ||||
| 			"doctype": "Purchase Taxes and Charges", | ||||
| 			"category": "Valuation and Total", | ||||
| 			"charge_type": "Actual", | ||||
| 			"account_head": shipping_rule.account, | ||||
| 			"cost_center": shipping_rule.cost_center, | ||||
| 			"tax_amount": shipping_amount, | ||||
| 			"description": shipping_rule.name, | ||||
| 			"add_deduct_tax": "Add" | ||||
| 		} | ||||
| 		pi.append("taxes", shipping_charge) | ||||
| 		pi.save() | ||||
| 
 | ||||
| 		self.assertEqual(pi.net_total, 1250) | ||||
| 
 | ||||
| 		self.assertEqual(pi.total_taxes_and_charges, 462.3) | ||||
| 		self.assertEqual(pi.grand_total, 1712.3) | ||||
| 		self.assertEqual(pi.total_taxes_and_charges, 354.1) | ||||
| 		self.assertEqual(pi.grand_total, 1604.1) | ||||
| 
 | ||||
| 	def test_make_pi_without_terms(self): | ||||
| 		pi = make_purchase_invoice(do_not_save=1) | ||||
| @ -1155,25 +1160,21 @@ class TestPurchaseInvoice(unittest.TestCase): | ||||
| 		# Create Purchase Order with TDS applied | ||||
| 		po = create_purchase_order(do_not_save=1, supplier=supplier.name, rate=3000, item='_Test Non Stock Item', | ||||
| 			posting_date='2021-09-15') | ||||
| 		po.apply_tds = 1 | ||||
| 		po.tax_withholding_category = 'TDS - 194 - Dividends - Individual' | ||||
| 		po.save() | ||||
| 		po.submit() | ||||
| 
 | ||||
| 		# Update Unrealized Profit / Loss Account which is used as default advance tax account | ||||
| 		frappe.db.set_value('Company', '_Test Company', 'unrealized_profit_loss_account', '_Test Account Excise Duty - _TC') | ||||
| 
 | ||||
| 		# Create Payment Entry Against the order | ||||
| 		payment_entry = get_payment_entry(dt='Purchase Order', dn=po.name) | ||||
| 		payment_entry.paid_from = 'Cash - _TC' | ||||
| 		payment_entry.apply_tax_withholding_amount = 1 | ||||
| 		payment_entry.tax_withholding_category = 'TDS - 194 - Dividends - Individual' | ||||
| 		payment_entry.save() | ||||
| 		payment_entry.submit() | ||||
| 
 | ||||
| 		# Check GLE for Payment Entry | ||||
| 		expected_gle = [ | ||||
| 			['_Test Account Excise Duty - _TC', 3000, 0], | ||||
| 			['Cash - _TC', 0, 27000], | ||||
| 			['Creditors - _TC', 27000, 0], | ||||
| 			['Creditors - _TC', 30000, 0], | ||||
| 			['TDS Payable - _TC', 0, 3000], | ||||
| 		] | ||||
| 
 | ||||
| @ -1199,9 +1200,7 @@ class TestPurchaseInvoice(unittest.TestCase): | ||||
| 		# Zero net effect on final TDS Payable on invoice | ||||
| 		expected_gle = [ | ||||
| 			['_Test Account Cost for Goods Sold - _TC', 30000], | ||||
| 			['_Test Account Excise Duty - _TC', -3000], | ||||
| 			['Creditors - _TC', -27000], | ||||
| 			['TDS Payable - _TC', 0] | ||||
| 			['Creditors - _TC', -30000] | ||||
| 		] | ||||
| 
 | ||||
| 		gl_entries = frappe.db.sql("""select account, sum(debit - credit) as amount | ||||
| @ -1214,6 +1213,14 @@ class TestPurchaseInvoice(unittest.TestCase): | ||||
| 			self.assertEqual(expected_gle[i][0], gle.account) | ||||
| 			self.assertEqual(expected_gle[i][1], gle.amount) | ||||
| 
 | ||||
| 		payment_entry.load_from_db() | ||||
| 		self.assertEqual(payment_entry.taxes[0].allocated_amount, 3000) | ||||
| 
 | ||||
| 		purchase_invoice.cancel() | ||||
| 
 | ||||
| 		payment_entry.load_from_db() | ||||
| 		self.assertEqual(payment_entry.taxes[0].allocated_amount, 0) | ||||
| 
 | ||||
| def check_gl_entries(doc, voucher_no, expected_gle, posting_date): | ||||
| 	gl_entries = frappe.db.sql("""select account, debit, credit, posting_date | ||||
| 		from `tabGL Entry` | ||||
|  | ||||
| @ -22,10 +22,10 @@ | ||||
|   "received_qty", | ||||
|   "qty", | ||||
|   "rejected_qty", | ||||
|   "stock_uom", | ||||
|   "col_break2", | ||||
|   "uom", | ||||
|   "conversion_factor", | ||||
|   "stock_uom", | ||||
|   "stock_qty", | ||||
|   "sec_break1", | ||||
|   "price_list_rate", | ||||
| @ -175,7 +175,8 @@ | ||||
|   { | ||||
|    "fieldname": "received_qty", | ||||
|    "fieldtype": "Float", | ||||
|    "label": "Received Qty" | ||||
|    "label": "Received Qty", | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "bold": 1, | ||||
| @ -223,7 +224,7 @@ | ||||
|   { | ||||
|    "fieldname": "stock_qty", | ||||
|    "fieldtype": "Float", | ||||
|    "label": "Stock Qty", | ||||
|    "label": "Accepted Qty in Stock UOM", | ||||
|    "print_hide": 1, | ||||
|    "read_only": 1, | ||||
|    "reqd": 1 | ||||
| @ -870,10 +871,11 @@ | ||||
|  "idx": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-09-01 16:04:03.538643", | ||||
|  "modified": "2021-11-15 17:04:07.191013", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Purchase Invoice Item", | ||||
|  "naming_rule": "Random", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [], | ||||
|  "sort_field": "modified", | ||||
|  | ||||
| @ -15,8 +15,17 @@ erpnext.accounts.SalesInvoiceController = class SalesInvoiceController extends e | ||||
| 
 | ||||
| 		let me = this; | ||||
| 		if (this.frm.doc.company) { | ||||
| 			frappe.db.get_value('Company', this.frm.doc.company, 'default_receivable_account', (r) => { | ||||
| 				me.frm.set_value('debit_to', r.default_receivable_account); | ||||
| 			frappe.call({ | ||||
| 				method: | ||||
| 					"erpnext.accounts.party.get_party_account", | ||||
| 				args: { | ||||
| 					party_type: 'Customer', | ||||
| 					party: this.frm.doc.customer, | ||||
| 					company: this.frm.doc.company | ||||
| 				}, | ||||
| 				callback: (response) => { | ||||
| 					if (response) me.frm.set_value("debit_to", response.message); | ||||
| 				}, | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| @ -507,15 +516,6 @@ cur_frm.fields_dict.write_off_cost_center.get_query = function(doc) { | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // project name
 | ||||
| //--------------------------
 | ||||
| cur_frm.fields_dict['project'].get_query = function(doc, cdt, cdn) { | ||||
| 	return{ | ||||
| 		query: "erpnext.controllers.queries.get_project_name", | ||||
| 		filters: {'customer': doc.customer} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Income Account in Details Table
 | ||||
| // --------------------------------
 | ||||
| cur_frm.set_query("income_account", "items", function(doc) { | ||||
| @ -969,7 +969,7 @@ frappe.ui.form.on('Sales Invoice', { | ||||
| 		} | ||||
| 
 | ||||
| 		if (frm.doc.is_debit_note) { | ||||
| 			frm.set_df_property('return_against', 'label', 'Adjustment Against'); | ||||
| 			frm.set_df_property('return_against', 'label', __('Adjustment Against')); | ||||
| 		} | ||||
| 
 | ||||
| 		if (frappe.boot.active_domains.includes("Healthcare")) { | ||||
| @ -979,10 +979,10 @@ frappe.ui.form.on('Sales Invoice', { | ||||
| 			if (cint(frm.doc.docstatus==0) && cur_frm.page.current_view_name!=="pos" && !frm.doc.is_return) { | ||||
| 				frm.add_custom_button(__('Healthcare Services'), function() { | ||||
| 					get_healthcare_services_to_invoice(frm); | ||||
| 				},"Get Items From"); | ||||
| 				},__("Get Items From")); | ||||
| 				frm.add_custom_button(__('Prescriptions'), function() { | ||||
| 					get_drugs_to_invoice(frm); | ||||
| 				},"Get Items From"); | ||||
| 				},__("Get Items From")); | ||||
| 			} | ||||
| 		} | ||||
| 		else { | ||||
|  | ||||
| @ -182,6 +182,7 @@ | ||||
|   "sales_team_section_break", | ||||
|   "sales_partner", | ||||
|   "column_break10", | ||||
|   "amount_eligible_for_commission", | ||||
|   "commission_rate", | ||||
|   "total_commission", | ||||
|   "section_break2", | ||||
| @ -2019,6 +2020,12 @@ | ||||
|    "label": "Total Billing Hours", | ||||
|    "print_hide": 1, | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "amount_eligible_for_commission", | ||||
|    "fieldtype": "Currency", | ||||
|    "label": "Amount Eligible for Commission", | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "icon": "fa fa-file-text", | ||||
| @ -2031,7 +2038,7 @@ | ||||
|    "link_fieldname": "consolidated_invoice" | ||||
|   } | ||||
|  ], | ||||
|  "modified": "2021-10-11 20:19:38.667508", | ||||
|  "modified": "2021-10-21 20:19:38.667508", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Sales Invoice", | ||||
|  | ||||
| @ -831,8 +831,6 @@ class SalesInvoice(SellingController): | ||||
| 		self.make_exchange_gain_loss_gl_entries(gl_entries) | ||||
| 		self.make_internal_transfer_gl_entries(gl_entries) | ||||
| 
 | ||||
| 		self.allocate_advance_taxes(gl_entries) | ||||
| 
 | ||||
| 		self.make_item_gl_entries(gl_entries) | ||||
| 		self.make_discount_gl_entries(gl_entries) | ||||
| 
 | ||||
| @ -1935,7 +1933,7 @@ def get_mode_of_payments_info(mode_of_payments, company): | ||||
| 			mpa.parent = mp.name and | ||||
| 			mpa.company = %s and | ||||
| 			mp.enabled = 1 and | ||||
| 			mp.name in (%s) | ||||
| 			mp.name in %s | ||||
| 		group by | ||||
| 			mp.name | ||||
| 		""", (company, mode_of_payments), as_dict=1) | ||||
|  | ||||
| @ -1603,28 +1603,12 @@ class TestSalesInvoice(unittest.TestCase): | ||||
| 
 | ||||
| 		si.shipping_rule = shipping_rule.name | ||||
| 		si.insert() | ||||
| 
 | ||||
| 		shipping_amount = 0.0 | ||||
| 		for condition in shipping_rule.get("conditions"): | ||||
| 			if not condition.to_value or (flt(condition.from_value) <= si.net_total <= flt(condition.to_value)): | ||||
| 				shipping_amount = condition.shipping_amount | ||||
| 
 | ||||
| 		shipping_charge = { | ||||
| 			"doctype": "Sales Taxes and Charges", | ||||
| 			"category": "Valuation and Total", | ||||
| 			"charge_type": "Actual", | ||||
| 			"account_head": shipping_rule.account, | ||||
| 			"cost_center": shipping_rule.cost_center, | ||||
| 			"tax_amount": shipping_amount, | ||||
| 			"description": shipping_rule.name | ||||
| 		} | ||||
| 		si.append("taxes", shipping_charge) | ||||
| 		si.save() | ||||
| 
 | ||||
| 		self.assertEqual(si.net_total, 1250) | ||||
| 
 | ||||
| 		self.assertEqual(si.total_taxes_and_charges, 577.05) | ||||
| 		self.assertEqual(si.grand_total, 1827.05) | ||||
| 		self.assertEqual(si.total_taxes_and_charges, 468.85) | ||||
| 		self.assertEqual(si.grand_total, 1718.85) | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| @ -2316,6 +2300,7 @@ class TestSalesInvoice(unittest.TestCase): | ||||
| 		from erpnext.accounts.doctype.opening_invoice_creation_tool.test_opening_invoice_creation_tool import ( | ||||
| 			make_customer, | ||||
| 		) | ||||
| 		from erpnext.accounts.doctype.party_link.party_link import create_party_link | ||||
| 		from erpnext.buying.doctype.supplier.test_supplier import create_supplier | ||||
| 
 | ||||
| 		# create a customer | ||||
| @ -2324,13 +2309,7 @@ class TestSalesInvoice(unittest.TestCase): | ||||
| 		supplier = create_supplier(supplier_name="_Test Common Supplier").name | ||||
| 
 | ||||
| 		# create a party link between customer & supplier | ||||
| 		# set primary role as supplier | ||||
| 		party_link = frappe.new_doc("Party Link") | ||||
| 		party_link.primary_role = "Supplier" | ||||
| 		party_link.primary_party = supplier | ||||
| 		party_link.secondary_role = "Customer" | ||||
| 		party_link.secondary_party = customer | ||||
| 		party_link.save() | ||||
| 		party_link = create_party_link("Supplier", supplier, customer) | ||||
| 
 | ||||
| 		# enable common party accounting | ||||
| 		frappe.db.set_value('Accounts Settings', None, 'enable_common_party_accounting', 1) | ||||
| @ -2406,6 +2385,29 @@ class TestSalesInvoice(unittest.TestCase): | ||||
| 		si.reload() | ||||
| 		self.assertEqual(si.status, "Paid") | ||||
| 
 | ||||
| 	def test_sales_commission(self): | ||||
| 		si = frappe.copy_doc(test_records[0]) | ||||
| 		item = copy.deepcopy(si.get('items')[0]) | ||||
| 		item.update({ | ||||
| 			"qty": 1, | ||||
| 			"rate": 500, | ||||
| 			"grant_commission": 1 | ||||
| 		}) | ||||
| 		si.append("items", item) | ||||
| 
 | ||||
| 		# Test valid values | ||||
| 		for commission_rate, total_commission in ((0, 0), (10, 50), (100, 500)): | ||||
| 			si.commission_rate = commission_rate | ||||
| 			si.save() | ||||
| 			self.assertEqual(si.amount_eligible_for_commission, 500) | ||||
| 			self.assertEqual(si.total_commission, total_commission) | ||||
| 
 | ||||
| 		# Test invalid values | ||||
| 		for commission_rate in (101, -1): | ||||
| 			si.reload() | ||||
| 			si.commission_rate = commission_rate | ||||
| 			self.assertRaises(frappe.ValidationError, si.save) | ||||
| 
 | ||||
| 	def test_sales_invoice_submission_post_account_freezing_date(self): | ||||
| 		frappe.db.set_value('Accounts Settings', None, 'acc_frozen_upto', add_days(getdate(), 1)) | ||||
| 		si = create_sales_invoice(do_not_save=True) | ||||
| @ -2418,6 +2420,32 @@ class TestSalesInvoice(unittest.TestCase): | ||||
| 
 | ||||
| 		frappe.db.set_value('Accounts Settings', None, 'acc_frozen_upto', None) | ||||
| 
 | ||||
| 	def test_over_billing_case_against_delivery_note(self): | ||||
| 		''' | ||||
| 			Test a case where duplicating the item with qty = 1 in the invoice | ||||
| 			allows overbilling even if it is disabled | ||||
| 		''' | ||||
| 		from erpnext.stock.doctype.delivery_note.test_delivery_note import create_delivery_note | ||||
| 
 | ||||
| 		over_billing_allowance = frappe.db.get_single_value('Accounts Settings', 'over_billing_allowance') | ||||
| 		frappe.db.set_value('Accounts Settings', None, 'over_billing_allowance', 0) | ||||
| 
 | ||||
| 		dn = create_delivery_note() | ||||
| 		dn.submit() | ||||
| 
 | ||||
| 		si = make_sales_invoice(dn.name) | ||||
| 		# make a copy of first item and add it to invoice | ||||
| 		item_copy = frappe.copy_doc(si.items[0]) | ||||
| 		si.append('items', item_copy) | ||||
| 		si.save() | ||||
| 
 | ||||
| 		with self.assertRaises(frappe.ValidationError) as err: | ||||
| 			si.submit() | ||||
| 
 | ||||
| 		self.assertTrue("cannot overbill" in str(err.exception).lower()) | ||||
| 
 | ||||
| 		frappe.db.set_value('Accounts Settings', None, 'over_billing_allowance', over_billing_allowance) | ||||
| 
 | ||||
| def get_sales_invoice_for_e_invoice(): | ||||
| 	si = make_sales_invoice_for_ewaybill() | ||||
| 	si.naming_series = 'INV-2020-.#####' | ||||
|  | ||||
| @ -47,6 +47,7 @@ | ||||
|   "pricing_rules", | ||||
|   "stock_uom_rate", | ||||
|   "is_free_item", | ||||
|   "grant_commission", | ||||
|   "section_break_21", | ||||
|   "net_rate", | ||||
|   "net_amount", | ||||
| @ -828,15 +829,23 @@ | ||||
|    "fieldtype": "Link", | ||||
|    "label": "Discount Account", | ||||
|    "options": "Account" | ||||
|   }, | ||||
|   { | ||||
|    "default": "0", | ||||
|    "fieldname": "grant_commission", | ||||
|    "fieldtype": "Check", | ||||
|    "label": "Grant Commission", | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "idx": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-08-19 13:41:53.435827", | ||||
|  "modified": "2021-10-05 12:24:54.968907", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Sales Invoice Item", | ||||
|  "naming_rule": "Random", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [], | ||||
|  "sort_field": "modified", | ||||
|  | ||||
| @ -519,8 +519,6 @@ class Subscription(Document): | ||||
| 		2. Change the `Subscription` status to 'Past Due Date' | ||||
| 		3. Change the `Subscription` status to 'Cancelled' | ||||
| 		""" | ||||
| 		if getdate() > getdate(self.current_invoice_end) and self.is_prepaid_to_invoice(): | ||||
| 			self.update_subscription_period(add_days(self.current_invoice_end, 1)) | ||||
| 
 | ||||
| 		if not self.is_current_invoice_generated(self.current_invoice_start, self.current_invoice_end) \ | ||||
| 			and (self.is_postpaid_to_invoice() or self.is_prepaid_to_invoice()): | ||||
| @ -528,6 +526,9 @@ class Subscription(Document): | ||||
| 			prorate = frappe.db.get_single_value('Subscription Settings', 'prorate') | ||||
| 			self.generate_invoice(prorate) | ||||
| 
 | ||||
| 		if getdate() > getdate(self.current_invoice_end) and self.is_prepaid_to_invoice(): | ||||
| 			self.update_subscription_period(add_days(self.current_invoice_end, 1)) | ||||
| 
 | ||||
| 		if self.cancel_at_period_end and getdate() > getdate(self.current_invoice_end): | ||||
| 			self.cancel_subscription_at_period_end() | ||||
| 
 | ||||
|  | ||||
| @ -95,7 +95,7 @@ def get_party_tax_withholding_details(inv, tax_withholding_category=None): | ||||
| 		frappe.throw(_('Tax Withholding Category {} against Company {} for Customer {} should have Cumulative Threshold value.') | ||||
| 			.format(tax_withholding_category, inv.company, party)) | ||||
| 
 | ||||
| 	tax_amount, tax_deducted = get_tax_amount( | ||||
| 	tax_amount, tax_deducted, tax_deducted_on_advances = get_tax_amount( | ||||
| 		party_type, parties, | ||||
| 		inv, tax_details, | ||||
| 		posting_date, pan_no | ||||
| @ -106,6 +106,9 @@ def get_party_tax_withholding_details(inv, tax_withholding_category=None): | ||||
| 	else: | ||||
| 		tax_row = get_tax_row_for_tcs(inv, tax_details, tax_amount, tax_deducted) | ||||
| 
 | ||||
| 	if inv.doctype == 'Purchase Invoice': | ||||
| 		return tax_row, tax_deducted_on_advances | ||||
| 	else: | ||||
| 		return tax_row | ||||
| 
 | ||||
| def get_tax_withholding_details(tax_withholding_category, posting_date, company): | ||||
| @ -194,6 +197,10 @@ def get_tax_amount(party_type, parties, inv, tax_details, posting_date, pan_no=N | ||||
| 	advance_vouchers = get_advance_vouchers(parties, company=inv.company, from_date=tax_details.from_date, | ||||
| 		to_date=tax_details.to_date, party_type=party_type) | ||||
| 	taxable_vouchers = vouchers + advance_vouchers | ||||
| 	tax_deducted_on_advances = 0 | ||||
| 
 | ||||
| 	if inv.doctype == 'Purchase Invoice': | ||||
| 		tax_deducted_on_advances = get_taxes_deducted_on_advances_allocated(inv, tax_details) | ||||
| 
 | ||||
| 	tax_deducted = 0 | ||||
| 	if taxable_vouchers: | ||||
| @ -223,7 +230,7 @@ def get_tax_amount(party_type, parties, inv, tax_details, posting_date, pan_no=N | ||||
| 	if cint(tax_details.round_off_tax_amount): | ||||
| 		tax_amount = round(tax_amount) | ||||
| 
 | ||||
| 	return tax_amount, tax_deducted | ||||
| 	return tax_amount, tax_deducted, tax_deducted_on_advances | ||||
| 
 | ||||
| def get_invoice_vouchers(parties, tax_details, company, party_type='Supplier'): | ||||
| 	dr_or_cr = 'credit' if party_type == 'Supplier' else 'debit' | ||||
| @ -281,6 +288,29 @@ def get_advance_vouchers(parties, company=None, from_date=None, to_date=None, pa | ||||
| 
 | ||||
| 	return frappe.get_all('GL Entry', filters=filters, distinct=1, pluck='voucher_no') or [""] | ||||
| 
 | ||||
| def get_taxes_deducted_on_advances_allocated(inv, tax_details): | ||||
| 	advances = [d.reference_name for d in inv.get('advances')] | ||||
| 	tax_info = [] | ||||
| 
 | ||||
| 	if advances: | ||||
| 		pe = frappe.qb.DocType("Payment Entry").as_("pe") | ||||
| 		at = frappe.qb.DocType("Advance Taxes and Charges").as_("at") | ||||
| 
 | ||||
| 		tax_info = frappe.qb.from_(at).inner_join(pe).on( | ||||
| 			pe.name == at.parent | ||||
| 		).select( | ||||
| 			at.parent, at.name, at.tax_amount, at.allocated_amount | ||||
| 		).where( | ||||
| 			pe.tax_withholding_category == tax_details.get('tax_withholding_category') | ||||
| 		).where( | ||||
| 			at.parent.isin(advances) | ||||
| 		).where( | ||||
| 			at.account_head == tax_details.account_head | ||||
| 		).run(as_dict=True) | ||||
| 
 | ||||
| 	return tax_info | ||||
| 
 | ||||
| 
 | ||||
| def get_deducted_tax(taxable_vouchers, tax_details): | ||||
| 	# check if TDS / TCS account is already charged on taxable vouchers | ||||
| 	filters = { | ||||
|  | ||||
| @ -73,8 +73,28 @@ def process_gl_map(gl_map, merge_entries=True, precision=None): | ||||
| 				flt(entry.debit_in_account_currency) - flt(entry.credit_in_account_currency) | ||||
| 			entry.credit_in_account_currency = 0.0 | ||||
| 
 | ||||
| 		update_net_values(entry) | ||||
| 
 | ||||
| 	return gl_map | ||||
| 
 | ||||
| def update_net_values(entry): | ||||
| 	# In some scenarios net value needs to be shown in the ledger | ||||
| 	# This method updates net values as debit or credit | ||||
| 	if entry.post_net_value and entry.debit and entry.credit: | ||||
| 		if entry.debit > entry.credit: | ||||
| 			entry.debit = entry.debit - entry.credit | ||||
| 			entry.debit_in_account_currency = entry.debit_in_account_currency \ | ||||
| 				- entry.credit_in_account_currency | ||||
| 			entry.credit = 0 | ||||
| 			entry.credit_in_account_currency = 0 | ||||
| 		else: | ||||
| 			entry.credit = entry.credit - entry.debit | ||||
| 			entry.credit_in_account_currency = entry.credit_in_account_currency \ | ||||
| 				- entry.debit_in_account_currency | ||||
| 
 | ||||
| 			entry.debit = 0 | ||||
| 			entry.debit_in_account_currency = 0 | ||||
| 
 | ||||
| def merge_similar_entries(gl_map, precision=None): | ||||
| 	merged_gl_map = [] | ||||
| 	accounting_dimensions = get_accounting_dimensions() | ||||
|  | ||||
| @ -83,7 +83,8 @@ def _get_party_details(party=None, account=None, party_type="Customer", company= | ||||
| 	if party_type=="Customer": | ||||
| 		party_details["sales_team"] = [{ | ||||
| 			"sales_person": d.sales_person, | ||||
| 			"allocated_percentage": d.allocated_percentage or None | ||||
| 			"allocated_percentage": d.allocated_percentage or None, | ||||
| 			"commission_rate": d.commission_rate | ||||
| 		} for d in party.get("sales_team")] | ||||
| 
 | ||||
| 	# supplier tax withholding category | ||||
| @ -218,7 +219,7 @@ def set_account_and_due_date(party, account, party_type, company, posting_date, | ||||
| 	return out | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| def get_party_account(party_type, party, company=None): | ||||
| def get_party_account(party_type, party=None, company=None): | ||||
| 	"""Returns the account for the given `party`. | ||||
| 		Will first search in party (Customer / Supplier) record, if not found, | ||||
| 		will search in group (Customer Group / Supplier Group), | ||||
| @ -226,8 +227,11 @@ def get_party_account(party_type, party, company=None): | ||||
| 	if not company: | ||||
| 		frappe.throw(_("Please select a Company")) | ||||
| 
 | ||||
| 	if not party: | ||||
| 		return | ||||
| 	if not party and party_type in ['Customer', 'Supplier']: | ||||
| 		default_account_name = "default_receivable_account" \ | ||||
| 			if party_type=="Customer" else "default_payable_account" | ||||
| 
 | ||||
| 		return frappe.get_cached_value('Company',  company,  default_account_name) | ||||
| 
 | ||||
| 	account = frappe.db.get_value("Party Account", | ||||
| 		{"parenttype": party_type, "parent": party, "company": company}, "account") | ||||
|  | ||||
| @ -92,6 +92,11 @@ frappe.require("assets/erpnext/js/financial_statements.js", function() { | ||||
| 				"label": __("Include Default Book Entries"), | ||||
| 				"fieldtype": "Check", | ||||
| 				"default": 1 | ||||
| 			}, | ||||
| 			{ | ||||
| 				"fieldname": "show_zero_values", | ||||
| 				"label": __("Show zero values"), | ||||
| 				"fieldtype": "Check" | ||||
| 			} | ||||
| 		], | ||||
| 		"formatter": function(value, row, column, data, default_formatter) { | ||||
|  | ||||
| @ -22,7 +22,11 @@ from erpnext.accounts.report.cash_flow.cash_flow import ( | ||||
| 	get_cash_flow_accounts, | ||||
| ) | ||||
| from erpnext.accounts.report.cash_flow.cash_flow import get_report_summary as get_cash_flow_summary | ||||
| from erpnext.accounts.report.financial_statements import get_fiscal_year_data, sort_accounts | ||||
| from erpnext.accounts.report.financial_statements import ( | ||||
| 	filter_out_zero_value_rows, | ||||
| 	get_fiscal_year_data, | ||||
| 	sort_accounts, | ||||
| ) | ||||
| from erpnext.accounts.report.profit_and_loss_statement.profit_and_loss_statement import ( | ||||
| 	get_chart_data as get_pl_chart_data, | ||||
| ) | ||||
| @ -265,7 +269,7 @@ def get_columns(companies, filters): | ||||
| 	return columns | ||||
| 
 | ||||
| def get_data(companies, root_type, balance_must_be, fiscal_year, filters=None, ignore_closing_entries=False): | ||||
| 	accounts, accounts_by_name = get_account_heads(root_type, | ||||
| 	accounts, accounts_by_name, parent_children_map = get_account_heads(root_type, | ||||
| 		companies, filters) | ||||
| 
 | ||||
| 	if not accounts: return [] | ||||
| @ -294,6 +298,8 @@ def get_data(companies, root_type, balance_must_be, fiscal_year, filters=None, i | ||||
| 
 | ||||
| 	out = prepare_data(accounts, start_date, end_date, balance_must_be, companies, company_currency, filters) | ||||
| 
 | ||||
| 	out = filter_out_zero_value_rows(out, parent_children_map, show_zero_values=filters.get("show_zero_values")) | ||||
| 
 | ||||
| 	if out: | ||||
| 		add_total_row(out, root_type, balance_must_be, companies, company_currency) | ||||
| 
 | ||||
| @ -370,7 +376,7 @@ def get_account_heads(root_type, companies, filters): | ||||
| 
 | ||||
| 	accounts, accounts_by_name, parent_children_map = filter_accounts(accounts) | ||||
| 
 | ||||
| 	return accounts, accounts_by_name | ||||
| 	return accounts, accounts_by_name, parent_children_map | ||||
| 
 | ||||
| def update_parent_account_names(accounts): | ||||
| 	"""Update parent_account_name in accounts list. | ||||
|  | ||||
| @ -420,8 +420,7 @@ def set_gl_entries_by_account( | ||||
| 			{additional_conditions} | ||||
| 			and posting_date <= %(to_date)s | ||||
| 			and is_cancelled = 0 | ||||
| 			{distributed_cost_center_query} | ||||
| 			order by account, posting_date""".format( | ||||
| 			{distributed_cost_center_query}""".format( | ||||
| 				additional_conditions=additional_conditions, | ||||
| 				distributed_cost_center_query=distributed_cost_center_query), gl_filters, as_dict=True) #nosec | ||||
| 
 | ||||
|  | ||||
| @ -6,7 +6,7 @@ from collections import OrderedDict | ||||
| 
 | ||||
| import frappe | ||||
| from frappe import _, _dict | ||||
| from frappe.utils import cstr, flt, getdate | ||||
| from frappe.utils import cstr, getdate | ||||
| 
 | ||||
| from erpnext import get_company_currency, get_default_company | ||||
| from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import ( | ||||
| @ -17,6 +17,8 @@ from erpnext.accounts.report.financial_statements import get_cost_centers_with_c | ||||
| from erpnext.accounts.report.utils import convert_to_presentation_currency, get_currency | ||||
| from erpnext.accounts.utils import get_account_currency | ||||
| 
 | ||||
| # to cache translations | ||||
| TRANSLATIONS = frappe._dict() | ||||
| 
 | ||||
| def execute(filters=None): | ||||
| 	if not filters: | ||||
| @ -42,10 +44,20 @@ def execute(filters=None): | ||||
| 
 | ||||
| 	columns = get_columns(filters) | ||||
| 
 | ||||
| 	update_translations() | ||||
| 
 | ||||
| 	res = get_result(filters, account_details) | ||||
| 
 | ||||
| 	return columns, res | ||||
| 
 | ||||
| def update_translations(): | ||||
| 	TRANSLATIONS.update( | ||||
| 		dict( | ||||
| 			OPENING = _('Opening'), | ||||
| 			TOTAL = _('Total'), | ||||
| 			CLOSING_TOTAL = _('Closing (Opening + Total)') | ||||
| 		) | ||||
| 	) | ||||
| 
 | ||||
| def validate_filters(filters, account_details): | ||||
| 	if not filters.get("company"): | ||||
| @ -351,9 +363,9 @@ def get_totals_dict(): | ||||
| 			credit_in_account_currency=0.0 | ||||
| 		) | ||||
| 	return _dict( | ||||
| 		opening = _get_debit_credit_dict(_('Opening')), | ||||
| 		total = _get_debit_credit_dict(_('Total')), | ||||
| 		closing = _get_debit_credit_dict(_('Closing (Opening + Total)')) | ||||
| 		opening = _get_debit_credit_dict(TRANSLATIONS.OPENING), | ||||
| 		total = _get_debit_credit_dict(TRANSLATIONS.TOTAL), | ||||
| 		closing = _get_debit_credit_dict(TRANSLATIONS.CLOSING_TOTAL) | ||||
| 	) | ||||
| 
 | ||||
| def group_by_field(group_by): | ||||
| @ -378,22 +390,23 @@ def get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map): | ||||
| 	entries = [] | ||||
| 	consolidated_gle = OrderedDict() | ||||
| 	group_by = group_by_field(filters.get('group_by')) | ||||
| 	group_by_voucher_consolidated = filters.get("group_by") == 'Group by Voucher (Consolidated)' | ||||
| 
 | ||||
| 	if filters.get('show_net_values_in_party_account'): | ||||
| 		account_type_map = get_account_type_map(filters.get('company')) | ||||
| 
 | ||||
| 	def update_value_in_dict(data, key, gle): | ||||
| 		data[key].debit += flt(gle.debit) | ||||
| 		data[key].credit += flt(gle.credit) | ||||
| 		data[key].debit += gle.debit | ||||
| 		data[key].credit += gle.credit | ||||
| 
 | ||||
| 		data[key].debit_in_account_currency += flt(gle.debit_in_account_currency) | ||||
| 		data[key].credit_in_account_currency += flt(gle.credit_in_account_currency) | ||||
| 		data[key].debit_in_account_currency += gle.debit_in_account_currency | ||||
| 		data[key].credit_in_account_currency += gle.credit_in_account_currency | ||||
| 
 | ||||
| 		if filters.get('show_net_values_in_party_account') and \ | ||||
| 			account_type_map.get(data[key].account) in ('Receivable', 'Payable'): | ||||
| 			net_value = flt(data[key].debit) - flt(data[key].credit) | ||||
| 			net_value_in_account_currency = flt(data[key].debit_in_account_currency) \ | ||||
| 				- flt(data[key].credit_in_account_currency) | ||||
| 			net_value = data[key].debit - data[key].credit | ||||
| 			net_value_in_account_currency = data[key].debit_in_account_currency \ | ||||
| 				- data[key].credit_in_account_currency | ||||
| 
 | ||||
| 			if net_value < 0: | ||||
| 				dr_or_cr = 'credit' | ||||
| @ -411,19 +424,29 @@ def get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map): | ||||
| 			data[key].against_voucher += ', ' + gle.against_voucher | ||||
| 
 | ||||
| 	from_date, to_date = getdate(filters.from_date), getdate(filters.to_date) | ||||
| 	for gle in gl_entries: | ||||
| 		if (gle.posting_date < from_date or | ||||
| 			(cstr(gle.is_opening) == "Yes" and not filters.get("show_opening_entries"))): | ||||
| 			update_value_in_dict(gle_map[gle.get(group_by)].totals, 'opening', gle) | ||||
| 			update_value_in_dict(totals, 'opening', gle) | ||||
| 	show_opening_entries = filters.get("show_opening_entries") | ||||
| 
 | ||||
| 			update_value_in_dict(gle_map[gle.get(group_by)].totals, 'closing', gle) | ||||
| 	for gle in gl_entries: | ||||
| 		group_by_value = gle.get(group_by) | ||||
| 
 | ||||
| 		if (gle.posting_date < from_date or (cstr(gle.is_opening) == "Yes" and not show_opening_entries)): | ||||
| 			if not group_by_voucher_consolidated: | ||||
| 				update_value_in_dict(gle_map[group_by_value].totals, 'opening', gle) | ||||
| 				update_value_in_dict(gle_map[group_by_value].totals, 'closing', gle) | ||||
| 
 | ||||
| 			update_value_in_dict(totals, 'opening', gle) | ||||
| 			update_value_in_dict(totals, 'closing', gle) | ||||
| 
 | ||||
| 		elif gle.posting_date <= to_date: | ||||
| 			if filters.get("group_by") != 'Group by Voucher (Consolidated)': | ||||
| 				gle_map[gle.get(group_by)].entries.append(gle) | ||||
| 			elif filters.get("group_by") == 'Group by Voucher (Consolidated)': | ||||
| 			if not group_by_voucher_consolidated: | ||||
| 				update_value_in_dict(gle_map[group_by_value].totals, 'total', gle) | ||||
| 				update_value_in_dict(gle_map[group_by_value].totals, 'closing', gle) | ||||
| 				update_value_in_dict(totals, 'total', gle) | ||||
| 				update_value_in_dict(totals, 'closing', gle) | ||||
| 
 | ||||
| 				gle_map[group_by_value].entries.append(gle) | ||||
| 
 | ||||
| 			elif group_by_voucher_consolidated: | ||||
| 				keylist = [gle.get("voucher_type"), gle.get("voucher_no"), gle.get("account")] | ||||
| 				for dim in accounting_dimensions: | ||||
| 					keylist.append(gle.get(dim)) | ||||
| @ -435,9 +458,7 @@ def get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map): | ||||
| 					update_value_in_dict(consolidated_gle, key, gle) | ||||
| 
 | ||||
| 	for key, value in consolidated_gle.items(): | ||||
| 		update_value_in_dict(gle_map[value.get(group_by)].totals, 'total', value) | ||||
| 		update_value_in_dict(totals, 'total', value) | ||||
| 		update_value_in_dict(gle_map[value.get(group_by)].totals, 'closing', value) | ||||
| 		update_value_in_dict(totals, 'closing', value) | ||||
| 		entries.append(value) | ||||
| 
 | ||||
|  | ||||
| @ -44,7 +44,7 @@ frappe.query_reports["Gross Profit"] = { | ||||
| 	"formatter": function(value, row, column, data, default_formatter) { | ||||
| 		value = default_formatter(value, row, column, data); | ||||
| 
 | ||||
| 		if (data && data.indent == 0.0) { | ||||
| 		if (data && (data.indent == 0.0 || row[1].content == "Total")) { | ||||
| 			value = $(`<span>${value}</span>`); | ||||
| 			var $value = $(value).css("font-weight", "bold"); | ||||
| 			value = $value.wrap("<p></p>").parent().html(); | ||||
|  | ||||
| @ -9,7 +9,7 @@ | ||||
|  "filters": [], | ||||
|  "idx": 3, | ||||
|  "is_standard": "Yes", | ||||
|  "modified": "2021-08-19 18:57:07.468202", | ||||
|  "modified": "2021-11-13 19:14:23.730198", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Gross Profit", | ||||
|  | ||||
| @ -19,7 +19,7 @@ def execute(filters=None): | ||||
| 	data = [] | ||||
| 
 | ||||
| 	group_wise_columns = frappe._dict({ | ||||
| 		"invoice": ["parent", "customer", "customer_group", "posting_date","item_code", "item_name","item_group", "brand", "description", \ | ||||
| 		"invoice": ["invoice_or_item", "customer", "customer_group", "posting_date","item_code", "item_name","item_group", "brand", "description", | ||||
| 			"warehouse", "qty", "base_rate", "buying_rate", "base_amount", | ||||
| 			"buying_amount", "gross_profit", "gross_profit_percent", "project"], | ||||
| 		"item_code": ["item_code", "item_name", "brand", "description", "qty", "base_rate", | ||||
| @ -77,13 +77,15 @@ def get_data_when_not_grouped_by_invoice(gross_profit_data, filters, group_wise_ | ||||
| 
 | ||||
| 		row.append(filters.currency) | ||||
| 		if idx == len(gross_profit_data.grouped_data)-1: | ||||
| 			row[0] = frappe.bold("Total") | ||||
| 			row[0] = "Total" | ||||
| 
 | ||||
| 		data.append(row) | ||||
| 
 | ||||
| def get_columns(group_wise_columns, filters): | ||||
| 	columns = [] | ||||
| 	column_map = frappe._dict({ | ||||
| 		"parent": _("Sales Invoice") + ":Link/Sales Invoice:120", | ||||
| 		"invoice_or_item": _("Sales Invoice") + ":Link/Sales Invoice:120", | ||||
| 		"posting_date": _("Posting Date") + ":Date:100", | ||||
| 		"posting_time": _("Posting Time") + ":Data:100", | ||||
| 		"item_code": _("Item Code") + ":Link/Item:100", | ||||
| @ -122,7 +124,7 @@ def get_columns(group_wise_columns, filters): | ||||
| 
 | ||||
| def get_column_names(): | ||||
| 	return frappe._dict({ | ||||
| 		'parent': 'sales_invoice', | ||||
| 		'invoice_or_item': 'sales_invoice', | ||||
| 		'customer': 'customer', | ||||
| 		'customer_group': 'customer_group', | ||||
| 		'posting_date': 'posting_date', | ||||
| @ -245,6 +247,7 @@ class GrossProfitGenerator(object): | ||||
| 				self.add_to_totals(new_row) | ||||
| 			else: | ||||
| 				for i, row in enumerate(self.grouped[key]): | ||||
| 					if row.indent == 1.0: | ||||
| 						if row.parent in self.returned_invoices \ | ||||
| 								and row.item_code in self.returned_invoices[row.parent]: | ||||
| 							returned_item_rows = self.returned_invoices[row.parent][row.item_code] | ||||
| @ -252,11 +255,19 @@ class GrossProfitGenerator(object): | ||||
| 								row.qty += flt(returned_item_row.qty) | ||||
| 								row.base_amount += flt(returned_item_row.base_amount, self.currency_precision) | ||||
| 							row.buying_amount = flt(flt(row.qty) * flt(row.buying_rate), self.currency_precision) | ||||
| 					if (flt(row.qty) or row.base_amount) and self.is_not_invoice_row(row): | ||||
| 						if (flt(row.qty) or row.base_amount): | ||||
| 							row = self.set_average_rate(row) | ||||
| 							self.grouped_data.append(row) | ||||
| 						self.add_to_totals(row) | ||||
| 
 | ||||
| 		self.set_average_gross_profit(self.totals) | ||||
| 
 | ||||
| 		if self.filters.get("group_by") == "Invoice": | ||||
| 			self.totals.indent = 0.0 | ||||
| 			self.totals.parent_invoice = "" | ||||
| 			self.totals.invoice_or_item = "Total" | ||||
| 			self.si_list.append(self.totals) | ||||
| 		else: | ||||
| 			self.grouped_data.append(self.totals) | ||||
| 
 | ||||
| 	def is_not_invoice_row(self, row): | ||||
| @ -446,7 +457,7 @@ class GrossProfitGenerator(object): | ||||
| 				if not row.indent: | ||||
| 					row.indent = 1.0 | ||||
| 					row.parent_invoice = row.parent | ||||
| 					row.parent = row.item_code | ||||
| 					row.invoice_or_item = row.item_code | ||||
| 
 | ||||
| 					if frappe.db.exists('Product Bundle', row.item_code): | ||||
| 						self.add_bundle_items(row, index) | ||||
| @ -455,7 +466,8 @@ class GrossProfitGenerator(object): | ||||
| 		return frappe._dict({ | ||||
| 			'parent_invoice': "", | ||||
| 			'indent': 0.0, | ||||
| 			'parent': row.parent, | ||||
| 			'invoice_or_item': row.parent, | ||||
| 			'parent': None, | ||||
| 			'posting_date': row.posting_date, | ||||
| 			'posting_time': row.posting_time, | ||||
| 			'project': row.project, | ||||
| @ -499,7 +511,8 @@ class GrossProfitGenerator(object): | ||||
| 		return frappe._dict({ | ||||
| 			'parent_invoice': product_bundle.item_code, | ||||
| 			'indent': product_bundle.indent + 1, | ||||
| 			'parent': item.item_code, | ||||
| 			'parent': None, | ||||
| 			'invoice_or_item': item.item_code, | ||||
| 			'posting_date': product_bundle.posting_date, | ||||
| 			'posting_time': product_bundle.posting_time, | ||||
| 			'project': product_bundle.project, | ||||
|  | ||||
| @ -1,18 +1,21 @@ | ||||
| { | ||||
|  "add_total_row": 0, | ||||
|  "apply_user_permissions": 1,  | ||||
|  "columns": [], | ||||
|  "creation": "2013-05-06 12:28:23", | ||||
|  "disable_prepared_report": 0, | ||||
|  "disabled": 0, | ||||
|  "docstatus": 0, | ||||
|  "doctype": "Report", | ||||
|  "filters": [], | ||||
|  "idx": 3, | ||||
|  "is_standard": "Yes", | ||||
|  "modified": "2017-03-06 05:52:57.645281",  | ||||
|  "modified": "2021-10-06 06:26:07.881340", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Sales Partners Commission", | ||||
|  "owner": "Administrator", | ||||
|  "query": "SELECT\n    sales_partner as \"Sales Partner:Link/Sales Partner:150\",\n\tsum(base_net_total) as \"Invoiced Amount (Exclusive Tax):Currency:210\",\n\tsum(total_commission) as \"Total Commission:Currency:150\",\n\tsum(total_commission)*100/sum(base_net_total) as \"Average Commission Rate:Currency:170\"\nFROM\n\t`tabSales Invoice`\nWHERE\n\tdocstatus = 1 and ifnull(base_net_total, 0) > 0 and ifnull(total_commission, 0) > 0\nGROUP BY\n\tsales_partner\nORDER BY\n\t\"Total Commission:Currency:120\"",  | ||||
|  "prepared_report": 0, | ||||
|  "query": "SELECT\n    sales_partner as \"Sales Partner:Link/Sales Partner:220\",\n\tsum(base_net_total) as \"Invoiced Amount (Excl. Tax):Currency:220\",\n\tsum(amount_eligible_for_commission) as \"Amount Eligible for Commission:Currency:220\",\n\tsum(total_commission) as \"Total Commission:Currency:170\",\n\tsum(total_commission)*100/sum(amount_eligible_for_commission) as \"Average Commission Rate:Percent:220\"\nFROM\n\t`tabSales Invoice`\nWHERE\n\tdocstatus = 1 and ifnull(base_net_total, 0) > 0 and ifnull(total_commission, 0) > 0\nGROUP BY\n\tsales_partner\nORDER BY\n\t\"Total Commission:Currency:120\"", | ||||
|  "ref_doctype": "Sales Invoice", | ||||
|  "report_name": "Sales Partners Commission", | ||||
|  "report_type": "Query Report", | ||||
|  | ||||
| @ -60,6 +60,10 @@ frappe.ui.form.on('Asset Repair', { | ||||
| 		if (frm.doc.repair_status == "Completed") { | ||||
| 			frm.set_value('completion_date', frappe.datetime.now_datetime()); | ||||
| 		} | ||||
| 	}, | ||||
| 
 | ||||
| 	stock_items_on_form_rendered() { | ||||
| 		erpnext.setup_serial_or_batch_no(); | ||||
| 	} | ||||
| }); | ||||
| 
 | ||||
|  | ||||
| @ -118,9 +118,10 @@ class AssetRepair(AccountsController): | ||||
| 		for stock_item in self.get('stock_items'): | ||||
| 			stock_entry.append('items', { | ||||
| 				"s_warehouse": self.warehouse, | ||||
| 				"item_code": stock_item.item, | ||||
| 				"item_code": stock_item.item_code, | ||||
| 				"qty": stock_item.consumed_quantity, | ||||
| 				"basic_rate": stock_item.valuation_rate | ||||
| 				"basic_rate": stock_item.valuation_rate, | ||||
| 				"serial_no": stock_item.serial_no | ||||
| 			}) | ||||
| 
 | ||||
| 		stock_entry.insert() | ||||
|  | ||||
| @ -11,12 +11,15 @@ from erpnext.assets.doctype.asset.test_asset import ( | ||||
| 	create_asset_data, | ||||
| 	set_depreciation_settings_in_company, | ||||
| ) | ||||
| from erpnext.stock.doctype.item.test_item import create_item | ||||
| 
 | ||||
| 
 | ||||
| class TestAssetRepair(unittest.TestCase): | ||||
| 	def setUp(self): | ||||
| 	@classmethod | ||||
| 	def setUpClass(cls): | ||||
| 		set_depreciation_settings_in_company() | ||||
| 		create_asset_data() | ||||
| 		create_item("_Test Stock Item") | ||||
| 		frappe.db.sql("delete from `tabTax Rule`") | ||||
| 
 | ||||
| 	def test_update_status(self): | ||||
| @ -70,9 +73,28 @@ class TestAssetRepair(unittest.TestCase): | ||||
| 
 | ||||
| 		self.assertEqual(stock_entry.stock_entry_type, "Material Issue") | ||||
| 		self.assertEqual(stock_entry.items[0].s_warehouse, asset_repair.warehouse) | ||||
| 		self.assertEqual(stock_entry.items[0].item_code, asset_repair.stock_items[0].item) | ||||
| 		self.assertEqual(stock_entry.items[0].item_code, asset_repair.stock_items[0].item_code) | ||||
| 		self.assertEqual(stock_entry.items[0].qty, asset_repair.stock_items[0].consumed_quantity) | ||||
| 
 | ||||
| 	def test_serialized_item_consumption(self): | ||||
| 		from erpnext.stock.doctype.serial_no.serial_no import SerialNoRequiredError | ||||
| 		from erpnext.stock.doctype.stock_entry.test_stock_entry import make_serialized_item | ||||
| 
 | ||||
| 		stock_entry = make_serialized_item() | ||||
| 		serial_nos = stock_entry.get("items")[0].serial_no | ||||
| 		serial_no = serial_nos.split("\n")[0] | ||||
| 
 | ||||
| 		# should not raise any error | ||||
| 		create_asset_repair(stock_consumption = 1, item_code = stock_entry.get("items")[0].item_code, | ||||
| 			warehouse = "_Test Warehouse - _TC", serial_no = serial_no, submit = 1) | ||||
| 
 | ||||
| 		# should raise error | ||||
| 		asset_repair = create_asset_repair(stock_consumption = 1, warehouse = "_Test Warehouse - _TC", | ||||
| 			item_code = stock_entry.get("items")[0].item_code) | ||||
| 
 | ||||
| 		asset_repair.repair_status = "Completed" | ||||
| 		self.assertRaises(SerialNoRequiredError, asset_repair.submit) | ||||
| 
 | ||||
| 	def test_increase_in_asset_value_due_to_stock_consumption(self): | ||||
| 		asset = create_asset(calculate_depreciation = 1, submit=1) | ||||
| 		initial_asset_value = get_asset_value(asset) | ||||
| @ -137,11 +159,12 @@ def create_asset_repair(**args): | ||||
| 
 | ||||
| 	if args.stock_consumption: | ||||
| 		asset_repair.stock_consumption = 1 | ||||
| 		asset_repair.warehouse = create_warehouse("Test Warehouse", company = asset.company) | ||||
| 		asset_repair.warehouse = args.warehouse or create_warehouse("Test Warehouse", company = asset.company) | ||||
| 		asset_repair.append("stock_items", { | ||||
| 			"item": args.item or args.item_code or "_Test Item", | ||||
| 			"item_code": args.item_code or "_Test Stock Item", | ||||
| 			"valuation_rate": args.rate if args.get("rate") is not None else 100, | ||||
| 			"consumed_quantity": args.qty or 1 | ||||
| 			"consumed_quantity": args.qty or 1, | ||||
| 			"serial_no": args.serial_no | ||||
| 		}) | ||||
| 
 | ||||
| 	asset_repair.insert(ignore_if_duplicate=True) | ||||
| @ -158,7 +181,7 @@ def create_asset_repair(**args): | ||||
| 			}) | ||||
| 			stock_entry.append('items', { | ||||
| 				"t_warehouse": asset_repair.warehouse, | ||||
| 				"item_code": asset_repair.stock_items[0].item, | ||||
| 				"item_code": asset_repair.stock_items[0].item_code, | ||||
| 				"qty": asset_repair.stock_items[0].consumed_quantity | ||||
| 			}) | ||||
| 			stock_entry.submit() | ||||
|  | ||||
| @ -5,19 +5,13 @@ | ||||
|  "editable_grid": 1, | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "item", | ||||
|   "item_code", | ||||
|   "valuation_rate", | ||||
|   "consumed_quantity", | ||||
|   "total_value" | ||||
|   "total_value", | ||||
|   "serial_no" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "fieldname": "item", | ||||
|    "fieldtype": "Link", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Item", | ||||
|    "options": "Item" | ||||
|   }, | ||||
|   { | ||||
|    "fetch_from": "item.valuation_rate", | ||||
|    "fieldname": "valuation_rate", | ||||
| @ -38,12 +32,24 @@ | ||||
|    "in_list_view": 1, | ||||
|    "label": "Total Value", | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "serial_no", | ||||
|    "fieldtype": "Small Text", | ||||
|    "label": "Serial No" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "item_code", | ||||
|    "fieldtype": "Link", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Item", | ||||
|    "options": "Item" | ||||
|   } | ||||
|  ], | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-05-12 03:19:55.006300", | ||||
|  "modified": "2021-11-11 18:23:00.492483", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Assets", | ||||
|  "name": "Asset Repair Consumed Item", | ||||
|  | ||||
| @ -14,6 +14,14 @@ frappe.ui.form.on('Asset Value Adjustment', { | ||||
| 				} | ||||
| 			} | ||||
| 		}); | ||||
| 		frm.set_query('asset', function() { | ||||
| 			return { | ||||
| 				filters: { | ||||
| 					calculate_depreciation: 1, | ||||
| 					docstatus: 1 | ||||
| 				} | ||||
| 			}; | ||||
| 		}); | ||||
| 	}, | ||||
| 
 | ||||
| 	onload: function(frm) { | ||||
|  | ||||
| @ -10,7 +10,11 @@ from frappe.utils import cint, date_diff, flt, formatdate, getdate | ||||
| from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import ( | ||||
| 	get_checks_for_pl_and_bs_accounts, | ||||
| ) | ||||
| from erpnext.assets.doctype.asset.asset import get_depreciation_amount | ||||
| from erpnext.assets.doctype.asset.depreciation import get_depreciation_accounts | ||||
| from erpnext.regional.india.utils import ( | ||||
| 	get_depreciation_amount as get_depreciation_amount_for_india, | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| class AssetValueAdjustment(Document): | ||||
| @ -90,6 +94,7 @@ class AssetValueAdjustment(Document): | ||||
| 
 | ||||
| 	def reschedule_depreciations(self, asset_value): | ||||
| 		asset = frappe.get_doc('Asset', self.asset) | ||||
| 		country = frappe.get_value('Company', self.company, 'country') | ||||
| 
 | ||||
| 		for d in asset.finance_books: | ||||
| 			d.value_after_depreciation = asset_value | ||||
| @ -111,8 +116,10 @@ class AssetValueAdjustment(Document): | ||||
| 						depreciation_amount = days * rate_per_day | ||||
| 						from_date = data.schedule_date | ||||
| 					else: | ||||
| 						depreciation_amount = asset.get_depreciation_amount(value_after_depreciation, | ||||
| 							no_of_depreciations, d) | ||||
| 						if country == "India": | ||||
| 							depreciation_amount = get_depreciation_amount_for_india(asset, value_after_depreciation, d) | ||||
| 						else: | ||||
| 							depreciation_amount = get_depreciation_amount(asset, value_after_depreciation, d) | ||||
| 
 | ||||
| 					if depreciation_amount: | ||||
| 						value_after_depreciation -= flt(depreciation_amount) | ||||
|  | ||||
| @ -83,6 +83,12 @@ frappe.ui.form.on("Supplier", { | ||||
| 				frm.trigger("get_supplier_group_details"); | ||||
| 			}, __('Actions')); | ||||
| 
 | ||||
| 			if (cint(frappe.defaults.get_default("enable_common_party_accounting"))) { | ||||
| 				frm.add_custom_button(__('Link with Customer'), function () { | ||||
| 					frm.trigger('show_party_link_dialog'); | ||||
| 				}, __('Actions')); | ||||
| 			} | ||||
| 
 | ||||
| 			// indicators
 | ||||
| 			erpnext.utils.set_party_dashboard_indicators(frm); | ||||
| 		} | ||||
| @ -128,5 +134,42 @@ frappe.ui.form.on("Supplier", { | ||||
| 		else { | ||||
| 			frm.toggle_reqd("represents_company", false); | ||||
| 		} | ||||
| 	}, | ||||
| 	show_party_link_dialog: function(frm) { | ||||
| 		const dialog = new frappe.ui.Dialog({ | ||||
| 			title: __('Select a Customer'), | ||||
| 			fields: [{ | ||||
| 				fieldtype: 'Link', label: __('Customer'), | ||||
| 				options: 'Customer', fieldname: 'customer', reqd: 1 | ||||
| 			}], | ||||
| 			primary_action: function({ customer }) { | ||||
| 				frappe.call({ | ||||
| 					method: 'erpnext.accounts.doctype.party_link.party_link.create_party_link', | ||||
| 					args: { | ||||
| 						primary_role: 'Supplier', | ||||
| 						primary_party: frm.doc.name, | ||||
| 						secondary_party: customer | ||||
| 					}, | ||||
| 					freeze: true, | ||||
| 					callback: function() { | ||||
| 						dialog.hide(); | ||||
| 						frappe.msgprint({ | ||||
| 							message: __('Successfully linked to Customer'), | ||||
| 							alert: true | ||||
| 						}); | ||||
| 					}, | ||||
| 					error: function() { | ||||
| 						dialog.hide(); | ||||
| 						frappe.msgprint({ | ||||
| 							message: __('Linking to Customer Failed. Please try again.'), | ||||
| 							title: __('Linking Failed'), | ||||
| 							indicator: 'red' | ||||
| 						}); | ||||
| 					} | ||||
| 				}); | ||||
| 			}, | ||||
| 			primary_action_label: __('Create Link') | ||||
| 		}); | ||||
| 		dialog.show(); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| @ -11,7 +11,11 @@ from frappe.contacts.address_and_contact import ( | ||||
| ) | ||||
| from frappe.model.naming import set_name_by_naming_series, set_name_from_naming_options | ||||
| 
 | ||||
| from erpnext.accounts.party import get_dashboard_info, validate_party_accounts | ||||
| from erpnext.accounts.party import (  # noqa | ||||
| 	get_dashboard_info, | ||||
| 	get_timeline_data, | ||||
| 	validate_party_accounts, | ||||
| ) | ||||
| from erpnext.utilities.transaction_base import TransactionBase | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -1,184 +1,70 @@ | ||||
| { | ||||
|  "allow_copy": 0,  | ||||
|  "allow_guest_to_view": 0,  | ||||
|  "allow_import": 0,  | ||||
|  "allow_rename": 0,  | ||||
|  "actions": [], | ||||
|  "allow_rename": 1, | ||||
|  "autoname": "field:criteria_name", | ||||
|  "beta": 0,  | ||||
|  "creation": "2017-05-29 01:32:43.064891", | ||||
|  "custom": 0,  | ||||
|  "docstatus": 0,  | ||||
|  "doctype": "DocType", | ||||
|  "document_type": "",  | ||||
|  "editable_grid": 1, | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "criteria_name", | ||||
|   "max_score", | ||||
|   "formula", | ||||
|   "weight" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "criteria_name", | ||||
|    "fieldtype": "Data", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Criteria Name", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1, | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "unique": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "default": "100", | ||||
|    "fieldname": "max_score", | ||||
|    "fieldtype": "Float", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Max Score", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "formula", | ||||
|    "fieldtype": "Small Text", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 1, | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Criteria Formula", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "weight", | ||||
|    "fieldtype": "Percent", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Criteria Weight",  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "unique": 0 | ||||
|    "label": "Criteria Weight" | ||||
|   } | ||||
|  ], | ||||
|  "has_web_view": 0,  | ||||
|  "hide_heading": 0,  | ||||
|  "hide_toolbar": 0,  | ||||
|  "idx": 0,  | ||||
|  "image_view": 0,  | ||||
|  "in_create": 0,  | ||||
|  "is_submittable": 0,  | ||||
|  "issingle": 0,  | ||||
|  "istable": 0,  | ||||
|  "max_attachments": 0,  | ||||
|  "modified": "2019-01-22 10:47:00.000822",  | ||||
|  "links": [], | ||||
|  "modified": "2021-11-11 18:34:58.477648", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Buying", | ||||
|  "name": "Supplier Scorecard Criteria", | ||||
|  "name_case": "",  | ||||
|  "naming_rule": "By fieldname", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|    "amend": 0,  | ||||
|    "apply_user_permissions": 0,  | ||||
|    "cancel": 0,  | ||||
|    "create": 1, | ||||
|    "delete": 1, | ||||
|    "email": 1, | ||||
|    "export": 1, | ||||
|    "if_owner": 0,  | ||||
|    "import": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "print": 1, | ||||
|    "read": 1, | ||||
|    "report": 1, | ||||
|    "role": "System Manager", | ||||
|    "set_user_permissions": 0,  | ||||
|    "share": 1, | ||||
|    "submit": 0,  | ||||
|    "write": 1 | ||||
|   } | ||||
|  ], | ||||
|  "quick_entry": 1, | ||||
|  "read_only": 0,  | ||||
|  "read_only_onload": 0,  | ||||
|  "show_name_in_global_search": 0,  | ||||
|  "sort_field": "modified", | ||||
|  "sort_order": "DESC", | ||||
|  "track_changes": 1,  | ||||
|  "track_seen": 0 | ||||
|  "track_changes": 1 | ||||
| } | ||||
| @ -146,11 +146,6 @@ class AccountsController(TransactionBase): | ||||
| 		self.validate_party() | ||||
| 		self.validate_currency() | ||||
| 
 | ||||
| 		if self.doctype == 'Purchase Invoice': | ||||
| 			self.calculate_paid_amount() | ||||
| 			# apply tax withholding only if checked and applicable | ||||
| 			self.set_tax_withholding() | ||||
| 
 | ||||
| 		if self.doctype in ['Purchase Invoice', 'Sales Invoice']: | ||||
| 			pos_check_field = "is_pos" if self.doctype=="Sales Invoice" else "is_paid" | ||||
| 			if cint(self.allocate_advances_automatically) and not cint(self.get(pos_check_field)): | ||||
| @ -165,6 +160,11 @@ class AccountsController(TransactionBase): | ||||
| 
 | ||||
| 			self.set_inter_company_account() | ||||
| 
 | ||||
| 		if self.doctype == 'Purchase Invoice': | ||||
| 			self.calculate_paid_amount() | ||||
| 			# apply tax withholding only if checked and applicable | ||||
| 			self.set_tax_withholding() | ||||
| 
 | ||||
| 		validate_regional(self) | ||||
| 
 | ||||
| 		if self.doctype != 'Material Request': | ||||
| @ -251,7 +251,12 @@ class AccountsController(TransactionBase): | ||||
| 		from erpnext.controllers.taxes_and_totals import calculate_taxes_and_totals | ||||
| 		calculate_taxes_and_totals(self) | ||||
| 
 | ||||
| 		if self.doctype in ["Quotation", "Sales Order", "Delivery Note", "Sales Invoice"]: | ||||
| 		if self.doctype in ( | ||||
| 			'Sales Order', | ||||
| 			'Delivery Note', | ||||
| 			'Sales Invoice', | ||||
| 			'POS Invoice', | ||||
| 		): | ||||
| 			self.calculate_commission() | ||||
| 			self.calculate_contribution() | ||||
| 
 | ||||
| @ -525,7 +530,8 @@ class AccountsController(TransactionBase): | ||||
| 			'is_opening': self.get("is_opening") or "No", | ||||
| 			'party_type': None, | ||||
| 			'party': None, | ||||
| 			'project': self.get("project") | ||||
| 			'project': self.get("project"), | ||||
| 			'post_net_value': args.get('post_net_value') | ||||
| 		}) | ||||
| 
 | ||||
| 		accounting_dimensions = get_accounting_dimensions() | ||||
| @ -806,7 +812,6 @@ class AccountsController(TransactionBase): | ||||
| 		from erpnext.accounts.utils import unlink_ref_doc_from_payment_entries | ||||
| 
 | ||||
| 		if self.doctype in ["Sales Invoice", "Purchase Invoice"]: | ||||
| 			self.update_allocated_advance_taxes_on_cancel() | ||||
| 			if frappe.db.get_single_value('Accounts Settings', 'unlink_payment_on_cancellation_of_invoice'): | ||||
| 				unlink_ref_doc_from_payment_entries(self) | ||||
| 
 | ||||
| @ -854,29 +859,6 @@ class AccountsController(TransactionBase): | ||||
| 
 | ||||
| 		return tax_map | ||||
| 
 | ||||
| 	def update_allocated_advance_taxes_on_cancel(self): | ||||
| 		if self.get('advances'): | ||||
| 			tax_accounts = [d.account_head for d in self.get('taxes')] | ||||
| 			allocated_tax_map = frappe._dict(frappe.get_all('GL Entry', fields=['account', 'sum(credit - debit)'], | ||||
| 				filters={'voucher_no': self.name, 'account': ('in', tax_accounts)}, | ||||
| 				group_by='account', as_list=1)) | ||||
| 
 | ||||
| 			tax_map = self.get_tax_map() | ||||
| 
 | ||||
| 			for pe in self.get('advances'): | ||||
| 				if pe.reference_type == 'Payment Entry': | ||||
| 					pe = frappe.get_doc('Payment Entry', pe.reference_name) | ||||
| 					for tax in pe.get('taxes'): | ||||
| 						allocated_amount = tax_map.get(tax.account_head) - allocated_tax_map.get(tax.account_head) | ||||
| 						if allocated_amount > tax.tax_amount: | ||||
| 							allocated_amount = tax.tax_amount | ||||
| 
 | ||||
| 						if allocated_amount: | ||||
| 							frappe.db.set_value('Advance Taxes and Charges', tax.name, 'allocated_amount', | ||||
| 								tax.allocated_amount - allocated_amount) | ||||
| 							tax_map[tax.account_head] -= allocated_amount | ||||
| 							allocated_tax_map[tax.account_head] -= allocated_amount | ||||
| 
 | ||||
| 	def get_amount_and_base_amount(self, item, enable_discount_accounting): | ||||
| 		amount = item.net_amount | ||||
| 		base_amount = item.base_net_amount | ||||
| @ -960,58 +942,10 @@ class AccountsController(TransactionBase): | ||||
| 					}, item=self) | ||||
| 				) | ||||
| 
 | ||||
| 	def allocate_advance_taxes(self, gl_entries): | ||||
| 		tax_map = self.get_tax_map() | ||||
| 		for pe in self.get("advances"): | ||||
| 			if pe.reference_type == "Payment Entry" and \ | ||||
| 				frappe.db.get_value('Payment Entry', pe.reference_name, 'advance_tax_account'): | ||||
| 				pe = frappe.get_doc("Payment Entry", pe.reference_name) | ||||
| 				for tax in pe.get("taxes"): | ||||
| 					account_currency = get_account_currency(tax.account_head) | ||||
| 
 | ||||
| 					if self.doctype == "Purchase Invoice": | ||||
| 						dr_or_cr = "debit" if tax.add_deduct_tax == "Add" else "credit" | ||||
| 						rev_dr_cr = "credit" if tax.add_deduct_tax == "Add" else "debit" | ||||
| 					else: | ||||
| 						dr_or_cr = "credit" if tax.add_deduct_tax == "Add" else "debit" | ||||
| 						rev_dr_cr = "debit" if tax.add_deduct_tax == "Add" else "credit" | ||||
| 
 | ||||
| 					party = self.supplier if self.doctype == "Purchase Invoice" else self.customer | ||||
| 					unallocated_amount = tax.tax_amount - tax.allocated_amount | ||||
| 					if tax_map.get(tax.account_head): | ||||
| 						amount = tax_map.get(tax.account_head) | ||||
| 						if amount < unallocated_amount: | ||||
| 							unallocated_amount = amount | ||||
| 
 | ||||
| 						gl_entries.append( | ||||
| 							self.get_gl_dict({ | ||||
| 								"account": tax.account_head, | ||||
| 								"against": party, | ||||
| 								dr_or_cr: unallocated_amount, | ||||
| 								dr_or_cr + "_in_account_currency": unallocated_amount | ||||
| 								if account_currency==self.company_currency | ||||
| 								else unallocated_amount, | ||||
| 								"cost_center": tax.cost_center | ||||
| 							}, account_currency, item=tax)) | ||||
| 
 | ||||
| 						gl_entries.append( | ||||
| 							self.get_gl_dict({ | ||||
| 								"account": pe.advance_tax_account, | ||||
| 								"against": party, | ||||
| 								rev_dr_cr: unallocated_amount, | ||||
| 								rev_dr_cr + "_in_account_currency": unallocated_amount | ||||
| 								if account_currency==self.company_currency | ||||
| 								else unallocated_amount, | ||||
| 								"cost_center": tax.cost_center | ||||
| 							}, account_currency, item=tax)) | ||||
| 
 | ||||
| 						frappe.db.set_value("Advance Taxes and Charges", tax.name, "allocated_amount", | ||||
| 							tax.allocated_amount + unallocated_amount) | ||||
| 
 | ||||
| 						tax_map[tax.account_head] -= unallocated_amount | ||||
| 
 | ||||
| 	def validate_multiple_billing(self, ref_dt, item_ref_dn, based_on, parentfield): | ||||
| 		from erpnext.controllers.status_updater import get_allowance_for | ||||
| 
 | ||||
| 		item_allowance = {} | ||||
| 		global_qty_allowance, global_amount_allowance = None, None | ||||
| 
 | ||||
| @ -1032,12 +966,7 @@ class AccountsController(TransactionBase): | ||||
| 						.format(item.item_code, ref_dt), title=_("Warning"), indicator="orange") | ||||
| 				continue | ||||
| 
 | ||||
| 			already_billed = frappe.db.sql(""" | ||||
| 				select sum(%s) | ||||
| 				from `tab%s` | ||||
| 				where %s=%s and docstatus=1 and parent != %s | ||||
| 			""" % (based_on, self.doctype + " Item", item_ref_dn, '%s', '%s'), | ||||
| 				(item.get(item_ref_dn), self.name))[0][0] | ||||
| 			already_billed = self.get_billed_amount_for_item(item, item_ref_dn, based_on) | ||||
| 
 | ||||
| 			total_billed_amt = flt(flt(already_billed) + flt(item.get(based_on)), | ||||
| 				self.precision(based_on, item)) | ||||
| @ -1065,6 +994,43 @@ class AccountsController(TransactionBase): | ||||
| 			frappe.msgprint(_("Overbilling of {} ignored because you have {} role.") | ||||
| 					.format(total_overbilled_amt, role_allowed_to_over_bill), indicator="orange", alert=True) | ||||
| 
 | ||||
| 	def get_billed_amount_for_item(self, item, item_ref_dn, based_on): | ||||
| 		''' | ||||
| 			Returns Sum of Amount of | ||||
| 			Sales/Purchase Invoice Items | ||||
| 			that are linked to `item_ref_dn` (`dn_detail` / `pr_detail`) | ||||
| 			that are submitted OR not submitted but are under current invoice | ||||
| 		''' | ||||
| 
 | ||||
| 		from frappe.query_builder import Criterion | ||||
| 		from frappe.query_builder.functions import Sum | ||||
| 
 | ||||
| 		item_doctype = frappe.qb.DocType(item.doctype) | ||||
| 		based_on_field = frappe.qb.Field(based_on) | ||||
| 		join_field = frappe.qb.Field(item_ref_dn) | ||||
| 
 | ||||
| 		result = ( | ||||
| 			frappe.qb.from_(item_doctype) | ||||
| 			.select(Sum(based_on_field)) | ||||
| 			.where( | ||||
| 				join_field == item.get(item_ref_dn) | ||||
| 			).where( | ||||
| 				Criterion.any([ # select all items from other invoices OR current invoices | ||||
| 					Criterion.all([ # for selecting items from other invoices | ||||
| 						item_doctype.docstatus == 1, | ||||
| 						item_doctype.parent != self.name | ||||
| 					]), | ||||
| 					Criterion.all([ # for selecting items from current invoice, that are linked to same reference | ||||
| 						item_doctype.docstatus == 0, | ||||
| 						item_doctype.parent == self.name, | ||||
| 						item_doctype.name != item.name | ||||
| 					]) | ||||
| 				]) | ||||
| 			) | ||||
| 		).run() | ||||
| 
 | ||||
| 		return result[0][0] if result else 0 | ||||
| 
 | ||||
| 	def throw_overbill_exception(self, item, max_allowed_amt): | ||||
| 		frappe.throw(_("Cannot overbill for Item {0} in row {1} more than {2}. To allow over-billing, please set allowance in Accounts Settings") | ||||
| 			.format(item.item_code, item.idx, max_allowed_amt)) | ||||
| @ -1403,8 +1369,8 @@ class AccountsController(TransactionBase): | ||||
| 					grand_total -= self.get("total_advance") | ||||
| 					base_grand_total = flt(grand_total * self.get("conversion_rate"), self.precision("base_grand_total")) | ||||
| 
 | ||||
| 			if flt(total, self.precision("grand_total")) != flt(grand_total, self.precision("grand_total")) or \ | ||||
| 				flt(base_total, self.precision("base_grand_total")) != flt(base_grand_total, self.precision("base_grand_total")): | ||||
| 			if flt(total, self.precision("grand_total")) - flt(grand_total, self.precision("grand_total")) > 0.1 or \ | ||||
| 				flt(base_total, self.precision("base_grand_total")) - flt(base_grand_total, self.precision("base_grand_total")) > 0.1: | ||||
| 				frappe.throw(_("Total Payment Amount in Payment Schedule must be equal to Grand / Rounded Total")) | ||||
| 
 | ||||
| 	def is_rounded_total_disabled(self): | ||||
|  | ||||
| @ -3,7 +3,7 @@ | ||||
| 
 | ||||
| 
 | ||||
| import frappe | ||||
| from frappe import _, msgprint | ||||
| from frappe import ValidationError, _, msgprint | ||||
| from frappe.contacts.doctype.address.address import get_address_display | ||||
| from frappe.utils import cint, cstr, flt, getdate | ||||
| 
 | ||||
| @ -17,6 +17,9 @@ from erpnext.stock.get_item_details import get_conversion_factor | ||||
| from erpnext.stock.utils import get_incoming_rate | ||||
| 
 | ||||
| 
 | ||||
| class QtyMismatchError(ValidationError): | ||||
| 	pass | ||||
| 
 | ||||
| class BuyingController(StockController, Subcontracting): | ||||
| 
 | ||||
| 	def get_feed(self): | ||||
| @ -360,19 +363,15 @@ class BuyingController(StockController, Subcontracting): | ||||
| 	def validate_accepted_rejected_qty(self): | ||||
| 		for d in self.get("items"): | ||||
| 			self.validate_negative_quantity(d, ["received_qty","qty", "rejected_qty"]) | ||||
| 			if not flt(d.received_qty) and flt(d.qty): | ||||
| 				d.received_qty = flt(d.qty) - flt(d.rejected_qty) | ||||
| 
 | ||||
| 			elif not flt(d.qty) and flt(d.rejected_qty): | ||||
| 				d.qty = flt(d.received_qty) - flt(d.rejected_qty) | ||||
| 			if not flt(d.received_qty) and (flt(d.qty) or flt(d.rejected_qty)): | ||||
| 				d.received_qty = flt(d.qty) + flt(d.rejected_qty) | ||||
| 
 | ||||
| 			elif not flt(d.rejected_qty): | ||||
| 				d.rejected_qty = flt(d.received_qty) -  flt(d.qty) | ||||
| 
 | ||||
| 			val  = flt(d.qty) + flt(d.rejected_qty) | ||||
| 			# Check Received Qty = Accepted Qty + Rejected Qty | ||||
| 			val = flt(d.qty) + flt(d.rejected_qty) | ||||
| 			if (flt(val, d.precision("received_qty")) != flt(d.received_qty, d.precision("received_qty"))): | ||||
| 				frappe.throw(_("Accepted + Rejected Qty must be equal to Received quantity for Item {0}").format(d.item_code)) | ||||
| 				message = _("Row #{0}: Received Qty must be equal to Accepted + Rejected Qty for Item {1}").format(d.idx, d.item_code) | ||||
| 				frappe.throw(msg=message, title=_("Mismatch"), exc=QtyMismatchError) | ||||
| 
 | ||||
| 	def validate_negative_quantity(self, item_row, field_list): | ||||
| 		if self.is_return: | ||||
|  | ||||
| @ -539,6 +539,10 @@ def get_filtered_dimensions(doctype, txt, searchfield, start, page_len, filters) | ||||
| 	dimension_filters = get_dimension_filter_map() | ||||
| 	dimension_filters = dimension_filters.get((filters.get('dimension'),filters.get('account'))) | ||||
| 	query_filters = [] | ||||
| 	or_filters = [] | ||||
| 	fields = ['name'] | ||||
| 
 | ||||
| 	searchfields = frappe.get_meta(doctype).get_search_fields() | ||||
| 
 | ||||
| 	meta = frappe.get_meta(doctype) | ||||
| 	if meta.is_tree: | ||||
| @ -550,8 +554,9 @@ def get_filtered_dimensions(doctype, txt, searchfield, start, page_len, filters) | ||||
| 	if meta.has_field('company'): | ||||
| 		query_filters.append(['company', '=', filters.get('company')]) | ||||
| 
 | ||||
| 	if txt: | ||||
| 		query_filters.append([searchfield, 'LIKE', "%%%s%%" % txt]) | ||||
| 	for field in searchfields: | ||||
| 		or_filters.append([field, 'LIKE', "%%%s%%" % txt]) | ||||
| 		fields.append(field) | ||||
| 
 | ||||
| 	if dimension_filters: | ||||
| 		if dimension_filters['allow_or_restrict'] == 'Allow': | ||||
| @ -566,10 +571,9 @@ def get_filtered_dimensions(doctype, txt, searchfield, start, page_len, filters) | ||||
| 
 | ||||
| 		query_filters.append(['name', query_selector, dimensions]) | ||||
| 
 | ||||
| 	output = frappe.get_list(doctype, filters=query_filters) | ||||
| 	result = [d.name for d in output] | ||||
| 	output = frappe.get_list(doctype, fields=fields, filters=query_filters, or_filters=or_filters, as_list=1) | ||||
| 
 | ||||
| 	return [(d,) for d in set(result)] | ||||
| 	return [tuple(d) for d in set(output)] | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| @frappe.validate_and_sanitize_search_inputs | ||||
|  | ||||
| @ -120,13 +120,27 @@ class SellingController(StockController): | ||||
| 			self.in_words = money_in_words(amount, self.currency) | ||||
| 
 | ||||
| 	def calculate_commission(self): | ||||
| 		if self.meta.get_field("commission_rate"): | ||||
| 			self.round_floats_in(self, ["base_net_total", "commission_rate"]) | ||||
| 			if self.commission_rate > 100.0: | ||||
| 				throw(_("Commission rate cannot be greater than 100")) | ||||
| 		if not self.meta.get_field("commission_rate"): | ||||
| 			return | ||||
| 
 | ||||
| 			self.total_commission = flt(self.base_net_total * self.commission_rate / 100.0, | ||||
| 				self.precision("total_commission")) | ||||
| 		self.round_floats_in( | ||||
| 			self, ("amount_eligible_for_commission", "commission_rate") | ||||
| 		) | ||||
| 
 | ||||
| 		if not (0 <= self.commission_rate <= 100.0): | ||||
| 			throw("{} {}".format( | ||||
| 				_(self.meta.get_label("commission_rate")), | ||||
| 				_("must be between 0 and 100"), | ||||
| 			)) | ||||
| 
 | ||||
| 		self.amount_eligible_for_commission = sum( | ||||
| 			item.base_net_amount for item in self.items if item.grant_commission | ||||
| 		) | ||||
| 
 | ||||
| 		self.total_commission = flt( | ||||
| 			self.amount_eligible_for_commission * self.commission_rate / 100.0, | ||||
| 			self.precision("total_commission") | ||||
| 		) | ||||
| 
 | ||||
| 	def calculate_contribution(self): | ||||
| 		if not self.meta.get_field("sales_team"): | ||||
| @ -138,7 +152,7 @@ class SellingController(StockController): | ||||
| 			self.round_floats_in(sales_person) | ||||
| 
 | ||||
| 			sales_person.allocated_amount = flt( | ||||
| 				self.base_net_total * sales_person.allocated_percentage / 100.0, | ||||
| 				self.amount_eligible_for_commission * sales_person.allocated_percentage / 100.0, | ||||
| 				self.precision("allocated_amount", sales_person)) | ||||
| 
 | ||||
| 			if sales_person.commission_rate: | ||||
|  | ||||
| @ -17,7 +17,7 @@ from erpnext.accounts.general_ledger import ( | ||||
| from erpnext.accounts.utils import get_fiscal_year | ||||
| from erpnext.controllers.accounts_controller import AccountsController | ||||
| from erpnext.stock import get_warehouse_account_map | ||||
| from erpnext.stock.stock_ledger import get_valuation_rate | ||||
| from erpnext.stock.stock_ledger import get_items_to_be_repost, get_valuation_rate | ||||
| 
 | ||||
| 
 | ||||
| class QualityInspectionRequiredError(frappe.ValidationError): pass | ||||
| @ -134,7 +134,7 @@ class StockController(AccountsController): | ||||
| 							"against": expense_account, | ||||
| 							"cost_center": item_row.cost_center, | ||||
| 							"project": item_row.project or self.get('project'), | ||||
| 							"remarks": self.get("remarks") or "Accounting Entry for Stock", | ||||
| 							"remarks": self.get("remarks") or _("Accounting Entry for Stock"), | ||||
| 							"debit": flt(sle.stock_value_difference, precision), | ||||
| 							"is_opening": item_row.get("is_opening") or self.get("is_opening") or "No", | ||||
| 						}, warehouse_account[sle.warehouse]["account_currency"], item=item_row)) | ||||
| @ -143,7 +143,7 @@ class StockController(AccountsController): | ||||
| 							"account": expense_account, | ||||
| 							"against": warehouse_account[sle.warehouse]["account"], | ||||
| 							"cost_center": item_row.cost_center, | ||||
| 							"remarks": self.get("remarks") or "Accounting Entry for Stock", | ||||
| 							"remarks": self.get("remarks") or _("Accounting Entry for Stock"), | ||||
| 							"credit": flt(sle.stock_value_difference, precision), | ||||
| 							"project": item_row.get("project") or self.get("project"), | ||||
| 							"is_opening": item_row.get("is_opening") or self.get("is_opening") or "No" | ||||
| @ -544,8 +544,13 @@ class StockController(AccountsController): | ||||
| 			"company": self.company | ||||
| 		}) | ||||
| 		if future_sle_exists(args): | ||||
| 			item_based_reposting =  cint(frappe.db.get_single_value("Stock Reposting Settings", "item_based_reposting")) | ||||
| 			if item_based_reposting: | ||||
| 				create_item_wise_repost_entries(voucher_type=self.doctype, voucher_no=self.name) | ||||
| 			else: | ||||
| 				create_repost_item_valuation_entry(args) | ||||
| 
 | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| def make_quality_inspections(doctype, docname, items): | ||||
| 	if isinstance(items, str): | ||||
| @ -676,5 +681,38 @@ def create_repost_item_valuation_entry(args): | ||||
| 	repost_entry.company = args.company | ||||
| 	repost_entry.allow_zero_rate = args.allow_zero_rate | ||||
| 	repost_entry.flags.ignore_links = True | ||||
| 	repost_entry.flags.ignore_permissions = True | ||||
| 	repost_entry.save() | ||||
| 	repost_entry.submit() | ||||
| 
 | ||||
| 
 | ||||
| def create_item_wise_repost_entries(voucher_type, voucher_no, allow_zero_rate=False): | ||||
| 	"""Using a voucher create repost item valuation records for all item-warehouse pairs.""" | ||||
| 
 | ||||
| 	stock_ledger_entries = get_items_to_be_repost(voucher_type, voucher_no) | ||||
| 
 | ||||
| 	distinct_item_warehouses = set() | ||||
| 	repost_entries = [] | ||||
| 
 | ||||
| 	for sle in stock_ledger_entries: | ||||
| 		item_wh = (sle.item_code, sle.warehouse) | ||||
| 		if item_wh in distinct_item_warehouses: | ||||
| 			continue | ||||
| 		distinct_item_warehouses.add(item_wh) | ||||
| 
 | ||||
| 		repost_entry = frappe.new_doc("Repost Item Valuation") | ||||
| 		repost_entry.based_on = "Item and Warehouse" | ||||
| 		repost_entry.voucher_type = voucher_type | ||||
| 		repost_entry.voucher_no = voucher_no | ||||
| 
 | ||||
| 		repost_entry.item_code = sle.item_code | ||||
| 		repost_entry.warehouse = sle.warehouse | ||||
| 		repost_entry.posting_date = sle.posting_date | ||||
| 		repost_entry.posting_time = sle.posting_time | ||||
| 		repost_entry.allow_zero_rate = allow_zero_rate | ||||
| 		repost_entry.flags.ignore_links = True | ||||
| 		repost_entry.flags.ignore_permissions = True | ||||
| 		repost_entry.submit() | ||||
| 		repost_entries.append(repost_entry) | ||||
| 
 | ||||
| 	return repost_entries | ||||
|  | ||||
| @ -50,6 +50,7 @@ class calculate_taxes_and_totals(object): | ||||
| 		self.initialize_taxes() | ||||
| 		self.determine_exclusive_rate() | ||||
| 		self.calculate_net_total() | ||||
| 		self.calculate_shipping_charges() | ||||
| 		self.calculate_taxes() | ||||
| 		self.manipulate_grand_total_for_inclusive_tax() | ||||
| 		self.calculate_totals() | ||||
| @ -258,6 +259,11 @@ class calculate_taxes_and_totals(object): | ||||
| 
 | ||||
| 		self.doc.round_floats_in(self.doc, ["total", "base_total", "net_total", "base_net_total"]) | ||||
| 
 | ||||
| 	def calculate_shipping_charges(self): | ||||
| 		if hasattr(self.doc, "shipping_rule") and self.doc.shipping_rule: | ||||
| 			shipping_rule = frappe.get_doc("Shipping Rule", self.doc.shipping_rule) | ||||
| 			shipping_rule.apply(self.doc) | ||||
| 
 | ||||
| 	def calculate_taxes(self): | ||||
| 		if not self.doc.get('is_consolidated'): | ||||
| 			self.doc.rounding_adjustment = 0 | ||||
|  | ||||
							
								
								
									
										0
									
								
								erpnext/crm/doctype/competitor/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								erpnext/crm/doctype/competitor/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										8
									
								
								erpnext/crm/doctype/competitor/competitor.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								erpnext/crm/doctype/competitor/competitor.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | ||||
| // Copyright (c) 2021, Frappe Technologies Pvt. Ltd. and contributors
 | ||||
| // For license information, please see license.txt
 | ||||
| 
 | ||||
| frappe.ui.form.on('Competitor', { | ||||
| 	// refresh: function(frm) {
 | ||||
| 
 | ||||
| 	// }
 | ||||
| }); | ||||
							
								
								
									
										68
									
								
								erpnext/crm/doctype/competitor/competitor.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								erpnext/crm/doctype/competitor/competitor.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,68 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "allow_rename": 1, | ||||
|  "autoname": "field:competitor_name", | ||||
|  "creation": "2021-10-21 10:28:52.071316", | ||||
|  "doctype": "DocType", | ||||
|  "editable_grid": 1, | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "competitor_name", | ||||
|   "website" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "fieldname": "competitor_name", | ||||
|    "fieldtype": "Data", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Competitor Name", | ||||
|    "reqd": 1, | ||||
|    "unique": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_in_quick_entry": 1, | ||||
|    "fieldname": "website", | ||||
|    "fieldtype": "Data", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Website", | ||||
|    "options": "URL" | ||||
|   } | ||||
|  ], | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-10-21 12:43:59.106807", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "CRM", | ||||
|  "name": "Competitor", | ||||
|  "naming_rule": "By fieldname", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|    "create": 1, | ||||
|    "delete": 1, | ||||
|    "email": 1, | ||||
|    "export": 1, | ||||
|    "print": 1, | ||||
|    "read": 1, | ||||
|    "report": 1, | ||||
|    "role": "System Manager", | ||||
|    "share": 1, | ||||
|    "write": 1 | ||||
|   }, | ||||
|   { | ||||
|    "create": 1, | ||||
|    "email": 1, | ||||
|    "export": 1, | ||||
|    "print": 1, | ||||
|    "read": 1, | ||||
|    "report": 1, | ||||
|    "role": "Sales User", | ||||
|    "share": 1, | ||||
|    "write": 1 | ||||
|   } | ||||
|  ], | ||||
|  "quick_entry": 1, | ||||
|  "sort_field": "modified", | ||||
|  "sort_order": "DESC", | ||||
|  "track_changes": 1 | ||||
| } | ||||
							
								
								
									
										9
									
								
								erpnext/crm/doctype/competitor/competitor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								erpnext/crm/doctype/competitor/competitor.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | ||||
| # Copyright (c) 2021, Frappe Technologies Pvt. Ltd. and contributors | ||||
| # For license information, please see license.txt | ||||
| 
 | ||||
| # import frappe | ||||
| from frappe.model.document import Document | ||||
| 
 | ||||
| 
 | ||||
| class Competitor(Document): | ||||
| 	pass | ||||
							
								
								
									
										9
									
								
								erpnext/crm/doctype/competitor/test_competitor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								erpnext/crm/doctype/competitor/test_competitor.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | ||||
| # Copyright (c) 2021, Frappe Technologies Pvt. Ltd. and Contributors | ||||
| # See license.txt | ||||
| 
 | ||||
| # import frappe | ||||
| import unittest | ||||
| 
 | ||||
| 
 | ||||
| class TestCompetitor(unittest.TestCase): | ||||
| 	pass | ||||
							
								
								
									
										0
									
								
								erpnext/crm/doctype/competitor_detail/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								erpnext/crm/doctype/competitor_detail/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										33
									
								
								erpnext/crm/doctype/competitor_detail/competitor_detail.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								erpnext/crm/doctype/competitor_detail/competitor_detail.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "allow_rename": 1, | ||||
|  "creation": "2021-10-21 10:34:58.841689", | ||||
|  "doctype": "DocType", | ||||
|  "editable_grid": 1, | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "competitor" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "fieldname": "competitor", | ||||
|    "fieldtype": "Link", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Competitor", | ||||
|    "options": "Competitor", | ||||
|    "reqd": 1 | ||||
|   } | ||||
|  ], | ||||
|  "index_web_pages_for_search": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-10-21 10:34:58.841689", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "CRM", | ||||
|  "name": "Competitor Detail", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [], | ||||
|  "sort_field": "modified", | ||||
|  "sort_order": "DESC", | ||||
|  "track_changes": 1 | ||||
| } | ||||
| @ -0,0 +1,9 @@ | ||||
| # Copyright (c) 2021, Frappe Technologies Pvt. Ltd. and contributors | ||||
| # For license information, please see license.txt | ||||
| 
 | ||||
| # import frappe | ||||
| from frappe.model.document import Document | ||||
| 
 | ||||
| 
 | ||||
| class CompetitorDetail(Document): | ||||
| 	pass | ||||
| @ -23,7 +23,6 @@ | ||||
|   "status", | ||||
|   "converted_by", | ||||
|   "sales_stage", | ||||
|   "order_lost_reason", | ||||
|   "first_response_time", | ||||
|   "expected_closing", | ||||
|   "next_contact", | ||||
| @ -64,7 +63,11 @@ | ||||
|   "transaction_date", | ||||
|   "language", | ||||
|   "amended_from", | ||||
|   "lost_reasons" | ||||
|   "lost_detail_section", | ||||
|   "lost_reasons", | ||||
|   "order_lost_reason", | ||||
|   "column_break_56", | ||||
|   "competitors" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
| @ -154,10 +157,9 @@ | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "depends_on": "eval:doc.status===\"Lost\"", | ||||
|    "fieldname": "order_lost_reason", | ||||
|    "fieldtype": "Small Text", | ||||
|    "label": "Lost Reason", | ||||
|    "label": "Detailed Reason", | ||||
|    "no_copy": 1, | ||||
|    "read_only": 1 | ||||
|   }, | ||||
| @ -409,6 +411,7 @@ | ||||
|    "width": "150px" | ||||
|   }, | ||||
|   { | ||||
|    "depends_on": "eval:doc.status===\"Lost\"", | ||||
|    "fieldname": "lost_reasons", | ||||
|    "fieldtype": "Table MultiSelect", | ||||
|    "label": "Lost Reasons", | ||||
| @ -486,15 +489,33 @@ | ||||
|    "label": "Grand Total", | ||||
|    "options": "currency", | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "lost_detail_section", | ||||
|    "fieldtype": "Section Break", | ||||
|    "label": "Lost Reasons" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "column_break_56", | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "fieldname": "competitors", | ||||
|    "fieldtype": "Table MultiSelect", | ||||
|    "label": "Competitors", | ||||
|    "options": "Competitor Detail", | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "icon": "fa fa-info-sign", | ||||
|  "idx": 195, | ||||
|  "links": [], | ||||
|  "modified": "2021-09-06 10:02:18.609136", | ||||
|  "migration_hash": "d87c646ea2579b6900197fd41e6c5c5a", | ||||
|  "modified": "2021-10-21 11:04:30.151379", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "CRM", | ||||
|  "name": "Opportunity", | ||||
|  "naming_rule": "By \"Naming Series\" field", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|  | ||||
| @ -116,16 +116,20 @@ class Opportunity(TransactionBase): | ||||
| 			self.party_name = lead_name | ||||
| 
 | ||||
| 	@frappe.whitelist() | ||||
| 	def declare_enquiry_lost(self, lost_reasons_list, detailed_reason=None): | ||||
| 	def declare_enquiry_lost(self, lost_reasons_list, competitors, detailed_reason=None): | ||||
| 		if not self.has_active_quotation(): | ||||
| 			frappe.db.set(self, 'status', 'Lost') | ||||
| 			self.status = 'Lost' | ||||
| 			self.lost_reasons = self.competitors = [] | ||||
| 
 | ||||
| 			if detailed_reason: | ||||
| 				frappe.db.set(self, 'order_lost_reason', detailed_reason) | ||||
| 				self.order_lost_reason = detailed_reason | ||||
| 
 | ||||
| 			for reason in lost_reasons_list: | ||||
| 				self.append('lost_reasons', reason) | ||||
| 
 | ||||
| 			for competitor in competitors: | ||||
| 				self.append('competitors', competitor) | ||||
| 
 | ||||
| 			self.save() | ||||
| 
 | ||||
| 		else: | ||||
|  | ||||
| @ -125,7 +125,7 @@ def get_student_guardians(student): | ||||
| 
 | ||||
| 	:param student: Student. | ||||
| 	""" | ||||
| 	guardians = frappe.get_list("Student Guardian", fields=["guardian"] , | ||||
| 	guardians = frappe.get_all("Student Guardian", fields=["guardian"] , | ||||
| 		filters={"parent": student}) | ||||
| 	return guardians | ||||
| 
 | ||||
| @ -137,10 +137,10 @@ def get_student_group_students(student_group, include_inactive=0): | ||||
| 	:param student_group: Student Group. | ||||
| 	""" | ||||
| 	if include_inactive: | ||||
| 		students = frappe.get_list("Student Group Student", fields=["student", "student_name"] , | ||||
| 		students = frappe.get_all("Student Group Student", fields=["student", "student_name"] , | ||||
| 			filters={"parent": student_group}, order_by= "group_roll_number") | ||||
| 	else: | ||||
| 		students = frappe.get_list("Student Group Student", fields=["student", "student_name"] , | ||||
| 		students = frappe.get_all("Student Group Student", fields=["student", "student_name"] , | ||||
| 			filters={"parent": student_group, "active": 1}, order_by= "group_roll_number") | ||||
| 	return students | ||||
| 
 | ||||
| @ -164,7 +164,7 @@ def get_fee_components(fee_structure): | ||||
| 	:param fee_structure: Fee Structure. | ||||
| 	""" | ||||
| 	if fee_structure: | ||||
| 		fs = frappe.get_list("Fee Component", fields=["fees_category", "description", "amount"] , filters={"parent": fee_structure}, order_by= "idx") | ||||
| 		fs = frappe.get_all("Fee Component", fields=["fees_category", "description", "amount"] , filters={"parent": fee_structure}, order_by= "idx") | ||||
| 		return fs | ||||
| 
 | ||||
| 
 | ||||
| @ -175,7 +175,7 @@ def get_fee_schedule(program, student_category=None): | ||||
| 	:param program: Program. | ||||
| 	:param student_category: Student Category | ||||
| 	""" | ||||
| 	fs = frappe.get_list("Program Fee", fields=["academic_term", "fee_structure", "due_date", "amount"] , | ||||
| 	fs = frappe.get_all("Program Fee", fields=["academic_term", "fee_structure", "due_date", "amount"] , | ||||
| 		filters={"parent": program, "student_category": student_category }, order_by= "idx") | ||||
| 	return fs | ||||
| 
 | ||||
| @ -220,7 +220,7 @@ def get_assessment_criteria(course): | ||||
| 
 | ||||
| 	:param Course: Course | ||||
| 	""" | ||||
| 	return frappe.get_list("Course Assessment Criteria", \ | ||||
| 	return frappe.get_all("Course Assessment Criteria", | ||||
| 		fields=["assessment_criteria", "weightage"], filters={"parent": course}, order_by= "idx") | ||||
| 
 | ||||
| 
 | ||||
| @ -253,7 +253,7 @@ def get_assessment_details(assessment_plan): | ||||
| 
 | ||||
| 	:param Assessment Plan: Assessment Plan | ||||
| 	""" | ||||
| 	return frappe.get_list("Assessment Plan Criteria", \ | ||||
| 	return frappe.get_all("Assessment Plan Criteria", | ||||
| 		fields=["assessment_criteria", "maximum_score", "docstatus"], filters={"parent": assessment_plan}, order_by= "idx") | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -1,520 +1,143 @@ | ||||
| { | ||||
|  "allow_copy": 0, | ||||
|  "allow_guest_to_view": 0, | ||||
|  "actions": [], | ||||
|  "allow_import": 1, | ||||
|  "allow_rename": 0, | ||||
|  "autoname": "naming_series:", | ||||
|  "beta": 0, | ||||
|  "creation": "2015-09-09 16:34:04.960369", | ||||
|  "custom": 0, | ||||
|  "docstatus": 0, | ||||
|  "doctype": "DocType", | ||||
|  "document_type": "Document", | ||||
|  "editable_grid": 0, | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "student_group", | ||||
|   "instructor", | ||||
|   "instructor_name", | ||||
|   "column_break_2", | ||||
|   "naming_series", | ||||
|   "course", | ||||
|   "color", | ||||
|   "section_break_6", | ||||
|   "schedule_date", | ||||
|   "room", | ||||
|   "column_break_9", | ||||
|   "from_time", | ||||
|   "to_time", | ||||
|   "title" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "student_group", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 1, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 1, | ||||
|    "label": "Student Group", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "options": "Student Group", | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 1, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "instructor", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 1, | ||||
|    "label": "Instructor", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "options": "Instructor", | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 1, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fetch_from": "instructor.Instructor_name", | ||||
|    "fetch_from": "instructor.instructor_name", | ||||
|    "fieldname": "instructor_name", | ||||
|    "fieldtype": "Read Only", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 1, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "Instructor Name", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "options": "", | ||||
|    "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 | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "column_break_2", | ||||
|    "fieldtype": "Column Break", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "default": "", | ||||
|    "fieldname": "naming_series", | ||||
|    "fieldtype": "Select", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "Naming Series", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "options": "EDU-CSH-.YYYY.-", | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 1, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "set_only_once": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "course", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 1, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "Course", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "options": "Course", | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 1, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "color", | ||||
|    "fieldtype": "Color", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "Color", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 1, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "print_hide": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "section_break_6", | ||||
|    "fieldtype": "Section Break", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Section Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "default": "Today", | ||||
|    "fieldname": "schedule_date", | ||||
|    "fieldtype": "Date", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "Schedule Date", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "label": "Schedule Date" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "room", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "Room", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "options": "Room", | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 1, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "column_break_9", | ||||
|    "fieldtype": "Column Break", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 0, | ||||
|    "in_standard_filter": 0, | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "from_time", | ||||
|    "fieldtype": "Time", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "From Time", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 1, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "to_time", | ||||
|    "fieldtype": "Time", | ||||
|    "hidden": 0, | ||||
|    "ignore_user_permissions": 0, | ||||
|    "ignore_xss_filter": 0, | ||||
|    "in_filter": 0, | ||||
|    "in_global_search": 0, | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0, | ||||
|    "label": "To Time", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 1, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0, | ||||
|    "allow_in_quick_entry": 0, | ||||
|    "allow_on_submit": 0, | ||||
|    "bold": 0, | ||||
|    "collapsible": 0, | ||||
|    "columns": 0, | ||||
|    "fieldname": "title", | ||||
|    "fieldtype": "Data", | ||||
|    "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": "Title", | ||||
|    "length": 0, | ||||
|    "no_copy": 0, | ||||
|    "permlevel": 0, | ||||
|    "precision": "", | ||||
|    "print_hide": 0, | ||||
|    "print_hide_if_no_value": 0, | ||||
|    "read_only": 0, | ||||
|    "remember_last_selected_value": 0, | ||||
|    "report_hide": 0, | ||||
|    "reqd": 0, | ||||
|    "search_index": 0, | ||||
|    "set_only_once": 0, | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "label": "Title" | ||||
|   } | ||||
|  ], | ||||
|  "has_web_view": 0, | ||||
|  "hide_heading": 0, | ||||
|  "hide_toolbar": 0, | ||||
|  "idx": 0, | ||||
|  "image_view": 0, | ||||
|  "in_create": 0, | ||||
|  "is_submittable": 0, | ||||
|  "issingle": 0, | ||||
|  "istable": 0, | ||||
|  "max_attachments": 0, | ||||
|  "menu_index": 0, | ||||
|  "modified": "2018-08-21 14:44:51.827225", | ||||
|  "links": [], | ||||
|  "modified": "2021-11-24 11:57:08.164449", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Education", | ||||
|  "name": "Course Schedule", | ||||
|  "name_case": "", | ||||
|  "naming_rule": "By \"Naming Series\" field", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|    "amend": 0, | ||||
|    "cancel": 0, | ||||
|    "create": 1, | ||||
|    "delete": 1, | ||||
|    "email": 1, | ||||
|    "export": 1, | ||||
|    "if_owner": 0, | ||||
|    "import": 0, | ||||
|    "permlevel": 0, | ||||
|    "print": 1, | ||||
|    "read": 1, | ||||
|    "report": 1, | ||||
|    "role": "Academics User", | ||||
|    "set_user_permissions": 0, | ||||
|    "share": 1, | ||||
|    "submit": 0, | ||||
|    "write": 1 | ||||
|   } | ||||
|  ], | ||||
|  "quick_entry": 0, | ||||
|  "read_only": 0, | ||||
|  "read_only_onload": 0, | ||||
|  "restrict_to_domain": "Education", | ||||
|  "show_name_in_global_search": 0, | ||||
|  "sort_field": "schedule_date", | ||||
|  "sort_order": "DESC", | ||||
|  "title_field": "title", | ||||
|  "track_changes": 0, | ||||
|  "track_seen": 0, | ||||
|  "track_views": 0 | ||||
|  "title_field": "title" | ||||
| } | ||||
| @ -1,661 +1,168 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "allow_copy": 1, | ||||
|  "allow_guest_to_view": 0,  | ||||
|  "allow_import": 0,  | ||||
|  "allow_rename": 0,  | ||||
|  "beta": 0,  | ||||
|  "creation": "2015-09-23 15:37:38.108475", | ||||
|  "custom": 0,  | ||||
|  "docstatus": 0,  | ||||
|  "doctype": "DocType", | ||||
|  "document_type": "Setup", | ||||
|  "editable_grid": 0,  | ||||
|  "engine": "InnoDB", | ||||
|  "field_order": [ | ||||
|   "student_group", | ||||
|   "course", | ||||
|   "program", | ||||
|   "column_break_3", | ||||
|   "academic_year", | ||||
|   "academic_term", | ||||
|   "section_break_6", | ||||
|   "instructor", | ||||
|   "instructor_name", | ||||
|   "column_break_9", | ||||
|   "room", | ||||
|   "section_break_7", | ||||
|   "course_start_date", | ||||
|   "course_end_date", | ||||
|   "day", | ||||
|   "reschedule", | ||||
|   "column_break_15", | ||||
|   "from_time", | ||||
|   "to_time" | ||||
|  ], | ||||
|  "fields": [ | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "student_group", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Student Group", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Student Group", | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "course", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Course", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Course", | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "program", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Program", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Program", | ||||
|    "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 | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "column_break_3", | ||||
|    "fieldtype": "Column Break",  | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "academic_year", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Academic Year", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Academic Year", | ||||
|    "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 | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "academic_term", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Academic Term", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Academic Term", | ||||
|    "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 | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "section_break_6", | ||||
|    "fieldtype": "Section Break",  | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Section Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "instructor", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Instructor", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Instructor", | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fetch_from": "instructor.instructor_name", | ||||
|    "fieldname": "instructor_name", | ||||
|    "fieldtype": "Read Only", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Instructor Name", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "",  | ||||
|    "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 | ||||
|    "read_only": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "column_break_9", | ||||
|    "fieldtype": "Column Break",  | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "room", | ||||
|    "fieldtype": "Link", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 1, | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Room", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "Room", | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "section_break_7", | ||||
|    "fieldtype": "Section Break",  | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Section Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "default": "",  | ||||
|    "fieldname": "from_time", | ||||
|    "fieldtype": "Time", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "From Time", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "default": "",  | ||||
|    "fieldname": "course_start_date", | ||||
|    "fieldtype": "Date", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Course Start Date", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "",  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "day", | ||||
|    "fieldtype": "Select", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Day", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "options": "\nMonday\nTuesday\nWednesday\nThursday\nFriday\nSaturday\nSunday", | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "default": "0", | ||||
|    "fieldname": "reschedule", | ||||
|    "fieldtype": "Check", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Reschedule",  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "label": "Reschedule" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "column_break_15", | ||||
|    "fieldtype": "Column Break",  | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 0,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "fieldname": "to_time", | ||||
|    "fieldtype": "Time", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "To TIme", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   }, | ||||
|   { | ||||
|    "allow_bulk_edit": 0,  | ||||
|    "allow_on_submit": 0,  | ||||
|    "bold": 0,  | ||||
|    "collapsible": 0,  | ||||
|    "columns": 0,  | ||||
|    "default": "",  | ||||
|    "fieldname": "course_end_date", | ||||
|    "fieldtype": "Date", | ||||
|    "hidden": 0,  | ||||
|    "ignore_user_permissions": 0,  | ||||
|    "ignore_xss_filter": 0,  | ||||
|    "in_filter": 0,  | ||||
|    "in_global_search": 0,  | ||||
|    "in_list_view": 0,  | ||||
|    "in_standard_filter": 0,  | ||||
|    "label": "Course End Date", | ||||
|    "length": 0,  | ||||
|    "no_copy": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "precision": "",  | ||||
|    "print_hide": 0,  | ||||
|    "print_hide_if_no_value": 0,  | ||||
|    "read_only": 0,  | ||||
|    "remember_last_selected_value": 0,  | ||||
|    "report_hide": 0,  | ||||
|    "reqd": 1,  | ||||
|    "search_index": 0,  | ||||
|    "set_only_once": 0,  | ||||
|    "translatable": 0, | ||||
|    "unique": 0 | ||||
|    "reqd": 1 | ||||
|   } | ||||
|  ], | ||||
|  "has_web_view": 0,  | ||||
|  "hide_heading": 1,  | ||||
|  "hide_toolbar": 1, | ||||
|  "idx": 0,  | ||||
|  "image_view": 0,  | ||||
|  "in_create": 0,  | ||||
|  "is_submittable": 0,  | ||||
|  "issingle": 1, | ||||
|  "istable": 0,  | ||||
|  "max_attachments": 0,  | ||||
|  "menu_index": 0,  | ||||
|  "modified": "2018-05-16 22:43:29.363798", | ||||
|  "links": [], | ||||
|  "modified": "2021-11-11 09:33:18.874445", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Education", | ||||
|  "name": "Course Scheduling Tool", | ||||
|  "name_case": "",  | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|    "amend": 0,  | ||||
|    "cancel": 0,  | ||||
|    "create": 1, | ||||
|    "delete": 0,  | ||||
|    "email": 0,  | ||||
|    "export": 0,  | ||||
|    "if_owner": 0,  | ||||
|    "import": 0,  | ||||
|    "permlevel": 0,  | ||||
|    "print": 0,  | ||||
|    "read": 1, | ||||
|    "report": 0,  | ||||
|    "role": "Academics User", | ||||
|    "set_user_permissions": 0,  | ||||
|    "share": 0,  | ||||
|    "submit": 0,  | ||||
|    "write": 1 | ||||
|   } | ||||
|  ], | ||||
|  "quick_entry": 0,  | ||||
|  "read_only": 0,  | ||||
|  "read_only_onload": 0,  | ||||
|  "restrict_to_domain": "Education", | ||||
|  "show_name_in_global_search": 0,  | ||||
|  "sort_field": "modified", | ||||
|  "sort_order": "DESC", | ||||
|  "track_changes": 1,  | ||||
|  "track_seen": 0 | ||||
|  "track_changes": 1 | ||||
| } | ||||
| @ -17,21 +17,33 @@ def get_student_attendance_records(based_on, date=None, student_group=None, cour | ||||
| 	if based_on == "Course Schedule": | ||||
| 		student_group = frappe.db.get_value("Course Schedule", course_schedule, "student_group") | ||||
| 		if student_group: | ||||
| 			student_list = frappe.get_list("Student Group Student", fields=["student", "student_name", "group_roll_number"] , \ | ||||
| 			student_list = frappe.get_all("Student Group Student", fields=["student", "student_name", "group_roll_number"], | ||||
| 			filters={"parent": student_group, "active": 1}, order_by= "group_roll_number") | ||||
| 
 | ||||
| 	if not student_list: | ||||
| 		student_list = frappe.get_list("Student Group Student", fields=["student", "student_name", "group_roll_number"] , | ||||
| 		student_list = frappe.get_all("Student Group Student", fields=["student", "student_name", "group_roll_number"], | ||||
| 			filters={"parent": student_group, "active": 1}, order_by= "group_roll_number") | ||||
| 
 | ||||
| 	table = frappe.qb.DocType("Student Attendance") | ||||
| 
 | ||||
| 	if course_schedule: | ||||
| 		student_attendance_list= frappe.db.sql('''select student, status from `tabStudent Attendance` where \ | ||||
| 			course_schedule= %s''', (course_schedule), as_dict=1) | ||||
| 		student_attendance_list = ( | ||||
| 			frappe.qb.from_(table) | ||||
| 				.select(table.student, table.status) | ||||
| 				.where( | ||||
| 					(table.course_schedule == course_schedule) | ||||
| 				) | ||||
| 			).run(as_dict=True) | ||||
| 	else: | ||||
| 		student_attendance_list= frappe.db.sql('''select student, status from `tabStudent Attendance` where \ | ||||
| 			student_group= %s and date= %s and \ | ||||
| 			(course_schedule is Null or course_schedule='')''', | ||||
| 			(student_group, date), as_dict=1) | ||||
| 		student_attendance_list = ( | ||||
| 			frappe.qb.from_(table) | ||||
| 				.select(table.student, table.status) | ||||
| 				.where( | ||||
| 					(table.student_group == student_group) | ||||
| 					& (table.date == date) | ||||
| 					& (table.course_schedule == "") | (table.course_schedule.isnull()) | ||||
| 				) | ||||
| 			).run(as_dict=True) | ||||
| 
 | ||||
| 	for attendance in student_attendance_list: | ||||
| 		for student in student_list: | ||||
|  | ||||
| @ -19,8 +19,7 @@ def verify_request(): | ||||
| 	) | ||||
| 
 | ||||
| 	if frappe.request.data and \ | ||||
| 		frappe.get_request_header("X-Wc-Webhook-Signature") and \ | ||||
| 		not sig == bytes(frappe.get_request_header("X-Wc-Webhook-Signature").encode()): | ||||
| 		not sig == frappe.get_request_header("X-Wc-Webhook-Signature", "").encode(): | ||||
| 			frappe.throw(_("Unverified Webhook Data")) | ||||
| 	frappe.set_user(woocommerce_settings.creation_user) | ||||
| 
 | ||||
|  | ||||
| @ -141,6 +141,9 @@ def verify_transaction(**kwargs): | ||||
| 	transaction_response = frappe._dict(kwargs["Body"]["stkCallback"]) | ||||
| 
 | ||||
| 	checkout_id = getattr(transaction_response, "CheckoutRequestID", "") | ||||
| 	if not isinstance(checkout_id, str): | ||||
| 		frappe.throw(_("Invalid Checkout Request ID")) | ||||
| 
 | ||||
| 	integration_request = frappe.get_doc("Integration Request", checkout_id) | ||||
| 	transaction_data = frappe._dict(loads(integration_request.data)) | ||||
| 	total_paid = 0 # for multiple integration request made against a pos invoice | ||||
| @ -231,6 +234,9 @@ def process_balance_info(**kwargs): | ||||
| 	account_balance_response = frappe._dict(kwargs["Result"]) | ||||
| 
 | ||||
| 	conversation_id = getattr(account_balance_response, "ConversationID", "") | ||||
| 	if not isinstance(conversation_id, str): | ||||
| 		frappe.throw(_("Invalid Conversation ID")) | ||||
| 
 | ||||
| 	request = frappe.get_doc("Integration Request", conversation_id) | ||||
| 
 | ||||
| 	if request.status == "Completed": | ||||
|  | ||||
| @ -23,7 +23,6 @@ def validate_webhooks_request(doctype,  hmac_key, secret_key='secret'): | ||||
| 			) | ||||
| 
 | ||||
| 			if frappe.request.data and \ | ||||
| 				frappe.get_request_header(hmac_key) and \ | ||||
| 				not sig == bytes(frappe.get_request_header(hmac_key).encode()): | ||||
| 					frappe.throw(_("Unverified Webhook Data")) | ||||
| 			frappe.set_user(settings.modified_by) | ||||
|  | ||||
| @ -29,17 +29,6 @@ | ||||
|    "onboard": 0, | ||||
|    "type": "Link" | ||||
|   }, | ||||
|   { | ||||
|    "dependencies": "", | ||||
|    "hidden": 0, | ||||
|    "is_query_report": 0, | ||||
|    "label": "Shopify Settings", | ||||
|    "link_count": 0, | ||||
|    "link_to": "Shopify Settings", | ||||
|    "link_type": "DocType", | ||||
|    "onboard": 0, | ||||
|    "type": "Link" | ||||
|   }, | ||||
|   { | ||||
|    "dependencies": "", | ||||
|    "hidden": 0, | ||||
| @ -74,7 +63,7 @@ | ||||
|    "type": "Link" | ||||
|   } | ||||
|  ], | ||||
|  "modified": "2021-08-05 12:15:58.951705", | ||||
|  "modified": "2021-11-23 04:30:33.106991", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "ERPNext Integrations", | ||||
|  "name": "ERPNext Integrations Settings", | ||||
|  | ||||
| @ -248,20 +248,18 @@ doc_events = { | ||||
| 		"validate": "erpnext.regional.india.utils.validate_tax_category" | ||||
| 	}, | ||||
| 	"Sales Invoice": { | ||||
| 		"after_insert": "erpnext.regional.saudi_arabia.utils.create_qr_code", | ||||
| 		"on_submit": [ | ||||
| 			"erpnext.regional.create_transaction_log", | ||||
| 			"erpnext.regional.italy.utils.sales_invoice_on_submit", | ||||
| 			"erpnext.regional.saudi_arabia.utils.create_qr_code", | ||||
| 			"erpnext.erpnext_integrations.taxjar_integration.create_transaction" | ||||
| 		], | ||||
| 		"on_cancel": [ | ||||
| 			"erpnext.regional.italy.utils.sales_invoice_on_cancel", | ||||
| 			"erpnext.erpnext_integrations.taxjar_integration.delete_transaction" | ||||
| 		], | ||||
| 		"on_trash": [ | ||||
| 			"erpnext.regional.check_deletion_permission", | ||||
| 			"erpnext.erpnext_integrations.taxjar_integration.delete_transaction", | ||||
| 			"erpnext.regional.saudi_arabia.utils.delete_qr_code_file" | ||||
| 		], | ||||
| 		"on_trash": "erpnext.regional.check_deletion_permission", | ||||
| 		"validate": [ | ||||
| 			"erpnext.regional.india.utils.validate_document_name", | ||||
| 			"erpnext.regional.india.utils.update_taxable_values" | ||||
| @ -306,7 +304,8 @@ doc_events = { | ||||
| 		'validate': ["erpnext.erpnext_integrations.taxjar_integration.set_sales_tax"] | ||||
| 	}, | ||||
| 	"Company": { | ||||
| 		"on_trash": "erpnext.regional.india.utils.delete_gst_settings_for_company" | ||||
| 		"on_trash": ["erpnext.regional.india.utils.delete_gst_settings_for_company", | ||||
| 			"erpnext.regional.saudi_arabia.utils.delete_vat_settings_for_company"] | ||||
| 	}, | ||||
| 	"Integration Request": { | ||||
| 		"validate": "erpnext.accounts.doctype.payment_request.payment_request.validate_payment" | ||||
|  | ||||
| @ -96,15 +96,8 @@ class Employee(NestedSet): | ||||
| 			'user': self.user_id | ||||
| 		}) | ||||
| 
 | ||||
| 		if employee_user_permission_exists: return | ||||
| 
 | ||||
| 		employee_user_permission_exists = frappe.db.exists('User Permission', { | ||||
| 			'allow': 'Employee', | ||||
| 			'for_value': self.name, | ||||
| 			'user': self.user_id | ||||
| 		}) | ||||
| 
 | ||||
| 		if employee_user_permission_exists: return | ||||
| 		if employee_user_permission_exists: | ||||
| 			return | ||||
| 
 | ||||
| 		add_user_permission("Employee", self.name, self.user_id) | ||||
| 		set_user_permission_if_allowed("Company", self.company, self.user_id) | ||||
|  | ||||
| @ -5,6 +5,7 @@ | ||||
| import frappe | ||||
| from frappe import _ | ||||
| from frappe.model.document import Document | ||||
| from frappe.query_builder.functions import Sum | ||||
| from frappe.utils import flt, nowdate | ||||
| 
 | ||||
| import erpnext | ||||
| @ -41,24 +42,34 @@ class EmployeeAdvance(Document): | ||||
| 			self.status = "Cancelled" | ||||
| 
 | ||||
| 	def set_total_advance_paid(self): | ||||
| 		paid_amount = frappe.db.sql(""" | ||||
| 			select ifnull(sum(debit), 0) as paid_amount | ||||
| 			from `tabGL Entry` | ||||
| 			where against_voucher_type = 'Employee Advance' | ||||
| 				and against_voucher = %s | ||||
| 				and party_type = 'Employee' | ||||
| 				and party = %s | ||||
| 		""", (self.name, self.employee), as_dict=1)[0].paid_amount | ||||
| 		gle = frappe.qb.DocType("GL Entry") | ||||
| 
 | ||||
| 		return_amount = frappe.db.sql(""" | ||||
| 			select ifnull(sum(credit), 0) as return_amount | ||||
| 			from `tabGL Entry` | ||||
| 			where against_voucher_type = 'Employee Advance' | ||||
| 				and voucher_type != 'Expense Claim' | ||||
| 				and against_voucher = %s | ||||
| 				and party_type = 'Employee' | ||||
| 				and party = %s | ||||
| 		""", (self.name, self.employee), as_dict=1)[0].return_amount | ||||
| 		paid_amount = ( | ||||
| 			frappe.qb.from_(gle) | ||||
| 				.select(Sum(gle.debit).as_("paid_amount")) | ||||
| 				.where( | ||||
| 					(gle.against_voucher_type == 'Employee Advance') | ||||
| 					& (gle.against_voucher == self.name) | ||||
| 					& (gle.party_type == 'Employee') | ||||
| 					& (gle.party == self.employee) | ||||
| 					& (gle.docstatus == 1) | ||||
| 					& (gle.is_cancelled == 0) | ||||
| 				) | ||||
| 			).run(as_dict=True)[0].paid_amount or 0 | ||||
| 
 | ||||
| 		return_amount = ( | ||||
| 			frappe.qb.from_(gle) | ||||
| 				.select(Sum(gle.credit).as_("return_amount")) | ||||
| 				.where( | ||||
| 					(gle.against_voucher_type == 'Employee Advance') | ||||
| 					& (gle.voucher_type != 'Expense Claim') | ||||
| 					& (gle.against_voucher == self.name) | ||||
| 					& (gle.party_type == 'Employee') | ||||
| 					& (gle.party == self.employee) | ||||
| 					& (gle.docstatus == 1) | ||||
| 					& (gle.is_cancelled == 0) | ||||
| 				) | ||||
| 			).run(as_dict=True)[0].return_amount or 0 | ||||
| 
 | ||||
| 		if paid_amount != 0: | ||||
| 			paid_amount = flt(paid_amount) / flt(self.exchange_rate) | ||||
|  | ||||
| @ -34,6 +34,24 @@ class TestEmployeeAdvance(unittest.TestCase): | ||||
| 		journal_entry1 = make_payment_entry(advance) | ||||
| 		self.assertRaises(EmployeeAdvanceOverPayment, journal_entry1.submit) | ||||
| 
 | ||||
| 	def test_paid_amount_on_pe_cancellation(self): | ||||
| 		employee_name = make_employee("_T@employe.advance") | ||||
| 		advance = make_employee_advance(employee_name) | ||||
| 
 | ||||
| 		pe = make_payment_entry(advance) | ||||
| 		pe.submit() | ||||
| 
 | ||||
| 		advance.reload() | ||||
| 
 | ||||
| 		self.assertEqual(advance.paid_amount, 1000) | ||||
| 		self.assertEqual(advance.status, "Paid") | ||||
| 
 | ||||
| 		pe.cancel() | ||||
| 		advance.reload() | ||||
| 
 | ||||
| 		self.assertEqual(advance.paid_amount, 0) | ||||
| 		self.assertEqual(advance.status, "Unpaid") | ||||
| 
 | ||||
| 	def test_repay_unclaimed_amount_from_salary(self): | ||||
| 		employee_name = make_employee("_T@employe.advance") | ||||
| 		advance = make_employee_advance(employee_name, {"repay_unclaimed_amount_from_salary": 1}) | ||||
|  | ||||
| @ -2,7 +2,6 @@ | ||||
| # See license.txt | ||||
| 
 | ||||
| import unittest | ||||
| from datetime import date | ||||
| 
 | ||||
| import frappe | ||||
| from frappe.utils import add_days, getdate | ||||
| @ -12,16 +11,14 @@ from erpnext.hr.doctype.employee.test_employee import make_employee | ||||
| 
 | ||||
| class TestEmployeeTransfer(unittest.TestCase): | ||||
| 	def setUp(self): | ||||
| 		make_employee("employee2@transfers.com") | ||||
| 		make_employee("employee3@transfers.com") | ||||
| 		create_company() | ||||
| 		create_employee() | ||||
| 		create_employee_transfer() | ||||
| 
 | ||||
| 	def tearDown(self): | ||||
| 		frappe.db.rollback() | ||||
| 
 | ||||
| 	def test_submit_before_transfer_date(self): | ||||
| 		make_employee("employee2@transfers.com") | ||||
| 
 | ||||
| 		transfer_obj = frappe.get_doc({ | ||||
| 			"doctype": "Employee Transfer", | ||||
| 			"employee": frappe.get_value("Employee", {"user_id":"employee2@transfers.com"}, "name"), | ||||
| @ -43,6 +40,8 @@ class TestEmployeeTransfer(unittest.TestCase): | ||||
| 		self.assertEqual(transfer.docstatus, 1) | ||||
| 
 | ||||
| 	def test_new_employee_creation(self): | ||||
| 		make_employee("employee3@transfers.com") | ||||
| 
 | ||||
| 		transfer = frappe.get_doc({ | ||||
| 			"doctype": "Employee Transfer", | ||||
| 			"employee": frappe.get_value("Employee", {"user_id":"employee3@transfers.com"}, "name"), | ||||
| @ -63,60 +62,51 @@ class TestEmployeeTransfer(unittest.TestCase): | ||||
| 		self.assertEqual(frappe.get_value("Employee", transfer.employee, "status"), "Left") | ||||
| 
 | ||||
| 	def test_employee_history(self): | ||||
| 		name = frappe.get_value("Employee", {"first_name": "John", "company": "Test Company"}, "name") | ||||
| 		doc = frappe.get_doc("Employee",name) | ||||
| 		employee = make_employee("employee4@transfers.com", | ||||
| 			company="Test Company", | ||||
| 			date_of_birth=getdate("30-09-1980"), | ||||
| 			date_of_joining=getdate("01-10-2021"), | ||||
| 			department="Accounts - TC", | ||||
| 			designation="Accountant" | ||||
| 		) | ||||
| 		transfer = create_employee_transfer(employee) | ||||
| 
 | ||||
| 		count = 0 | ||||
| 		department = ["Accounts - TC", "Management - TC"] | ||||
| 		designation = ["Accountant", "Manager"] | ||||
| 		dt = [getdate("01-10-2021"), date.today()] | ||||
| 		dt = [getdate("01-10-2021"), getdate()] | ||||
| 
 | ||||
| 		for data in doc.internal_work_history: | ||||
| 		employee = frappe.get_doc("Employee", employee) | ||||
| 		for data in employee.internal_work_history: | ||||
| 			self.assertEqual(data.department, department[count]) | ||||
| 			self.assertEqual(data.designation, designation[count]) | ||||
| 			self.assertEqual(data.from_date, dt[count]) | ||||
| 			count = count + 1 | ||||
| 
 | ||||
| 		data = frappe.db.get_list("Employee Transfer", filters={"employee":name}, fields=["*"]) | ||||
| 		doc = frappe.get_doc("Employee Transfer", data[0]["name"]) | ||||
| 		doc.cancel() | ||||
| 		employee_doc = frappe.get_doc("Employee",name) | ||||
| 		transfer.cancel() | ||||
| 		employee.reload() | ||||
| 
 | ||||
| 		for data in employee_doc.internal_work_history: | ||||
| 		for data in employee.internal_work_history: | ||||
| 			self.assertEqual(data.designation, designation[0]) | ||||
| 			self.assertEqual(data.department, department[0]) | ||||
| 			self.assertEqual(data.from_date, dt[0]) | ||||
| 
 | ||||
| def create_employee(): | ||||
| 	doc = frappe.get_doc({ | ||||
| 			"doctype": "Employee", | ||||
| 			"first_name": "John", | ||||
| 			"company": "Test Company", | ||||
| 			"gender": "Male", | ||||
| 			"date_of_birth": getdate("30-09-1980"), | ||||
| 			"date_of_joining": getdate("01-10-2021"), | ||||
| 			"department": "Accounts - TC", | ||||
| 			"designation": "Accountant" | ||||
| 	}) | ||||
| 
 | ||||
| 	doc.save() | ||||
| 
 | ||||
| def create_company(): | ||||
| 	exists = frappe.db.exists("Company", "Test Company") | ||||
| 	if not exists: | ||||
| 		doc = frappe.get_doc({ | ||||
| 	if not frappe.db.exists("Company", "Test Company"): | ||||
| 		frappe.get_doc({ | ||||
| 			"doctype": "Company", | ||||
| 			"company_name": "Test Company", | ||||
| 			"default_currency": "INR", | ||||
| 			"country": "India" | ||||
| 		}) | ||||
| 		}).insert() | ||||
| 
 | ||||
| 		doc.save() | ||||
| 
 | ||||
| def create_employee_transfer(): | ||||
| def create_employee_transfer(employee): | ||||
| 	doc = frappe.get_doc({ | ||||
| 		"doctype": "Employee Transfer", | ||||
| 		"employee": frappe.get_value("Employee", {"first_name": "John", "company": "Test Company"}, "name"), | ||||
| 		"transfer_date": date.today(), | ||||
| 		"employee": employee, | ||||
| 		"transfer_date": getdate(), | ||||
| 		"transfer_details": [ | ||||
| 			{ | ||||
| 				"property": "Designation", | ||||
| @ -135,3 +125,5 @@ def create_employee_transfer(): | ||||
| 
 | ||||
| 	doc.save() | ||||
| 	doc.submit() | ||||
| 
 | ||||
| 	return doc | ||||
| @ -389,7 +389,9 @@ frappe.ui.form.on("Expense Claim Detail", { | ||||
| 	sanctioned_amount: function(frm, cdt, cdn) { | ||||
| 		cur_frm.cscript.calculate_total(frm.doc, cdt, cdn); | ||||
| 		frm.trigger("get_taxes"); | ||||
| 		frm.trigger("calculate_grand_total"); | ||||
| 	}, | ||||
| 
 | ||||
| 	cost_center: function(frm, cdt, cdn) { | ||||
| 		erpnext.utils.copy_value_in_all_rows(frm.doc, cdt, cdn, "expenses", "cost_center"); | ||||
| 	} | ||||
|  | ||||
| @ -379,11 +379,12 @@ | ||||
|  "idx": 1, | ||||
|  "is_submittable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-05-04 05:35:12.040199", | ||||
|  "modified": "2021-11-22 16:26:57.787838", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "HR", | ||||
|  "name": "Expense Claim", | ||||
|  "name_case": "Title Case", | ||||
|  "naming_rule": "By \"Naming Series\" field", | ||||
|  "owner": "Administrator", | ||||
|  "permissions": [ | ||||
|   { | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "creation": "2017-10-09 16:53:26.410762", | ||||
|  "doctype": "DocType", | ||||
|  "document_type": "Document", | ||||
| @ -50,7 +51,7 @@ | ||||
|    "fieldname": "unclaimed_amount", | ||||
|    "fieldtype": "Currency", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Unclaimed amount", | ||||
|    "label": "Unclaimed Amount", | ||||
|    "no_copy": 1, | ||||
|    "oldfieldname": "advance_amount", | ||||
|    "oldfieldtype": "Currency", | ||||
| @ -65,7 +66,7 @@ | ||||
|    "fieldname": "allocated_amount", | ||||
|    "fieldtype": "Currency", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Allocated amount", | ||||
|    "label": "Allocated Amount", | ||||
|    "no_copy": 1, | ||||
|    "oldfieldname": "allocated_amount", | ||||
|    "oldfieldtype": "Currency", | ||||
| @ -87,7 +88,7 @@ | ||||
|  ], | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2019-12-17 13:53:22.111766", | ||||
|  "modified": "2021-11-22 16:33:58.515819", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "HR", | ||||
|  "name": "Expense Claim Advance", | ||||
|  | ||||
| @ -94,7 +94,6 @@ | ||||
|    "fieldtype": "Currency", | ||||
|    "in_list_view": 1, | ||||
|    "label": "Sanctioned Amount", | ||||
|    "no_copy": 1, | ||||
|    "oldfieldname": "sanctioned_amount", | ||||
|    "oldfieldtype": "Currency", | ||||
|    "options": "Company:company:default_currency", | ||||
| @ -120,7 +119,7 @@ | ||||
|  "idx": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2020-09-18 17:26:09.703215", | ||||
|  "modified": "2021-11-26 14:23:45.539922", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "HR", | ||||
|  "name": "Expense Claim Detail", | ||||
|  | ||||
| @ -19,8 +19,8 @@ class ShiftAssignment(Document): | ||||
| 		validate_active_employee(self.employee) | ||||
| 		self.validate_overlapping_dates() | ||||
| 
 | ||||
| 		if self.end_date and self.end_date <= self.start_date: | ||||
| 			frappe.throw(_("End Date must not be lesser than Start Date")) | ||||
| 		if self.end_date: | ||||
| 			self.validate_from_to_dates('start_date', 'end_date') | ||||
| 
 | ||||
| 	def validate_overlapping_dates(self): | ||||
| 		if not self.name: | ||||
|  | ||||
| @ -3,7 +3,39 @@ | ||||
| 
 | ||||
| import unittest | ||||
| 
 | ||||
| import frappe | ||||
| from frappe.utils.data import today | ||||
| 
 | ||||
| # test_records = frappe.get_test_records('Maintenance Visit') | ||||
| 
 | ||||
| class TestMaintenanceVisit(unittest.TestCase): | ||||
| 	pass | ||||
| 
 | ||||
| def make_maintenance_visit(): | ||||
| 	mv = frappe.new_doc("Maintenance Visit") | ||||
| 	mv.company = "_Test Company" | ||||
| 	mv.customer = "_Test Customer" | ||||
| 	mv.mntc_date = today() | ||||
| 	mv.completion_status = "Partially Completed" | ||||
| 
 | ||||
| 	sales_person = make_sales_person("Dwight Schrute") | ||||
| 
 | ||||
| 	mv.append("purposes", { | ||||
| 		"item_code": "_Test Item", | ||||
| 		"sales_person": "Sales Team", | ||||
| 		"description": "Test Item", | ||||
| 		"work_done": "Test Work Done", | ||||
| 		"service_person": sales_person.name | ||||
| 	}) | ||||
| 	mv.insert(ignore_permissions=True) | ||||
| 
 | ||||
| 	return mv | ||||
| 
 | ||||
| def make_sales_person(name): | ||||
| 	sales_person = frappe.get_doc({ | ||||
| 		'doctype': "Sales Person", | ||||
| 		'sales_person_name': name | ||||
| 	}) | ||||
| 	sales_person.insert(ignore_if_duplicate = True) | ||||
| 
 | ||||
| 	return sales_person | ||||
|  | ||||
| @ -680,12 +680,6 @@ frappe.ui.form.on("BOM Item", "items_remove", function(frm) { | ||||
| 	erpnext.bom.calculate_total(frm.doc); | ||||
| }); | ||||
| 
 | ||||
| frappe.ui.form.on("BOM", "with_operations", function(frm) { | ||||
| 	if(!cint(frm.doc.with_operations)) { | ||||
| 		frm.set_value("operations", []); | ||||
| 	} | ||||
| }); | ||||
| 
 | ||||
| frappe.tour['BOM'] = [ | ||||
| 	{ | ||||
| 		fieldname: "item", | ||||
|  | ||||
| @ -237,6 +237,7 @@ | ||||
|    "options": "Price List" | ||||
|   }, | ||||
|   { | ||||
|    "depends_on": "with_operations", | ||||
|    "fieldname": "operations_section", | ||||
|    "fieldtype": "Section Break", | ||||
|    "hide_border": 1, | ||||
| @ -539,7 +540,7 @@ | ||||
|  "image_field": "image", | ||||
|  "is_submittable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-10-27 14:52:04.500251", | ||||
|  "modified": "2021-11-18 13:04:16.271975", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Manufacturing", | ||||
|  "name": "BOM", | ||||
|  | ||||
| @ -16,26 +16,15 @@ | ||||
| 			</div> | ||||
| 			<hr style="margin: 15px -15px;"> | ||||
| 			<p> | ||||
| 				{% if data.value %} | ||||
| 				<a style="margin-right: 7px; margin-bottom: 7px" class="btn btn-default btn-xs" href="#Form/BOM/{{ data.value }}"> | ||||
| 				{% if data.value && data.value != "BOM" %} | ||||
| 				<a style="margin-right: 7px; margin-bottom: 7px" class="btn btn-default btn-xs" href="/app/bom/{{ data.value }}"> | ||||
| 					{{ __("Open BOM {0}", [data.value.bold()]) }}</a> | ||||
| 				{% endif %} | ||||
| 				{% if data.item_code %} | ||||
| 				<a class="btn btn-default btn-xs" href="#Form/Item/{{ data.item_code }}"> | ||||
| 				<a style="margin-right: 7px; margin-bottom: 7px"  class="btn btn-default btn-xs" href="/app/item/{{ data.item_code }}"> | ||||
| 					{{ __("Open Item {0}", [data.item_code.bold()]) }}</a> | ||||
| 				{% endif %} | ||||
| 			</p> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<hr style="margin: 15px -15px;"> | ||||
| 	<p> | ||||
| 		{% if data.value %} | ||||
| 		<a style="margin-right: 7px; margin-bottom: 7px" class="btn btn-default btn-xs" href="/app/Form/BOM/{{ data.value }}"> | ||||
| 			{{ __("Open BOM {0}", [data.value.bold()]) }}</a> | ||||
| 		{% endif %} | ||||
| 		{% if data.item_code %} | ||||
| 		<a class="btn btn-default btn-xs" href="/app/Form/Item/{{ data.item_code }}"> | ||||
| 			{{ __("Open Item {0}", [data.item_code.bold()]) }}</a> | ||||
| 		{% endif %} | ||||
| 	</p> | ||||
| </div> | ||||
|  | ||||
| @ -66,6 +66,7 @@ frappe.treeview_settings["BOM"] = { | ||||
| 				var bom = frappe.model.get_doc("BOM", node.data.value); | ||||
| 				node.data.image = escape(bom.image) || ""; | ||||
| 				node.data.description = bom.description || ""; | ||||
| 				node.data.item_code = bom.item || ""; | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| @ -28,6 +28,11 @@ frappe.ui.form.on('Job Card', { | ||||
| 		frappe.flags.resume_job = 0; | ||||
| 		let has_items = frm.doc.items && frm.doc.items.length; | ||||
| 
 | ||||
| 		if (frm.doc.__onload.work_order_closed) { | ||||
| 			frm.disable_save(); | ||||
| 			return; | ||||
| 		} | ||||
| 
 | ||||
| 		if (!frm.doc.__islocal && has_items && frm.doc.docstatus < 2) { | ||||
| 			let to_request = frm.doc.for_quantity > frm.doc.transferred_qty; | ||||
| 			let excess_transfer_allowed = frm.doc.__onload.job_card_excess_transfer; | ||||
|  | ||||
| @ -396,6 +396,7 @@ | ||||
|    "options": "Batch" | ||||
|   }, | ||||
|   { | ||||
|    "collapsible": 1, | ||||
|    "fieldname": "scrap_items_section", | ||||
|    "fieldtype": "Section Break", | ||||
|    "label": "Scrap Items" | ||||
| @ -411,7 +412,7 @@ | ||||
|  ], | ||||
|  "is_submittable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2021-09-14 00:38:46.873105", | ||||
|  "modified": "2021-11-12 10:15:03.572401", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Manufacturing", | ||||
|  "name": "Job Card", | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user