Merge branch 'develop' into hsn_wise_json
This commit is contained in:
		
						commit
						29f187f4f3
					
				| @ -9,6 +9,8 @@ from erpnext.selling.doctype.sales_order.test_sales_order import make_sales_orde | ||||
| from erpnext.stock.get_item_details import get_item_details | ||||
| from frappe.test_runner import make_test_objects | ||||
| 
 | ||||
| test_dependencies = ['Item'] | ||||
| 
 | ||||
| def test_create_test_data(): | ||||
| 	frappe.set_user("Administrator") | ||||
| 	# create test item | ||||
| @ -95,7 +97,6 @@ def test_create_test_data(): | ||||
| 		}) | ||||
| 		coupon_code.insert() | ||||
| 
 | ||||
| 
 | ||||
| class TestCouponCode(unittest.TestCase): | ||||
| 	def setUp(self): | ||||
| 		test_create_test_data() | ||||
|  | ||||
| @ -21,7 +21,7 @@ from six import iteritems | ||||
| class POSInvoice(SalesInvoice): | ||||
| 	def __init__(self, *args, **kwargs): | ||||
| 		super(POSInvoice, self).__init__(*args, **kwargs) | ||||
| 	 | ||||
| 
 | ||||
| 	def validate(self): | ||||
| 		if not cint(self.is_pos): | ||||
| 			frappe.throw(_("POS Invoice should have {} field checked.").format(frappe.bold("Include Payment"))) | ||||
| @ -58,7 +58,7 @@ class POSInvoice(SalesInvoice): | ||||
| 		if self.redeem_loyalty_points and self.loyalty_points: | ||||
| 			self.apply_loyalty_points() | ||||
| 		self.set_status(update=True) | ||||
| 	 | ||||
| 
 | ||||
| 	def on_cancel(self): | ||||
| 		# run on cancel method of selling controller | ||||
| 		super(SalesInvoice, self).on_cancel() | ||||
| @ -68,10 +68,10 @@ class POSInvoice(SalesInvoice): | ||||
| 			against_psi_doc = frappe.get_doc("POS Invoice", self.return_against) | ||||
| 			against_psi_doc.delete_loyalty_point_entry() | ||||
| 			against_psi_doc.make_loyalty_point_entry() | ||||
| 		 | ||||
| 
 | ||||
| 	def validate_stock_availablility(self): | ||||
| 		allow_negative_stock = frappe.db.get_value('Stock Settings', None, 'allow_negative_stock') | ||||
| 		 | ||||
| 
 | ||||
| 		for d in self.get('items'): | ||||
| 			if d.serial_no: | ||||
| 				filters = { | ||||
| @ -89,11 +89,11 @@ class POSInvoice(SalesInvoice): | ||||
| 				for s in serial_nos: | ||||
| 					if s in reserved_serial_nos: | ||||
| 						invalid_serial_nos.append(s) | ||||
| 				 | ||||
| 
 | ||||
| 				if len(invalid_serial_nos): | ||||
| 					multiple_nos = 's' if len(invalid_serial_nos) > 1 else '' | ||||
| 					frappe.throw(_("Row #{}: Serial No{}. {} has already been transacted into another POS Invoice. \ | ||||
| 						Please select valid serial no.".format(d.idx, multiple_nos,  | ||||
| 						Please select valid serial no.".format(d.idx, multiple_nos, | ||||
| 						frappe.bold(', '.join(invalid_serial_nos)))), title=_("Not Available")) | ||||
| 			else: | ||||
| 				if allow_negative_stock: | ||||
| @ -105,9 +105,9 @@ class POSInvoice(SalesInvoice): | ||||
| 						.format(d.idx, frappe.bold(d.item_code), frappe.bold(d.warehouse))), title=_("Not Available")) | ||||
| 				elif flt(available_stock) < flt(d.qty): | ||||
| 					frappe.msgprint(_('Row #{}: Stock quantity not enough for Item Code: {} under warehouse {}. \ | ||||
| 						Available quantity {}.'.format(d.idx, frappe.bold(d.item_code),  | ||||
| 						Available quantity {}.'.format(d.idx, frappe.bold(d.item_code), | ||||
| 						frappe.bold(d.warehouse), frappe.bold(d.qty))), title=_("Not Available")) | ||||
| 	 | ||||
| 
 | ||||
| 	def validate_serialised_or_batched_item(self): | ||||
| 		for d in self.get("items"): | ||||
| 			serialized = d.get("has_serial_no") | ||||
| @ -125,7 +125,7 @@ class POSInvoice(SalesInvoice): | ||||
| 			if batched and no_batch_selected: | ||||
| 				frappe.throw(_('Row #{}: No batch selected against item: {}. Please select a batch or remove it to complete transaction.' | ||||
| 						.format(d.idx, frappe.bold(d.item_code))), title=_("Invalid Item")) | ||||
| 	 | ||||
| 
 | ||||
| 	def validate_return_items(self): | ||||
| 		if not self.get("is_return"): return | ||||
| 
 | ||||
| @ -158,7 +158,7 @@ class POSInvoice(SalesInvoice): | ||||
| 				frappe.throw(_("Row #{0} (Payment Table): Amount must be positive").format(entry.idx)) | ||||
| 			if self.is_return and entry.amount > 0: | ||||
| 				frappe.throw(_("Row #{0} (Payment Table): Amount must be negative").format(entry.idx)) | ||||
| 	 | ||||
| 
 | ||||
| 	def validate_pos_return(self): | ||||
| 		if self.is_pos and self.is_return: | ||||
| 			total_amount_in_payments = 0 | ||||
| @ -167,12 +167,12 @@ class POSInvoice(SalesInvoice): | ||||
| 			invoice_total = self.rounded_total or self.grand_total | ||||
| 			if total_amount_in_payments < invoice_total: | ||||
| 				frappe.throw(_("Total payments amount can't be greater than {}".format(-invoice_total))) | ||||
| 	 | ||||
| 
 | ||||
| 	def validate_loyalty_transaction(self): | ||||
| 		if self.redeem_loyalty_points and (not self.loyalty_redemption_account or not self.loyalty_redemption_cost_center): | ||||
| 			expense_account, cost_center = frappe.db.get_value('Loyalty Program', self.loyalty_program, ["expense_account", "cost_center"]) | ||||
| 			if not self.loyalty_redemption_account: | ||||
| 				self.loyalty_redemption_account = expense_account  | ||||
| 				self.loyalty_redemption_account = expense_account | ||||
| 			if not self.loyalty_redemption_cost_center: | ||||
| 				self.loyalty_redemption_cost_center = cost_center | ||||
| 
 | ||||
| @ -212,7 +212,7 @@ class POSInvoice(SalesInvoice): | ||||
| 
 | ||||
| 		if update: | ||||
| 			self.db_set('status', self.status, update_modified = update_modified) | ||||
| 	 | ||||
| 
 | ||||
| 	def set_pos_fields(self, for_validate=False): | ||||
| 		"""Set retail related fields from POS Profiles""" | ||||
| 		from erpnext.stock.get_item_details import get_pos_profile_item_details, get_pos_profile | ||||
| @ -315,25 +315,25 @@ class POSInvoice(SalesInvoice): | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| def get_stock_availability(item_code, warehouse): | ||||
| 	latest_sle = frappe.db.sql("""select qty_after_transaction  | ||||
| 		from `tabStock Ledger Entry`  | ||||
| 	latest_sle = frappe.db.sql("""select qty_after_transaction | ||||
| 		from `tabStock Ledger Entry` | ||||
| 		where item_code = %s and warehouse = %s | ||||
| 		order by posting_date desc, posting_time desc | ||||
| 		limit 1""", (item_code, warehouse), as_dict=1) | ||||
| 	 | ||||
| 
 | ||||
| 	pos_sales_qty = frappe.db.sql("""select sum(p_item.qty) as qty | ||||
| 		from `tabPOS Invoice` p, `tabPOS Invoice Item` p_item | ||||
| 		where p.name = p_item.parent  | ||||
| 		and p.consolidated_invoice is NULL  | ||||
| 		where p.name = p_item.parent | ||||
| 		and p.consolidated_invoice is NULL | ||||
| 		and p.docstatus = 1 | ||||
| 		and p_item.docstatus = 1 | ||||
| 		and p_item.item_code = %s | ||||
| 		and p_item.warehouse = %s | ||||
| 		""", (item_code, warehouse), as_dict=1) | ||||
| 	 | ||||
| 
 | ||||
| 	sle_qty = latest_sle[0].qty_after_transaction or 0 if latest_sle else 0 | ||||
| 	pos_sales_qty = pos_sales_qty[0].qty or 0 if pos_sales_qty else 0 | ||||
| 	 | ||||
| 
 | ||||
| 	if sle_qty and pos_sales_qty and sle_qty > pos_sales_qty: | ||||
| 		return sle_qty - pos_sales_qty | ||||
| 	else: | ||||
| @ -360,14 +360,14 @@ def make_merge_log(invoices): | ||||
| 	merge_log = frappe.new_doc("POS Invoice Merge Log") | ||||
| 	merge_log.posting_date = getdate(nowdate()) | ||||
| 	for inv in invoices: | ||||
| 		inv_data = frappe.db.get_values("POS Invoice", inv.get('name'),  | ||||
| 		inv_data = frappe.db.get_values("POS Invoice", inv.get('name'), | ||||
| 			["customer", "posting_date", "grand_total"], as_dict=1)[0] | ||||
| 		merge_log.customer = inv_data.customer | ||||
| 		merge_log.append("pos_invoices", { | ||||
| 			'pos_invoice': inv.get('name'), | ||||
| 			'customer': inv_data.customer, | ||||
| 			'posting_date': inv_data.posting_date, | ||||
| 			'grand_total': inv_data.grand_total  | ||||
| 			'grand_total': inv_data.grand_total | ||||
| 		}) | ||||
| 
 | ||||
| 	if merge_log.get('pos_invoices'): | ||||
|  | ||||
| @ -8,6 +8,8 @@ import unittest | ||||
| from erpnext.stock.get_item_details import get_pos_profile | ||||
| from erpnext.accounts.doctype.pos_profile.pos_profile import get_child_nodes | ||||
| 
 | ||||
| test_dependencies = ['Item'] | ||||
| 
 | ||||
| class TestPOSProfile(unittest.TestCase): | ||||
| 	def test_pos_profile(self): | ||||
| 		make_pos_profile() | ||||
| @ -88,7 +90,7 @@ def make_pos_profile(**args): | ||||
| 		"write_off_account":  args.write_off_account or "_Test Write Off - _TC", | ||||
| 		"write_off_cost_center":  args.write_off_cost_center or "_Test Write Off Cost Center - _TC" | ||||
| 	}) | ||||
| 	 | ||||
| 
 | ||||
| 	payments = [{ | ||||
| 		'mode_of_payment': 'Cash', | ||||
| 		'default': 1 | ||||
|  | ||||
| @ -1,5 +1,4 @@ | ||||
| # Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors | ||||
| # MIT License. See license.txt | ||||
| 
 | ||||
| # For license information, please see license.txt | ||||
| 
 | ||||
| @ -208,7 +207,7 @@ def get_serial_no_for_item(args): | ||||
| 
 | ||||
| def get_pricing_rule_for_item(args, price_list_rate=0, doc=None, for_validate=False): | ||||
| 	from erpnext.accounts.doctype.pricing_rule.utils import (get_pricing_rules, | ||||
| 		get_applied_pricing_rules, get_pricing_rule_items, get_product_discount_rule) | ||||
| 			get_applied_pricing_rules, get_pricing_rule_items, get_product_discount_rule) | ||||
| 
 | ||||
| 	if isinstance(doc, string_types): | ||||
| 		doc = json.loads(doc) | ||||
| @ -237,7 +236,7 @@ def get_pricing_rule_for_item(args, price_list_rate=0, doc=None, for_validate=Fa | ||||
| 
 | ||||
| 	update_args_for_pricing_rule(args) | ||||
| 
 | ||||
| 	pricing_rules = (get_applied_pricing_rules(args) | ||||
| 	pricing_rules = (get_applied_pricing_rules(args.get('pricing_rules')) | ||||
| 		if for_validate and args.get("pricing_rules") else get_pricing_rules(args, doc)) | ||||
| 
 | ||||
| 	if pricing_rules: | ||||
| @ -365,8 +364,9 @@ def set_discount_amount(rate, item_details): | ||||
| 			item_details.rate = rate | ||||
| 
 | ||||
| def remove_pricing_rule_for_item(pricing_rules, item_details, item_code=None): | ||||
| 	from erpnext.accounts.doctype.pricing_rule.utils import get_pricing_rule_items | ||||
| 	for d in json.loads(pricing_rules): | ||||
| 	from erpnext.accounts.doctype.pricing_rule.utils import (get_applied_pricing_rules, | ||||
| 		get_pricing_rule_items) | ||||
| 	for d in get_applied_pricing_rules(pricing_rules): | ||||
| 		if not d or not frappe.db.exists("Pricing Rule", d): continue | ||||
| 		pricing_rule = frappe.get_cached_doc('Pricing Rule', d) | ||||
| 
 | ||||
|  | ||||
| @ -447,9 +447,14 @@ def apply_pricing_rule_on_transaction(doc): | ||||
| 				apply_pricing_rule_for_free_items(doc, item_details.free_item_data) | ||||
| 				doc.set_missing_values() | ||||
| 
 | ||||
| def get_applied_pricing_rules(item_row): | ||||
| 	return (json.loads(item_row.get("pricing_rules")) | ||||
| 		if item_row.get("pricing_rules") else []) | ||||
| def get_applied_pricing_rules(pricing_rules): | ||||
| 	if pricing_rules: | ||||
| 		if pricing_rules.startswith('['): | ||||
| 			return json.loads(pricing_rules) | ||||
| 		else: | ||||
| 			return pricing_rules.split(',') | ||||
| 
 | ||||
| 	return [] | ||||
| 
 | ||||
| def get_product_discount_rule(pricing_rule, item_details, args=None, doc=None): | ||||
| 	free_item = pricing_rule.free_item | ||||
|  | ||||
| @ -1,5 +1,4 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "autoname": "hash", | ||||
|  "creation": "2013-05-22 12:43:10", | ||||
|  "doctype": "DocType", | ||||
| @ -82,6 +81,7 @@ | ||||
|   "item_tax_rate", | ||||
|   "bom", | ||||
|   "include_exploded_items", | ||||
|   "purchase_invoice_item", | ||||
|   "col_break6", | ||||
|   "purchase_order", | ||||
|   "po_detail", | ||||
| @ -769,12 +769,21 @@ | ||||
|    "collapsible": 1, | ||||
|    "fieldname": "col_break7", | ||||
|    "fieldtype": "Column Break" | ||||
|   }, | ||||
|   { | ||||
|    "depends_on": "eval:parent.update_stock == 1", | ||||
|    "fieldname": "purchase_invoice_item", | ||||
|    "fieldtype": "Data", | ||||
|    "ignore_user_permissions": 1, | ||||
|    "label": "Purchase Invoice Item", | ||||
|    "no_copy": 1, | ||||
|    "print_hide": 1, | ||||
|    "read_only": 1 | ||||
|   } | ||||
|  ], | ||||
|  "idx": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2020-04-22 10:37:35.103176", | ||||
|  "modified": "2020-08-20 11:48:01.398356", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Purchase Invoice Item", | ||||
|  | ||||
| @ -13,7 +13,8 @@ def get_data(): | ||||
| 			'Auto Repeat': 'reference_document', | ||||
| 		}, | ||||
| 		'internal_links': { | ||||
| 			'Sales Order': ['items', 'sales_order'] | ||||
| 			'Sales Order': ['items', 'sales_order'], | ||||
| 			'Delivery Note': ['items', 'delivery_note'] | ||||
| 		}, | ||||
| 		'transactions': [ | ||||
| 			{ | ||||
|  | ||||
| @ -1,5 +1,4 @@ | ||||
| { | ||||
|  "actions": [], | ||||
|  "autoname": "hash", | ||||
|  "creation": "2013-06-04 11:02:19", | ||||
|  "doctype": "DocType", | ||||
| @ -87,6 +86,7 @@ | ||||
|   "edit_references", | ||||
|   "sales_order", | ||||
|   "so_detail", | ||||
|   "sales_invoice_item", | ||||
|   "column_break_74", | ||||
|   "delivery_note", | ||||
|   "dn_detail", | ||||
| @ -790,12 +790,22 @@ | ||||
|    "fieldtype": "Link", | ||||
|    "label": "Project", | ||||
|    "options": "Project" | ||||
|   } | ||||
|   }, | ||||
|   { | ||||
|     "depends_on": "eval:parent.update_stock == 1", | ||||
|     "fieldname": "sales_invoice_item", | ||||
|     "fieldtype": "Data", | ||||
|     "ignore_user_permissions": 1, | ||||
|     "label": "Sales Invoice Item", | ||||
|     "no_copy": 1, | ||||
|     "print_hide": 1, | ||||
|     "read_only": 1 | ||||
|    } | ||||
|  ], | ||||
|  "idx": 1, | ||||
|  "istable": 1, | ||||
|  "links": [], | ||||
|  "modified": "2020-07-18 12:24:41.749986", | ||||
|  "modified": "2020-08-20 11:24:41.749986", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Accounts", | ||||
|  "name": "Sales Invoice Item", | ||||
|  | ||||
| @ -378,7 +378,7 @@ def set_gl_entries_by_account(from_date, to_date, root_lft, root_rgt, filters, g | ||||
| 		if filters and filters.get('presentation_currency') != d.default_currency: | ||||
| 			currency_info['company'] = d.name | ||||
| 			currency_info['company_currency'] = d.default_currency | ||||
| 			convert_to_presentation_currency(gl_entries, currency_info) | ||||
| 			convert_to_presentation_currency(gl_entries, currency_info, filters.get('company')) | ||||
| 
 | ||||
| 		for entry in gl_entries: | ||||
| 			key = entry.account_number or entry.account_name | ||||
|  | ||||
| @ -423,7 +423,7 @@ def set_gl_entries_by_account( | ||||
| 				distributed_cost_center_query=distributed_cost_center_query), gl_filters, as_dict=True) #nosec | ||||
| 
 | ||||
| 		if filters and filters.get('presentation_currency'): | ||||
| 			convert_to_presentation_currency(gl_entries, get_currency(filters)) | ||||
| 			convert_to_presentation_currency(gl_entries, get_currency(filters), filters.get('company')) | ||||
| 
 | ||||
| 		for entry in gl_entries: | ||||
| 			gl_entries_by_account.setdefault(entry.account, []).append(entry) | ||||
|  | ||||
| @ -146,6 +146,12 @@ frappe.query_reports["General Ledger"] = { | ||||
| 				return frappe.db.get_link_options('Project', txt); | ||||
| 			} | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname": "include_dimensions", | ||||
| 			"label": __("Consider Accounting Dimensions"), | ||||
| 			"fieldtype": "Check", | ||||
| 			"default": 0 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname": "show_opening_entries", | ||||
| 			"label": __("Show Opening Entries"), | ||||
|  | ||||
| @ -106,15 +106,20 @@ def set_account_currency(filters): | ||||
| 	return filters | ||||
| 
 | ||||
| def get_result(filters, account_details): | ||||
| 	gl_entries = get_gl_entries(filters) | ||||
| 	accounting_dimensions = [] | ||||
| 	if filters.get("include_dimensions"): | ||||
| 		accounting_dimensions = get_accounting_dimensions() | ||||
| 
 | ||||
| 	data = get_data_with_opening_closing(filters, account_details, gl_entries) | ||||
| 	gl_entries = get_gl_entries(filters, accounting_dimensions) | ||||
| 
 | ||||
| 	data = get_data_with_opening_closing(filters, account_details, | ||||
| 		accounting_dimensions, gl_entries) | ||||
| 
 | ||||
| 	result = get_result_as_list(data, filters) | ||||
| 
 | ||||
| 	return result | ||||
| 
 | ||||
| def get_gl_entries(filters): | ||||
| def get_gl_entries(filters, accounting_dimensions): | ||||
| 	currency_map = get_currency(filters) | ||||
| 	select_fields = """, debit, credit, debit_in_account_currency, | ||||
| 		credit_in_account_currency """ | ||||
| @ -128,6 +133,10 @@ def get_gl_entries(filters): | ||||
| 		filters['company_fb'] = frappe.db.get_value("Company", | ||||
| 			filters.get("company"), 'default_finance_book') | ||||
| 
 | ||||
| 	dimension_fields = "" | ||||
| 	if accounting_dimensions: | ||||
| 		dimension_fields = ', '.join(accounting_dimensions) + ',' | ||||
| 
 | ||||
| 	distributed_cost_center_query = "" | ||||
| 	if filters and filters.get('cost_center'): | ||||
| 		select_fields_with_percentage = """, debit*(DCC_allocation.percentage_allocation/100) as debit, credit*(DCC_allocation.percentage_allocation/100) as credit, debit_in_account_currency*(DCC_allocation.percentage_allocation/100) as debit_in_account_currency, | ||||
| @ -141,7 +150,7 @@ def get_gl_entries(filters): | ||||
| 			party_type, | ||||
| 			party, | ||||
| 			voucher_type, | ||||
| 			voucher_no, | ||||
| 			voucher_no, {dimension_fields} | ||||
| 			cost_center, project, | ||||
| 			against_voucher_type, | ||||
| 			against_voucher, | ||||
| @ -160,13 +169,14 @@ def get_gl_entries(filters): | ||||
| 		{conditions} | ||||
| 		AND posting_date <= %(to_date)s | ||||
| 		AND cost_center = DCC_allocation.parent | ||||
| 		""".format(select_fields_with_percentage=select_fields_with_percentage, conditions=get_conditions(filters).replace("and cost_center in %(cost_center)s ", '')) | ||||
| 		""".format(dimension_fields=dimension_fields,select_fields_with_percentage=select_fields_with_percentage, conditions=get_conditions(filters).replace("and cost_center in %(cost_center)s ", '')) | ||||
| 
 | ||||
| 	gl_entries = frappe.db.sql( | ||||
| 		""" | ||||
| 		select | ||||
| 			name as gl_entry, posting_date, account, party_type, party, | ||||
| 			voucher_type, voucher_no, cost_center, project, | ||||
| 			voucher_type, voucher_no, {dimension_fields} | ||||
| 			cost_center, project, | ||||
| 			against_voucher_type, against_voucher, account_currency, | ||||
| 			remarks, against, is_opening, creation {select_fields} | ||||
| 		from `tabGL Entry` | ||||
| @ -174,13 +184,13 @@ def get_gl_entries(filters): | ||||
| 		{distributed_cost_center_query} | ||||
| 		{order_by_statement} | ||||
| 		""".format( | ||||
| 			select_fields=select_fields, conditions=get_conditions(filters), distributed_cost_center_query=distributed_cost_center_query, | ||||
| 			dimension_fields=dimension_fields, select_fields=select_fields, conditions=get_conditions(filters), distributed_cost_center_query=distributed_cost_center_query, | ||||
| 			order_by_statement=order_by_statement | ||||
| 		), | ||||
| 		filters, as_dict=1) | ||||
| 
 | ||||
| 	if filters.get('presentation_currency'): | ||||
| 		return convert_to_presentation_currency(gl_entries, currency_map) | ||||
| 		return convert_to_presentation_currency(gl_entries, currency_map, filters.get('company')) | ||||
| 	else: | ||||
| 		return gl_entries | ||||
| 
 | ||||
| @ -247,12 +257,12 @@ def get_conditions(filters): | ||||
| 	return "and {}".format(" and ".join(conditions)) if conditions else "" | ||||
| 
 | ||||
| 
 | ||||
| def get_data_with_opening_closing(filters, account_details, gl_entries): | ||||
| def get_data_with_opening_closing(filters, account_details, accounting_dimensions, gl_entries): | ||||
| 	data = [] | ||||
| 
 | ||||
| 	gle_map = initialize_gle_map(gl_entries, filters) | ||||
| 
 | ||||
| 	totals, entries = get_accountwise_gle(filters, gl_entries, gle_map) | ||||
| 	totals, entries = get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map) | ||||
| 
 | ||||
| 	# Opening for filtered account | ||||
| 	data.append(totals.opening) | ||||
| @ -318,7 +328,7 @@ def initialize_gle_map(gl_entries, filters): | ||||
| 	return gle_map | ||||
| 
 | ||||
| 
 | ||||
| def get_accountwise_gle(filters, gl_entries, gle_map): | ||||
| def get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map): | ||||
| 	totals = get_totals_dict() | ||||
| 	entries = [] | ||||
| 	consolidated_gle = OrderedDict() | ||||
| @ -350,8 +360,11 @@ def get_accountwise_gle(filters, gl_entries, gle_map): | ||||
| 			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)'): | ||||
| 				key = (gle.get("voucher_type"), gle.get("voucher_no"), | ||||
| 					gle.get("account"), gle.get("cost_center")) | ||||
| 				keylist = [gle.get("voucher_type"), gle.get("voucher_no"), gle.get("account")] | ||||
| 				for dim in accounting_dimensions: | ||||
| 					keylist.append(gle.get(dim)) | ||||
| 				keylist.append(gle.get("cost_center")) | ||||
| 				key = tuple(keylist) | ||||
| 				if key not in consolidated_gle: | ||||
| 					consolidated_gle.setdefault(key, gle) | ||||
| 				else: | ||||
| @ -478,7 +491,19 @@ def get_columns(filters): | ||||
| 			"options": "Project", | ||||
| 			"fieldname": "project", | ||||
| 			"width": 100 | ||||
| 		}, | ||||
| 		} | ||||
| 	]) | ||||
| 
 | ||||
| 	if filters.get("include_dimensions"): | ||||
| 		for dim in get_accounting_dimensions(as_list = False): | ||||
| 			columns.append({ | ||||
| 				"label": _(dim.label), | ||||
| 				"options": dim.label, | ||||
| 				"fieldname": dim.fieldname, | ||||
| 				"width": 100 | ||||
| 			}) | ||||
| 
 | ||||
| 	columns.extend([ | ||||
| 		{ | ||||
| 			"label": _("Cost Center"), | ||||
| 			"options": "Cost Center", | ||||
|  | ||||
| @ -6,10 +6,6 @@ from erpnext.accounts.doctype.fiscal_year.fiscal_year import get_from_and_to_dat | ||||
| from frappe.utils import cint, get_datetime_str, formatdate, flt | ||||
| 
 | ||||
| __exchange_rates = {} | ||||
| P_OR_L_ACCOUNTS = list( | ||||
| 	sum(frappe.get_list('Account', fields=['name'], or_filters=[{'root_type': 'Income'}, {'root_type': 'Expense'}], as_list=True), ()) | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| def get_currency(filters): | ||||
| 	""" | ||||
| @ -73,18 +69,7 @@ def get_rate_as_at(date, from_currency, to_currency): | ||||
| 
 | ||||
| 	return rate | ||||
| 
 | ||||
| 
 | ||||
| def is_p_or_l_account(account_name): | ||||
| 	""" | ||||
| 	Check if the given `account name` is an `Account` with `root_type` of either 'Income' | ||||
| 	or 'Expense'. | ||||
| 	:param account_name: | ||||
| 	:return: Boolean | ||||
| 	""" | ||||
| 	return account_name in P_OR_L_ACCOUNTS | ||||
| 
 | ||||
| 
 | ||||
| def convert_to_presentation_currency(gl_entries, currency_info): | ||||
| def convert_to_presentation_currency(gl_entries, currency_info, company): | ||||
| 	""" | ||||
| 	Take a list of GL Entries and change the 'debit' and 'credit' values to currencies | ||||
| 	in `currency_info`. | ||||
| @ -96,6 +81,9 @@ def convert_to_presentation_currency(gl_entries, currency_info): | ||||
| 	presentation_currency = currency_info['presentation_currency'] | ||||
| 	company_currency = currency_info['company_currency'] | ||||
| 
 | ||||
| 	pl_accounts = [d.name for d in frappe.get_list('Account', | ||||
| 		filters={'report_type': 'Profit and Loss', 'company': company})] | ||||
| 
 | ||||
| 	for entry in gl_entries: | ||||
| 		account = entry['account'] | ||||
| 		debit = flt(entry['debit']) | ||||
| @ -107,7 +95,7 @@ def convert_to_presentation_currency(gl_entries, currency_info): | ||||
| 		if account_currency != presentation_currency: | ||||
| 			value = debit or credit | ||||
| 
 | ||||
| 			date = currency_info['report_date'] if not is_p_or_l_account(account) else entry['posting_date'] | ||||
| 			date = entry['posting_date'] if account in pl_accounts else currency_info['report_date'] | ||||
| 			converted_value = convert(value, presentation_currency, company_currency, date) | ||||
| 
 | ||||
| 			if entry.get('debit'): | ||||
|  | ||||
| @ -325,7 +325,7 @@ class AccountsController(TransactionBase): | ||||
| 				apply_pricing_rule_for_free_items(self, pricing_rule_args.get('free_item_data')) | ||||
| 
 | ||||
| 		elif pricing_rule_args.get("validate_applied_rule"): | ||||
| 			for pricing_rule in get_applied_pricing_rules(item): | ||||
| 			for pricing_rule in get_applied_pricing_rules(item.get('pricing_rules')): | ||||
| 				pricing_rule_doc = frappe.get_cached_doc("Pricing Rule", pricing_rule) | ||||
| 				for field in ['discount_percentage', 'discount_amount', 'rate']: | ||||
| 					if item.get(field) < pricing_rule_doc.get(field): | ||||
|  | ||||
| @ -559,9 +559,19 @@ class BuyingController(StockController): | ||||
| 						"serial_no": cstr(d.serial_no).strip() | ||||
| 					}) | ||||
| 					if self.is_return: | ||||
| 						original_incoming_rate = frappe.db.get_value("Stock Ledger Entry", | ||||
| 							{"voucher_type": "Purchase Receipt", "voucher_no": self.return_against, | ||||
| 							"item_code": d.item_code}, "incoming_rate") | ||||
| 						filters = { | ||||
| 							"voucher_type": self.doctype, | ||||
| 							"voucher_no": self.return_against, | ||||
| 							"item_code": d.item_code | ||||
| 						} | ||||
| 
 | ||||
| 						if (self.doctype == "Purchase Invoice" and self.update_stock | ||||
| 							and d.get("purchase_invoice_item")): | ||||
| 							filters["voucher_detail_no"] = d.purchase_invoice_item | ||||
| 						elif self.doctype == "Purchase Receipt" and d.get("purchase_receipt_item"): | ||||
| 							filters["voucher_detail_no"] = d.purchase_receipt_item | ||||
| 
 | ||||
| 						original_incoming_rate = frappe.db.get_value("Stock Ledger Entry", filters, "incoming_rate") | ||||
| 
 | ||||
| 						sle.update({ | ||||
| 							"outgoing_rate": original_incoming_rate | ||||
|  | ||||
| @ -497,24 +497,18 @@ def warehouse_query(doctype, txt, searchfield, start, page_len, filters): | ||||
| 	conditions, bin_conditions = [], [] | ||||
| 	filter_dict = get_doctype_wise_filters(filters) | ||||
| 
 | ||||
| 	sub_query = """ select round(`tabBin`.actual_qty, 2) from `tabBin` | ||||
| 		where `tabBin`.warehouse = `tabWarehouse`.name | ||||
| 		{bin_conditions} """.format( | ||||
| 		bin_conditions=get_filters_cond(doctype, filter_dict.get("Bin"), | ||||
| 			bin_conditions, ignore_permissions=True)) | ||||
| 
 | ||||
| 	query = """select `tabWarehouse`.name, | ||||
| 		CONCAT_WS(" : ", "Actual Qty", ifnull( ({sub_query}), 0) ) as actual_qty | ||||
| 		from `tabWarehouse` | ||||
| 		CONCAT_WS(" : ", "Actual Qty", ifnull(round(`tabBin`.actual_qty, 2), 0 )) actual_qty | ||||
| 		from `tabWarehouse` left join `tabBin` | ||||
| 		on `tabBin`.warehouse = `tabWarehouse`.name {bin_conditions} | ||||
| 		where | ||||
| 		   `tabWarehouse`.`{key}` like {txt} | ||||
| 			`tabWarehouse`.`{key}` like {txt} | ||||
| 			{fcond} {mcond} | ||||
| 		order by | ||||
| 			`tabWarehouse`.name desc | ||||
| 		order by ifnull(`tabBin`.actual_qty, 0) desc | ||||
| 		limit | ||||
| 			{start}, {page_len} | ||||
| 		""".format( | ||||
| 			sub_query=sub_query, | ||||
| 			bin_conditions=get_filters_cond(doctype, filter_dict.get("Bin"),bin_conditions, ignore_permissions=True), | ||||
| 			key=searchfield, | ||||
| 			fcond=get_filters_cond(doctype, filter_dict.get("Warehouse"), conditions), | ||||
| 			mcond=get_match_cond(doctype), | ||||
|  | ||||
| @ -281,6 +281,8 @@ def make_return_doc(doctype, source_name, target_doc=None): | ||||
| 			target_doc.rejected_warehouse = source_doc.rejected_warehouse | ||||
| 			target_doc.po_detail = source_doc.po_detail | ||||
| 			target_doc.pr_detail = source_doc.pr_detail | ||||
| 			target_doc.purchase_invoice_item = source_doc.name | ||||
| 
 | ||||
| 		elif doctype == "Delivery Note": | ||||
| 			target_doc.against_sales_order = source_doc.against_sales_order | ||||
| 			target_doc.against_sales_invoice = source_doc.against_sales_invoice | ||||
| @ -296,6 +298,7 @@ def make_return_doc(doctype, source_name, target_doc=None): | ||||
| 			target_doc.so_detail = source_doc.so_detail | ||||
| 			target_doc.dn_detail = source_doc.dn_detail | ||||
| 			target_doc.expense_account = source_doc.expense_account | ||||
| 			target_doc.sales_invoice_item = source_doc.name | ||||
| 			if default_warehouse_for_sales_return: | ||||
| 				target_doc.warehouse = default_warehouse_for_sales_return | ||||
| 
 | ||||
|  | ||||
| @ -217,7 +217,9 @@ class SellingController(StockController): | ||||
| 							'target_warehouse': p.target_warehouse, | ||||
| 							'company': self.company, | ||||
| 							'voucher_type': self.doctype, | ||||
| 							'allow_zero_valuation': d.allow_zero_valuation_rate | ||||
| 							'allow_zero_valuation': d.allow_zero_valuation_rate, | ||||
| 							'sales_invoice_item': d.get("sales_invoice_item"), | ||||
| 							'delivery_note_item': d.get("dn_detail") | ||||
| 						})) | ||||
| 			else: | ||||
| 				il.append(frappe._dict({ | ||||
| @ -233,7 +235,9 @@ class SellingController(StockController): | ||||
| 					'target_warehouse': d.target_warehouse, | ||||
| 					'company': self.company, | ||||
| 					'voucher_type': self.doctype, | ||||
| 					'allow_zero_valuation': d.allow_zero_valuation_rate | ||||
| 					'allow_zero_valuation': d.allow_zero_valuation_rate, | ||||
| 					'sales_invoice_item': d.get("sales_invoice_item"), | ||||
| 					'delivery_note_item': d.get("dn_detail") | ||||
| 				})) | ||||
| 		return il | ||||
| 
 | ||||
| @ -302,7 +306,11 @@ class SellingController(StockController): | ||||
| 					d.conversion_factor = get_conversion_factor(d.item_code, d.uom).get("conversion_factor") or 1.0 | ||||
| 				return_rate = 0 | ||||
| 				if cint(self.is_return) and self.return_against and self.docstatus==1: | ||||
| 					return_rate = self.get_incoming_rate_for_return(d.item_code, self.return_against) | ||||
| 					against_document_no = (d.get("sales_invoice_item") | ||||
| 						if self.doctype == "Sales Invoice" else d.get("delivery_note_item")) | ||||
| 
 | ||||
| 					return_rate = self.get_incoming_rate_for_return(d.item_code, | ||||
| 						self.return_against, against_document_no) | ||||
| 
 | ||||
| 				# On cancellation or if return entry submission, make stock ledger entry for | ||||
| 				# target warehouse first, to update serial no values properly | ||||
|  | ||||
| @ -301,14 +301,19 @@ class StockController(AccountsController): | ||||
| 
 | ||||
| 		return serialized_items | ||||
| 
 | ||||
| 	def get_incoming_rate_for_return(self, item_code, against_document): | ||||
| 	def get_incoming_rate_for_return(self, item_code, against_document, against_document_no=None): | ||||
| 		incoming_rate = 0.0 | ||||
| 		cond = '' | ||||
| 		if against_document and item_code: | ||||
| 			if against_document_no: | ||||
| 				cond = " and voucher_detail_no = %s" %(frappe.db.escape(against_document_no)) | ||||
| 
 | ||||
| 			incoming_rate = frappe.db.sql("""select abs(stock_value_difference / actual_qty) | ||||
| 				from `tabStock Ledger Entry` | ||||
| 				where voucher_type = %s and voucher_no = %s | ||||
| 					and item_code = %s limit 1""", | ||||
| 					and item_code = %s {0} limit 1""".format(cond), | ||||
| 				(self.doctype, against_document, item_code)) | ||||
| 
 | ||||
| 			incoming_rate = incoming_rate[0][0] if incoming_rate else 0.0 | ||||
| 
 | ||||
| 		return incoming_rate | ||||
|  | ||||
| @ -325,7 +325,7 @@ def auto_close_opportunity(): | ||||
| 		doc.save() | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| def make_opportunity_from_communication(communication, ignore_communication_links=False): | ||||
| def make_opportunity_from_communication(communication, company, ignore_communication_links=False): | ||||
| 	from erpnext.crm.doctype.lead.lead import make_lead_from_communication | ||||
| 	doc = frappe.get_doc("Communication", communication) | ||||
| 
 | ||||
| @ -337,6 +337,7 @@ def make_opportunity_from_communication(communication, ignore_communication_link | ||||
| 
 | ||||
| 	opportunity = frappe.get_doc({ | ||||
| 		"doctype": "Opportunity", | ||||
| 		"company": company, | ||||
| 		"opportunity_from": opportunity_from, | ||||
| 		"party_name": lead | ||||
| 	}).insert(ignore_permissions=True) | ||||
|  | ||||
| @ -30,14 +30,14 @@ frappe.ui.form.on('Social Media Post', { | ||||
|                 let color = frm.doc.twitter_post_id ? "green" : "red"; | ||||
|                 let status = frm.doc.twitter_post_id ? "Posted" : "Not Posted"; | ||||
|                 html += `<div class="col-xs-6">
 | ||||
|                             <span class="indicator whitespace-nowrap ${color}"><span class="hidden-xs">Twitter : ${status} </span></span> | ||||
|                             <span class="indicator whitespace-nowrap ${color}"><span>Twitter : ${status} </span></span> | ||||
|                         </div>` ; | ||||
|             } | ||||
|             if (frm.doc.linkedin){ | ||||
|                 let color = frm.doc.linkedin_post_id ? "green" : "red"; | ||||
|                 let status = frm.doc.linkedin_post_id ? "Posted" : "Not Posted"; | ||||
|                 html += `<div class="col-xs-6">
 | ||||
|                             <span class="indicator whitespace-nowrap ${color}"><span class="hidden-xs">LinkedIn : ${status} </span></span> | ||||
|                             <span class="indicator whitespace-nowrap ${color}"><span>LinkedIn : ${status} </span></span> | ||||
|                         </div>` ; | ||||
|             } | ||||
|             html = `<div class="row">${html}</div>`; | ||||
|  | ||||
							
								
								
									
										52
									
								
								erpnext/crm/report/lead_details/lead_details.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								erpnext/crm/report/lead_details/lead_details.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,52 @@ | ||||
| // Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
 | ||||
| // For license information, please see license.txt
 | ||||
| /* eslint-disable */ | ||||
| 
 | ||||
| frappe.query_reports["Lead Details"] = { | ||||
| 	"filters": [ | ||||
| 		{ | ||||
| 			"fieldname":"company", | ||||
| 			"label": __("Company"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Company", | ||||
| 			"default": frappe.defaults.get_user_default("Company"), | ||||
| 			"reqd": 1 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"from_date", | ||||
| 			"label": __("From Date"), | ||||
| 			"fieldtype": "Date", | ||||
| 			"default": frappe.datetime.add_months(frappe.datetime.get_today(), -12), | ||||
| 			"reqd": 1 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"to_date", | ||||
| 			"label": __("To Date"), | ||||
| 			"fieldtype": "Date", | ||||
| 			"default": frappe.datetime.get_today(), | ||||
| 			"reqd": 1 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"status", | ||||
| 			"label": __("Status"), | ||||
| 			"fieldtype": "Select", | ||||
| 			options: [ | ||||
| 				{ "value": "Lead", "label": __("Lead") }, | ||||
| 				{ "value": "Open", "label": __("Open") }, | ||||
| 				{ "value": "Replied", "label": __("Replied") }, | ||||
| 				{ "value": "Opportunity", "label": __("Opportunity") }, | ||||
| 				{ "value": "Quotation", "label": __("Quotation") }, | ||||
| 				{ "value": "Lost Quotation", "label": __("Lost Quotation") }, | ||||
| 				{ "value": "Interested", "label": __("Interested") }, | ||||
| 				{ "value": "Converted", "label": __("Converted") }, | ||||
| 				{ "value": "Do Not Contact", "label": __("Do Not Contact") }, | ||||
| 			], | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"territory", | ||||
| 			"label": __("Territory"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Territory", | ||||
| 		} | ||||
| 	] | ||||
| }; | ||||
| @ -7,16 +7,15 @@ | ||||
|  "doctype": "Report", | ||||
|  "idx": 3, | ||||
|  "is_standard": "Yes", | ||||
|  "modified": "2020-01-22 16:51:56.591110", | ||||
|  "modified": "2020-07-26 23:59:49.897577", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "CRM", | ||||
|  "name": "Lead Details", | ||||
|  "owner": "Administrator", | ||||
|  "prepared_report": 0, | ||||
|  "query": "SELECT\n    `tabLead`.name as \"Lead Id:Link/Lead:120\",\n    `tabLead`.lead_name as \"Lead Name::120\",\n\t`tabLead`.company_name as \"Company Name::120\",\n\t`tabLead`.status as \"Status::120\",\n\tconcat_ws(', ', \n\t\ttrim(',' from `tabAddress`.address_line1), \n\t\ttrim(',' from tabAddress.address_line2)\n\t) as 'Address::180',\n\t`tabAddress`.state as \"State::100\",\n\t`tabAddress`.pincode as \"Pincode::70\",\n\t`tabAddress`.country as \"Country::100\",\n\t`tabLead`.phone as \"Phone::100\",\n\t`tabLead`.mobile_no as \"Mobile No::100\",\n\t`tabLead`.email_id as \"Email Id::120\",\n\t`tabLead`.lead_owner as \"Lead Owner::120\",\n\t`tabLead`.source as \"Source::120\",\n\t`tabLead`.territory as \"Territory::120\",\n\t`tabLead`.notes as \"Notes::360\",\n    `tabLead`.owner as \"Owner:Link/User:120\"\nFROM\n\t`tabLead`\n\tleft join `tabDynamic Link` on (\n\t\t`tabDynamic Link`.link_name=`tabLead`.name \n\t\tand `tabDynamic Link`.parenttype = 'Address'\n\t)\n\tleft join `tabAddress` on (\n\t\t`tabAddress`.name=`tabDynamic Link`.parent\n\t)\nWHERE\n\t`tabLead`.docstatus<2\nORDER BY\n\t`tabLead`.name asc", | ||||
|  "ref_doctype": "Lead", | ||||
|  "report_name": "Lead Details", | ||||
|  "report_type": "Query Report", | ||||
|  "report_type": "Script Report", | ||||
|  "roles": [ | ||||
|   { | ||||
|    "role": "Sales User" | ||||
|  | ||||
							
								
								
									
										158
									
								
								erpnext/crm/report/lead_details/lead_details.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										158
									
								
								erpnext/crm/report/lead_details/lead_details.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,158 @@ | ||||
| # Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors | ||||
| # For license information, please see license.txt | ||||
| 
 | ||||
| from __future__ import unicode_literals | ||||
| from frappe import _ | ||||
| import frappe | ||||
| 
 | ||||
| def execute(filters=None): | ||||
| 	columns, data = get_columns(), get_data(filters) | ||||
| 	return columns, data | ||||
| 
 | ||||
| def get_columns(): | ||||
| 	columns = [ | ||||
| 		{ | ||||
| 			"label": _("Lead"), | ||||
| 			"fieldname": "name", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Lead", | ||||
| 			"width": 150, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Lead Name"), | ||||
| 			"fieldname": "lead_name", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"status", | ||||
| 			"label": _("Status"), | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 100 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"lead_owner", | ||||
| 			"label": _("Lead Owner"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "User", | ||||
| 			"width": 100 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Territory"), | ||||
| 			"fieldname": "territory", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Territory", | ||||
| 			"width": 100 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Source"), | ||||
| 			"fieldname": "source", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Email"), | ||||
| 			"fieldname": "email_id", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Mobile"), | ||||
| 			"fieldname": "mobile_no", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Phone"), | ||||
| 			"fieldname": "phone", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Owner"), | ||||
| 			"fieldname": "owner", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "user", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Company"), | ||||
| 			"fieldname": "company", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Company", | ||||
| 			"width": 120 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"address", | ||||
| 			"label": _("Address"), | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 130 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"state", | ||||
| 			"label": _("State"), | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 100 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"pincode", | ||||
| 			"label": _("Postal Code"), | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 90 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"country", | ||||
| 			"label": _("Country"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Country", | ||||
| 			"width": 100 | ||||
| 		}, | ||||
| 		 | ||||
| 	] | ||||
| 	return columns | ||||
| 
 | ||||
| def get_data(filters): | ||||
| 	return frappe.db.sql(""" | ||||
| 		SELECT | ||||
| 			`tabLead`.name, | ||||
| 			`tabLead`.lead_name, | ||||
| 			`tabLead`.status, | ||||
| 			`tabLead`.lead_owner, | ||||
| 			`tabLead`.territory, | ||||
| 			`tabLead`.source, | ||||
| 			`tabLead`.email_id, | ||||
| 			`tabLead`.mobile_no, | ||||
| 			`tabLead`.phone, | ||||
| 			`tabLead`.owner, | ||||
| 			`tabLead`.company, | ||||
| 			concat_ws(', ', | ||||
| 				trim(',' from `tabAddress`.address_line1), | ||||
| 				trim(',' from tabAddress.address_line2) | ||||
| 			) AS address, | ||||
| 			`tabAddress`.state, | ||||
| 			`tabAddress`.pincode, | ||||
| 			`tabAddress`.country | ||||
| 		FROM | ||||
| 			`tabLead` left join `tabDynamic Link` on ( | ||||
| 			`tabLead`.name = `tabDynamic Link`.link_name and | ||||
| 			`tabDynamic Link`.parenttype = 'Address') | ||||
| 			left join `tabAddress` on ( | ||||
| 			`tabAddress`.name=`tabDynamic Link`.parent) | ||||
| 		WHERE | ||||
| 			company = %(company)s | ||||
| 			AND `tabLead`.creation BETWEEN %(from_date)s AND %(to_date)s | ||||
| 			{conditions} | ||||
| 		ORDER BY  | ||||
| 			`tabLead`.creation asc """.format(conditions=get_conditions(filters)), filters, as_dict=1) | ||||
| 
 | ||||
| def get_conditions(filters) : | ||||
| 	conditions = [] | ||||
| 
 | ||||
| 	if filters.get("territory"): | ||||
| 		conditions.append(" and `tabLead`.territory=%(territory)s") | ||||
| 
 | ||||
| 	if filters.get("status"): | ||||
| 		conditions.append(" and `tabLead`.status=%(status)s") | ||||
| 	 | ||||
| 	return " ".join(conditions) if conditions else "" | ||||
| 
 | ||||
							
								
								
									
										67
									
								
								erpnext/crm/report/lost_opportunity/lost_opportunity.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								erpnext/crm/report/lost_opportunity/lost_opportunity.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | ||||
| // Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
 | ||||
| // For license information, please see license.txt
 | ||||
| /* eslint-disable */ | ||||
| 
 | ||||
| frappe.query_reports["Lost Opportunity"] = { | ||||
| 	"filters": [ | ||||
| 		{ | ||||
| 			"fieldname":"company", | ||||
| 			"label": __("Company"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Company", | ||||
| 			"default": frappe.defaults.get_user_default("Company"), | ||||
| 			"reqd": 1 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"from_date", | ||||
| 			"label": __("From Date"), | ||||
| 			"fieldtype": "Date", | ||||
| 			"default": frappe.datetime.add_months(frappe.datetime.get_today(), -12), | ||||
| 			"reqd": 1 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"to_date", | ||||
| 			"label": __("To Date"), | ||||
| 			"fieldtype": "Date", | ||||
| 			"default": frappe.datetime.get_today(), | ||||
| 			"reqd": 1 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"lost_reason", | ||||
| 			"label": __("Lost Reason"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Opportunity Lost Reason" | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"territory", | ||||
| 			"label": __("Territory"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Territory" | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"opportunity_from", | ||||
| 			"label": __("Opportunity From"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "DocType", | ||||
| 			"get_query": function() { | ||||
| 				return { | ||||
| 					"filters": { | ||||
| 						"name": ["in", ["Customer", "Lead"]], | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"party_name", | ||||
| 			"label": __("Party"), | ||||
| 			"fieldtype": "Dynamic Link", | ||||
| 			"options": "opportunity_from" | ||||
| 		}, | ||||
| 		{ | ||||
| 			"fieldname":"contact_by", | ||||
| 			"label": __("Next Contact By"), | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "User" | ||||
| 		}, | ||||
| 	] | ||||
| }; | ||||
| @ -1,13 +1,14 @@ | ||||
| { | ||||
|  "add_total_row": 0, | ||||
|  "creation": "2018-12-31 16:30:57.188837", | ||||
|  "disable_prepared_report": 0, | ||||
|  "disabled": 0, | ||||
|  "docstatus": 0, | ||||
|  "doctype": "Report", | ||||
|  "idx": 0, | ||||
|  "is_standard": "Yes", | ||||
|  "json": "{\"order_by\": \"`tabOpportunity`.`modified` desc\", \"filters\": [[\"Opportunity\", \"status\", \"=\", \"Lost\"]], \"fields\": [[\"name\", \"Opportunity\"], [\"opportunity_from\", \"Opportunity\"], [\"party_name\", \"Opportunity\"], [\"customer_name\", \"Opportunity\"], [\"opportunity_type\", \"Opportunity\"], [\"status\", \"Opportunity\"], [\"contact_by\", \"Opportunity\"], [\"docstatus\", \"Opportunity\"], [\"lost_reason\", \"Lost Reason Detail\"]], \"add_totals_row\": 0, \"add_total_row\": 0, \"page_length\": 20}", | ||||
|  "modified": "2019-06-26 16:33:08.083618", | ||||
|  "modified": "2020-07-29 15:49:02.848845", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "CRM", | ||||
|  "name": "Lost Opportunity", | ||||
| @ -15,7 +16,7 @@ | ||||
|  "prepared_report": 0, | ||||
|  "ref_doctype": "Opportunity", | ||||
|  "report_name": "Lost Opportunity", | ||||
|  "report_type": "Report Builder", | ||||
|  "report_type": "Script Report", | ||||
|  "roles": [ | ||||
|   { | ||||
|    "role": "Sales User" | ||||
|  | ||||
							
								
								
									
										131
									
								
								erpnext/crm/report/lost_opportunity/lost_opportunity.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								erpnext/crm/report/lost_opportunity/lost_opportunity.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,131 @@ | ||||
| # Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors | ||||
| # For license information, please see license.txt | ||||
| 
 | ||||
| from __future__ import unicode_literals | ||||
| from frappe import _ | ||||
| import frappe | ||||
| 
 | ||||
| def execute(filters=None): | ||||
| 	columns, data = get_columns(), get_data(filters) | ||||
| 	return columns, data | ||||
| 
 | ||||
| def get_columns(): | ||||
| 	columns = [ | ||||
| 		{ | ||||
| 			"label": _("Opportunity"), | ||||
| 			"fieldname": "name", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Opportunity", | ||||
| 			"width": 170, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Opportunity From"), | ||||
| 			"fieldname": "opportunity_from", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "DocType", | ||||
| 			"width": 130 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Party"), | ||||
| 			"fieldname":"party_name", | ||||
| 			"fieldtype": "Dynamic Link", | ||||
| 			"options": "opportunity_from", | ||||
| 			"width": 160 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Customer/Lead Name"), | ||||
| 			"fieldname":"customer_name", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 150 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Opportunity Type"), | ||||
| 			"fieldname": "opportunity_type", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 130 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Lost Reasons"), | ||||
| 			"fieldname": "lost_reason", | ||||
| 			"fieldtype": "Data", | ||||
| 			"width": 220 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Sales Stage"), | ||||
| 			"fieldname": "sales_stage", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Sales Stage", | ||||
| 			"width": 150 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Territory"), | ||||
| 			"fieldname": "territory", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "Territory", | ||||
| 			"width": 150 | ||||
| 		}, | ||||
| 		{ | ||||
| 			"label": _("Next Contact By"), | ||||
| 			"fieldname": "contact_by", | ||||
| 			"fieldtype": "Link", | ||||
| 			"options": "User", | ||||
| 			"width": 150 | ||||
| 		} | ||||
| 	] | ||||
| 	return columns | ||||
| 
 | ||||
| def get_data(filters): | ||||
| 	return frappe.db.sql(""" | ||||
| 		SELECT | ||||
| 			`tabOpportunity`.name, | ||||
| 			`tabOpportunity`.opportunity_from, | ||||
| 			`tabOpportunity`.party_name, | ||||
| 			`tabOpportunity`.customer_name, | ||||
| 			`tabOpportunity`.opportunity_type, | ||||
| 			`tabOpportunity`.contact_by, | ||||
| 			GROUP_CONCAT(`tabOpportunity Lost Reason Detail`.lost_reason separator ', ') lost_reason, | ||||
| 			`tabOpportunity`.sales_stage, | ||||
| 			`tabOpportunity`.territory | ||||
| 		FROM | ||||
| 			`tabOpportunity`  | ||||
| 			{join} | ||||
| 		WHERE | ||||
| 			`tabOpportunity`.status = 'Lost' and `tabOpportunity`.company = %(company)s | ||||
| 			AND `tabOpportunity`.modified BETWEEN %(from_date)s AND %(to_date)s  | ||||
| 			{conditions}  | ||||
| 		GROUP BY  | ||||
| 			`tabOpportunity`.name  | ||||
| 		ORDER BY  | ||||
| 			`tabOpportunity`.creation asc  """.format(conditions=get_conditions(filters), join=get_join(filters)), filters, as_dict=1) | ||||
| 		 | ||||
| 
 | ||||
| def get_conditions(filters): | ||||
| 	conditions = [] | ||||
| 
 | ||||
| 	if filters.get("territory"): | ||||
| 		conditions.append(" and `tabOpportunity`.territory=%(territory)s") | ||||
| 
 | ||||
| 	if filters.get("opportunity_from"): | ||||
| 		conditions.append(" and `tabOpportunity`.opportunity_from=%(opportunity_from)s") | ||||
| 
 | ||||
| 	if filters.get("party_name"): | ||||
| 		conditions.append(" and `tabOpportunity`.party_name=%(party_name)s") | ||||
| 
 | ||||
| 	if filters.get("contact_by"): | ||||
| 		conditions.append(" and `tabOpportunity`.contact_by=%(contact_by)s") | ||||
| 
 | ||||
| 	return " ".join(conditions) if conditions else "" | ||||
| 
 | ||||
| def get_join(filters): | ||||
| 	join = """LEFT JOIN `tabOpportunity Lost Reason Detail`  | ||||
| 			ON `tabOpportunity Lost Reason Detail`.parenttype = 'Opportunity' and  | ||||
| 			`tabOpportunity Lost Reason Detail`.parent = `tabOpportunity`.name""" | ||||
| 
 | ||||
| 	if filters.get("lost_reason"): | ||||
| 		join = """JOIN `tabOpportunity Lost Reason Detail`  | ||||
| 			ON `tabOpportunity Lost Reason Detail`.parenttype = 'Opportunity' and  | ||||
| 			`tabOpportunity Lost Reason Detail`.parent = `tabOpportunity`.name and | ||||
| 			`tabOpportunity Lost Reason Detail`.lost_reason = '{0}' | ||||
| 			""".format(filters.get("lost_reason")) | ||||
| 	 | ||||
| 	return join | ||||
| @ -7,6 +7,8 @@ import unittest | ||||
| import frappe | ||||
| from erpnext.healthcare.doctype.patient_appointment.test_patient_appointment import create_healthcare_docs, create_clinical_procedure_template | ||||
| 
 | ||||
| test_dependencies = ['Item'] | ||||
| 
 | ||||
| class TestClinicalProcedure(unittest.TestCase): | ||||
| 	def test_procedure_template_item(self): | ||||
| 		patient, medical_department, practitioner = create_healthcare_docs() | ||||
|  | ||||
| @ -3,6 +3,7 @@ | ||||
| 
 | ||||
| from __future__ import unicode_literals | ||||
| import frappe | ||||
| from frappe.utils import cint | ||||
| from frappe.model.document import Document | ||||
| from frappe.model.mapper import get_mapped_doc | ||||
| from frappe import _ | ||||
| @ -24,8 +25,7 @@ class JobOffer(Document): | ||||
| 		check_vacancies = frappe.get_single("HR Settings").check_vacancies | ||||
| 		if staffing_plan and check_vacancies: | ||||
| 			job_offers = self.get_job_offer(staffing_plan.from_date, staffing_plan.to_date) | ||||
| 
 | ||||
| 			if not staffing_plan.get("vacancies") or staffing_plan.vacancies - len(job_offers) <= 0: | ||||
| 			if not staffing_plan.get("vacancies") or cint(staffing_plan.vacancies) - len(job_offers) <= 0: | ||||
| 				error_variable = 'for ' + frappe.bold(self.designation) | ||||
| 				if staffing_plan.get("parent"): | ||||
| 					error_variable = frappe.bold(get_link_to_form("Staffing Plan", staffing_plan.parent)) | ||||
| @ -65,7 +65,7 @@ def get_staffing_plan_detail(designation, company, offer_date): | ||||
| 			AND %s between sp.from_date and sp.to_date | ||||
| 	""", (designation, company, offer_date), as_dict=1) | ||||
| 
 | ||||
| 	return frappe._dict(detail[0]) if detail else None | ||||
| 	return frappe._dict(detail[0]) if (detail and detail[0].parent) else None | ||||
| 
 | ||||
| @frappe.whitelist() | ||||
| def make_employee(source_name, target_doc=None): | ||||
|  | ||||
| @ -7,7 +7,7 @@ frappe.ui.form.on("Communication", { | ||||
| 	}, | ||||
| 
 | ||||
| 	setup_custom_buttons: (frm) => { | ||||
| 		let confirm_msg = "Are you sure you want to create {0} from this email"; | ||||
| 		let confirm_msg = "Are you sure you want to create {0} from this email?"; | ||||
| 		if(frm.doc.reference_doctype !== "Issue") { | ||||
| 			frm.add_custom_button(__("Issue"), () => { | ||||
| 				frappe.confirm(__(confirm_msg, [__("Issue")]), () => { | ||||
| @ -62,17 +62,36 @@ frappe.ui.form.on("Communication", { | ||||
| 	}, | ||||
| 
 | ||||
| 	make_opportunity_from_communication: (frm) => { | ||||
| 		return frappe.call({ | ||||
| 			method: "erpnext.crm.doctype.opportunity.opportunity.make_opportunity_from_communication", | ||||
| 			args: { | ||||
| 				communication: frm.doc.name | ||||
| 			}, | ||||
| 			freeze: true, | ||||
| 			callback: (r) => { | ||||
| 				if(r.message) { | ||||
| 					frm.reload_doc() | ||||
| 		const fields = [{ | ||||
| 			fieldtype: 'Link', | ||||
| 			label: __('Select a Company'), | ||||
| 			fieldname: 'company', | ||||
| 			options: 'Company', | ||||
| 			reqd: 1, | ||||
| 			default: frappe.defaults.get_user_default("Company") | ||||
| 		}]; | ||||
| 
 | ||||
| 		frappe.prompt(fields, data => { | ||||
| 			frappe.call({ | ||||
| 				method: "erpnext.crm.doctype.opportunity.opportunity.make_opportunity_from_communication", | ||||
| 				args: { | ||||
| 					communication: frm.doc.name, | ||||
| 					company: data.company | ||||
| 				}, | ||||
| 				freeze: true, | ||||
| 				callback: (r) => { | ||||
| 					if(r.message) { | ||||
| 						frm.reload_doc(); | ||||
| 						frappe.show_alert({ | ||||
| 							message: __("Opportunity {0} created", | ||||
| 								['<a href="#Form/Opportunity/'+r.message+'">' + r.message + '</a>']), | ||||
| 							indicator: 'green' | ||||
| 						}); | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}) | ||||
| 			}); | ||||
| 		}, | ||||
| 		'Create an Opportunity', | ||||
| 		'Create'); | ||||
| 	} | ||||
| }); | ||||
| }); | ||||
|  | ||||
| @ -22,6 +22,8 @@ | ||||
| } | ||||
| 
 | ||||
| .filter-options { | ||||
| 	margin-left: -5px; | ||||
| 	padding-left: 5px; | ||||
| 	max-height: 300px; | ||||
| 	overflow: auto; | ||||
| } | ||||
|  | ||||
| @ -674,6 +674,9 @@ def update_grand_total_for_rcm(doc, method): | ||||
| 	if country != 'India': | ||||
| 		return | ||||
| 
 | ||||
| 	if not doc.total_taxes_and_charges: | ||||
| 		return | ||||
| 
 | ||||
| 	if doc.reverse_charge == 'Y': | ||||
| 		gst_accounts = get_gst_accounts(doc.company) | ||||
| 		gst_account_list = gst_accounts.get('cgst_account') + gst_accounts.get('sgst_account') \ | ||||
| @ -721,7 +724,7 @@ def make_regional_gl_entries(gl_entries, doc): | ||||
| 	country = frappe.get_cached_value('Company', doc.company, 'country') | ||||
| 
 | ||||
| 	if country != 'India': | ||||
| 		return | ||||
| 		return gl_entries | ||||
| 
 | ||||
| 	if doc.reverse_charge == 'Y': | ||||
| 		gst_accounts = get_gst_accounts(doc.company) | ||||
| @ -732,6 +735,7 @@ def make_regional_gl_entries(gl_entries, doc): | ||||
| 			if tax.category not in ("Total", "Valuation and Total"): | ||||
| 				continue | ||||
| 
 | ||||
| 			dr_or_cr = "credit" if tax.add_deduct_tax == "Add" else "debit" | ||||
| 			if flt(tax.base_tax_amount_after_discount_amount) and tax.account_head in gst_account_list: | ||||
| 				account_currency = get_account_currency(tax.account_head) | ||||
| 
 | ||||
| @ -741,8 +745,8 @@ def make_regional_gl_entries(gl_entries, doc): | ||||
| 						"cost_center": tax.cost_center, | ||||
| 						"posting_date": doc.posting_date, | ||||
| 						"against": doc.supplier, | ||||
| 						"credit": tax.base_tax_amount_after_discount_amount, | ||||
| 						"credits_in_account_currency": tax.base_tax_amount_after_discount_amount \ | ||||
| 						dr_or_cr: tax.base_tax_amount_after_discount_amount, | ||||
| 						dr_or_cr + "_in_account_currency": tax.base_tax_amount_after_discount_amount \ | ||||
| 							if account_currency==doc.company_currency \ | ||||
| 							else tax.tax_amount_after_discount_amount | ||||
| 					}, account_currency, item=tax) | ||||
|  | ||||
| @ -3,7 +3,7 @@ | ||||
|   { | ||||
|    "hidden": 0, | ||||
|    "label": "Retail Operations", | ||||
|    "links": "[\n    {\n        \"description\": \"Setup default values for POS Invoices\",\n        \"label\": \"Point-of-Sale Profile\",\n        \"name\": \"POS Profile\",\n        \"onboard\": 1,\n        \"type\": \"doctype\"\n    },\n    {\n        \"dependencies\": [\n            \"POS Profile\"\n        ],\n        \"description\": \"Point of Sale\",\n        \"label\": \"POS\",\n        \"name\": \"pos\",\n        \"onboard\": 1,\n        \"type\": \"page\"\n    },\n    {\n        \"description\": \"Cashier Closing\",\n        \"label\": \"Cashier Closing\",\n        \"name\": \"Cashier Closing\",\n        \"type\": \"doctype\"\n    },\n    {\n        \"description\": \"Setup mode of POS (Online / Offline)\",\n        \"label\": \"POS Settings\",\n        \"name\": \"POS Settings\",\n        \"type\": \"doctype\"\n    },\n    {\n        \"description\": \"To make Customer based incentive schemes.\",\n        \"label\": \"Loyalty Program\",\n        \"name\": \"Loyalty Program\",\n        \"type\": \"doctype\"\n    },\n    {\n        \"description\": \"To view logs of Loyalty Points assigned to a Customer.\",\n        \"label\": \"Loyalty Point Entry\",\n        \"name\": \"Loyalty Point Entry\",\n        \"type\": \"doctype\"\n    }\n]" | ||||
|    "links": "[\n    {\n        \"description\": \"Setup default values for POS Invoices\",\n        \"label\": \"Point of Sale Profile\",\n        \"name\": \"POS Profile\",\n        \"onboard\": 1,\n        \"type\": \"doctype\"\n    },\n    {\n        \"dependencies\": [\n            \"POS Profile\"\n        ],\n        \"description\": \"Point of Sale\",\n        \"label\": \"Point of Sale\",\n        \"name\": \"point-of-sale\",\n        \"onboard\": 1,\n        \"type\": \"page\"\n    },\n    {\n        \"description\": \"Setup mode of POS (Online / Offline)\",\n        \"label\": \"POS Settings\",\n        \"name\": \"POS Settings\",\n        \"type\": \"doctype\"\n    },\n    {\n        \"description\": \"Cashier Closing\",\n        \"label\": \"Cashier Closing\",\n        \"name\": \"Cashier Closing\",\n        \"type\": \"doctype\"\n    },\n    {\n        \"description\": \"To make Customer based incentive schemes.\",\n        \"label\": \"Loyalty Program\",\n        \"name\": \"Loyalty Program\",\n        \"type\": \"doctype\"\n    },\n    {\n        \"description\": \"To view logs of Loyalty Points assigned to a Customer.\",\n        \"label\": \"Loyalty Point Entry\",\n        \"name\": \"Loyalty Point Entry\",\n        \"type\": \"doctype\"\n    }\n]" | ||||
|   } | ||||
|  ], | ||||
|  "category": "Domains", | ||||
| @ -14,10 +14,11 @@ | ||||
|  "docstatus": 0, | ||||
|  "doctype": "Desk Page", | ||||
|  "extends_another_page": 0, | ||||
|  "hide_custom": 0, | ||||
|  "idx": 0, | ||||
|  "is_standard": 1, | ||||
|  "label": "Retail", | ||||
|  "modified": "2020-04-26 22:42:39.346750", | ||||
|  "modified": "2020-08-20 18:00:07.515691", | ||||
|  "modified_by": "Administrator", | ||||
|  "module": "Selling", | ||||
|  "name": "Retail", | ||||
| @ -25,5 +26,27 @@ | ||||
|  "pin_to_bottom": 0, | ||||
|  "pin_to_top": 0, | ||||
|  "restrict_to_domain": "Retail", | ||||
|  "shortcuts": [] | ||||
|  "shortcuts": [ | ||||
|   { | ||||
|    "color": "#9deca2", | ||||
|    "doc_view": "", | ||||
|    "format": "{} Active", | ||||
|    "label": "Point of Sale Profile", | ||||
|    "link_to": "POS Profile", | ||||
|    "stats_filter": "{\n    \"disabled\": 0\n}", | ||||
|    "type": "DocType" | ||||
|   }, | ||||
|   { | ||||
|    "doc_view": "", | ||||
|    "label": "Point of Sale", | ||||
|    "link_to": "point-of-sale", | ||||
|    "type": "Page" | ||||
|   }, | ||||
|   { | ||||
|    "doc_view": "", | ||||
|    "label": "POS Settings", | ||||
|    "link_to": "POS Settings", | ||||
|    "type": "DocType" | ||||
|   } | ||||
|  ] | ||||
| } | ||||
| @ -33,7 +33,7 @@ def get_data(): | ||||
| 			}, | ||||
| 			{ | ||||
| 				'label': _('Support'), | ||||
| 				'items': ['Issue'] | ||||
| 				'items': ['Issue', 'Maintenance Visit'] | ||||
| 			}, | ||||
| 			{ | ||||
| 				'label': _('Projects'), | ||||
|  | ||||
| @ -10,6 +10,7 @@ def get_data(): | ||||
| 			'Payment Entry': 'reference_name', | ||||
| 			'Payment Request': 'reference_name', | ||||
| 			'Auto Repeat': 'reference_document', | ||||
| 			'Maintenance Visit': 'prevdoc_docname' | ||||
| 		}, | ||||
| 		'internal_links': { | ||||
| 			'Quotation': ['items', 'prevdoc_docname'] | ||||
| @ -17,7 +18,7 @@ def get_data(): | ||||
| 		'transactions': [ | ||||
| 			{ | ||||
| 				'label': _('Fulfillment'), | ||||
| 				'items': ['Sales Invoice', 'Pick List', 'Delivery Note'] | ||||
| 				'items': ['Sales Invoice', 'Pick List', 'Delivery Note', 'Maintenance Visit'] | ||||
| 			}, | ||||
| 			{ | ||||
| 				'label': _('Purchasing'), | ||||
|  | ||||
| @ -86,7 +86,7 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
|         this.$summary_container.append( | ||||
|             `<div class="summary-btns flex summary-btns justify-between w-full f-shrink-0"></div>` | ||||
|         ) | ||||
|          | ||||
| 
 | ||||
|         this.$summary_btns = this.$summary_container.find('.summary-btns'); | ||||
|     } | ||||
| 
 | ||||
| @ -110,7 +110,10 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
|                 {fieldname:'print', fieldtype:'Data', label:'Print Preview'} | ||||
|             ], | ||||
|             primary_action: () => { | ||||
|                 this.events.get_frm().print_preview.printit(true); | ||||
|                 const frm = this.events.get_frm(); | ||||
|                 frm.doc = this.doc; | ||||
|                 frm.print_preview.lang_code = frm.doc.language; | ||||
|                 frm.print_preview.printit(true); | ||||
|             }, | ||||
|             primary_action_label: __('Print'), | ||||
|         }); | ||||
| @ -174,7 +177,7 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
|                     <div class="flex"> | ||||
|                         <div class="text-md-0 text-dark-grey text-bold w-fit">Tax Charges</div> | ||||
|                         <div class="flex ml-6 text-dark-grey"> | ||||
|                         ${	 | ||||
|                         ${ | ||||
|                             doc.taxes.map((t, i) => { | ||||
|                                 let margin_left = ''; | ||||
|                                 if (i !== 0) margin_left = 'ml-2'; | ||||
| @ -271,6 +274,7 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
|             // this.print_dialog.show();
 | ||||
|             const frm = this.events.get_frm(); | ||||
|             frm.doc = this.doc; | ||||
|             frm.print_preview.lang_code = frm.doc.language; | ||||
|             frm.print_preview.printit(true); | ||||
|         }); | ||||
|     } | ||||
| @ -284,9 +288,9 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
|             this.$summary_container.find('.print-btn').click(); | ||||
|         }); | ||||
|     } | ||||
|      | ||||
| 
 | ||||
|     toggle_component(show) { | ||||
|         show ?  | ||||
|         show ? | ||||
|         this.$component.removeClass('d-none') : | ||||
|         this.$component.addClass('d-none'); | ||||
|     } | ||||
| @ -372,9 +376,9 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
|     } | ||||
| 
 | ||||
|     get_condition_btn_map(after_submission) { | ||||
|         if (after_submission)  | ||||
|         if (after_submission) | ||||
|             return [{ condition: true, visible_btns: ['Print Receipt', 'Email Receipt', 'New Order'] }]; | ||||
|          | ||||
| 
 | ||||
|         return [ | ||||
|             { condition: this.doc.docstatus === 0, visible_btns: ['Edit Order'] }, | ||||
|             { condition: !this.doc.is_return && this.doc.docstatus === 1, visible_btns: ['Print Receipt', 'Email Receipt', 'Return']}, | ||||
| @ -384,7 +388,7 @@ erpnext.PointOfSale.PastOrderSummary = class { | ||||
| 
 | ||||
|     load_summary_of(doc, after_submission=false) { | ||||
|         this.$summary_wrapper.removeClass("d-none"); | ||||
|          | ||||
| 
 | ||||
|         after_submission ? | ||||
|             this.switch_to_post_submit_summary() : this.switch_to_recent_invoice_summary(); | ||||
| 
 | ||||
|  | ||||
| @ -7,6 +7,7 @@ import frappe | ||||
| from erpnext.accounts.doctype.cash_flow_mapper.default_cash_flow_mapper import DEFAULT_MAPPERS | ||||
| from .default_success_action import get_default_success_action | ||||
| from frappe import _ | ||||
| from frappe.utils import cint | ||||
| from frappe.desk.page.setup_wizard.setup_wizard import add_all_roles_to | ||||
| from frappe.custom.doctype.custom_field.custom_field import create_custom_field | ||||
| from erpnext.setup.default_energy_point_rules import get_default_energy_point_rules | ||||
| @ -29,8 +30,8 @@ def after_install(): | ||||
| 
 | ||||
| 
 | ||||
| def check_setup_wizard_not_completed(): | ||||
| 	if frappe.db.get_default('desktop:home_page') != 'setup-wizard': | ||||
| 		message = """ERPNext can only be installed on a fresh site where the setup wizard is not completed.  | ||||
| 	if cint(frappe.db.get_single_value('System Settings', 'setup_complete') or 0): | ||||
| 		message = """ERPNext can only be installed on a fresh site where the setup wizard is not completed. | ||||
| You can reinstall this site (after saving your data) using: bench --site [sitename] reinstall""" | ||||
| 		frappe.throw(message) | ||||
| 
 | ||||
|  | ||||
| @ -111,6 +111,7 @@ class Item(WebsiteGenerator): | ||||
| 		self.synced_with_hub = 0 | ||||
| 
 | ||||
| 		self.validate_has_variants() | ||||
| 		self.validate_attributes_in_variants() | ||||
| 		self.validate_stock_exists_for_template_item() | ||||
| 		self.validate_attributes() | ||||
| 		self.validate_variant_attributes() | ||||
| @ -806,6 +807,77 @@ class Item(WebsiteGenerator): | ||||
| 			if frappe.db.exists("Item", {"variant_of": self.name}): | ||||
| 				frappe.throw(_("Item has variants.")) | ||||
| 
 | ||||
| 	def validate_attributes_in_variants(self): | ||||
| 		if not self.has_variants or self.get("__islocal"): | ||||
| 			return | ||||
| 
 | ||||
| 		old_doc = self.get_doc_before_save() | ||||
| 		old_doc_attributes = set([attr.attribute for attr in old_doc.attributes]) | ||||
| 		own_attributes = [attr.attribute for attr in self.attributes] | ||||
| 
 | ||||
| 		# Check if old attributes were removed from the list | ||||
| 		# Is old_attrs is a subset of new ones | ||||
| 		# that means we need not check any changes | ||||
| 		if old_doc_attributes.issubset(set(own_attributes)): | ||||
| 			return | ||||
| 
 | ||||
| 		from collections import defaultdict | ||||
| 
 | ||||
| 		# get all item variants | ||||
| 		items = [item["name"] for item in frappe.get_all("Item", {"variant_of": self.name})] | ||||
| 
 | ||||
| 		# get all deleted attributes | ||||
| 		deleted_attribute = list(old_doc_attributes.difference(set(own_attributes))) | ||||
| 
 | ||||
| 		# fetch all attributes of these items | ||||
| 		item_attributes = frappe.get_all( | ||||
| 			"Item Variant Attribute", | ||||
| 			filters={ | ||||
| 				"parent": ["in", items], | ||||
| 				"attribute": ["in", deleted_attribute] | ||||
| 			}, | ||||
| 			fields=["attribute", "parent"] | ||||
| 		) | ||||
| 		not_included = defaultdict(list) | ||||
| 
 | ||||
| 		for attr in item_attributes: | ||||
| 			if attr["attribute"] not in own_attributes: | ||||
| 				not_included[attr["parent"]].append(attr["attribute"]) | ||||
| 
 | ||||
| 		if not len(not_included): | ||||
| 			return | ||||
| 
 | ||||
| 		def body(docnames): | ||||
| 			docnames.sort() | ||||
| 			return "<br>".join(docnames) | ||||
| 
 | ||||
| 		def table_row(title, body): | ||||
| 			return """<tr> | ||||
| 				<td>{0}</td> | ||||
| 				<td>{1}</td> | ||||
| 			</tr>""".format(title, body) | ||||
| 
 | ||||
| 		rows = '' | ||||
| 		for docname, attr_list in not_included.items(): | ||||
| 			link = "<a href='#Form/Item/{0}'>{0}</a>".format(frappe.bold(_(docname))) | ||||
| 			rows += table_row(link, body(attr_list)) | ||||
| 
 | ||||
| 		error_description = _('The following deleted attributes exist in Variants but not in the Template. You can either delete the Variants or keep the attribute(s) in template.') | ||||
| 
 | ||||
| 		message = """ | ||||
| 			<div>{0}</div><br> | ||||
| 			<table class="table"> | ||||
| 				<thead> | ||||
| 					<td>{1}</td> | ||||
| 					<td>{2}</td> | ||||
| 				</thead> | ||||
| 				{3} | ||||
| 			</table> | ||||
| 		""".format(error_description, _('Variant Items'), _('Attributes'), rows) | ||||
| 
 | ||||
| 		frappe.throw(message, title=_("Variant Attribute Error"), is_minimizable=True, wide=True) | ||||
| 
 | ||||
| 
 | ||||
| 	def validate_stock_exists_for_template_item(self): | ||||
| 		if self.stock_ledger_created() and self._doc_before_save: | ||||
| 			if (cint(self._doc_before_save.has_variants) != cint(self.has_variants) | ||||
|  | ||||
| @ -227,6 +227,14 @@ class PurchaseReceipt(BuyingController): | ||||
| 					if not stock_value_diff: | ||||
| 						continue | ||||
| 
 | ||||
| 					# If PR is sub-contracted and fg item rate is zero | ||||
| 					# in that case if account for shource and target warehouse are same, | ||||
| 					# then GL entries should not be posted | ||||
| 					if flt(stock_value_diff) == flt(d.rm_supp_cost) \ | ||||
| 						and warehouse_account.get(self.supplier_warehouse) \ | ||||
| 						and warehouse_account[d.warehouse]["account"] == warehouse_account[self.supplier_warehouse]["account"]: | ||||
| 							continue | ||||
| 
 | ||||
| 					gl_entries.append(self.get_gl_dict({ | ||||
| 						"account": warehouse_account[d.warehouse]["account"], | ||||
| 						"against": stock_rbnb, | ||||
| @ -242,16 +250,16 @@ class PurchaseReceipt(BuyingController): | ||||
| 
 | ||||
| 					credit_amount = flt(d.base_net_amount, d.precision("base_net_amount")) \ | ||||
| 						if credit_currency == self.company_currency else flt(d.net_amount, d.precision("net_amount")) | ||||
| 
 | ||||
| 					gl_entries.append(self.get_gl_dict({ | ||||
| 						"account":  warehouse_account[d.from_warehouse]['account'] \ | ||||
| 							if d.from_warehouse else stock_rbnb, | ||||
| 						"against": warehouse_account[d.warehouse]["account"], | ||||
| 						"cost_center": d.cost_center, | ||||
| 						"remarks": self.get("remarks") or _("Accounting Entry for Stock"), | ||||
| 						"debit": -1 * flt(d.base_net_amount, d.precision("base_net_amount")), | ||||
| 						"debit_in_account_currency": -1 * credit_amount | ||||
| 					}, credit_currency, item=d)) | ||||
| 					if credit_amount: | ||||
| 						gl_entries.append(self.get_gl_dict({ | ||||
| 							"account":  warehouse_account[d.from_warehouse]['account'] \ | ||||
| 								if d.from_warehouse else stock_rbnb, | ||||
| 							"against": warehouse_account[d.warehouse]["account"], | ||||
| 							"cost_center": d.cost_center, | ||||
| 							"remarks": self.get("remarks") or _("Accounting Entry for Stock"), | ||||
| 							"debit": -1 * flt(d.base_net_amount, d.precision("base_net_amount")), | ||||
| 							"debit_in_account_currency": -1 * credit_amount | ||||
| 						}, credit_currency, item=d)) | ||||
| 
 | ||||
| 					negative_expense_to_be_booked += flt(d.item_tax_amount) | ||||
| 
 | ||||
|  | ||||
| @ -18,6 +18,28 @@ class TestPurchaseReceipt(unittest.TestCase): | ||||
| 		set_perpetual_inventory(0) | ||||
| 		frappe.db.set_value("Buying Settings", None, "allow_multiple_items", 1) | ||||
| 
 | ||||
| 	def test_reverse_purchase_receipt_sle(self): | ||||
| 
 | ||||
| 		frappe.db.set_value('UOM', '_Test UOM', 'must_be_whole_number', 0) | ||||
| 
 | ||||
| 		pr = make_purchase_receipt(qty=0.5) | ||||
| 
 | ||||
| 		sl_entry = frappe.db.get_all("Stock Ledger Entry", {"voucher_type": "Purchase Receipt", | ||||
| 			"voucher_no": pr.name}, ['actual_qty']) | ||||
| 
 | ||||
| 		self.assertEqual(len(sl_entry), 1) | ||||
| 		self.assertEqual(sl_entry[0].actual_qty, 0.5) | ||||
| 
 | ||||
| 		pr.cancel() | ||||
| 
 | ||||
| 		sl_entry_cancelled = frappe.db.get_all("Stock Ledger Entry", {"voucher_type": "Purchase Receipt", | ||||
| 			"voucher_no": pr.name}, ['actual_qty'], order_by='creation') | ||||
| 
 | ||||
| 		self.assertEqual(len(sl_entry_cancelled), 2) | ||||
| 		self.assertEqual(sl_entry_cancelled[1].actual_qty, -0.5) | ||||
| 
 | ||||
| 		frappe.db.set_value('UOM', '_Test UOM', 'must_be_whole_number', 1) | ||||
| 
 | ||||
| 	def test_make_purchase_invoice(self): | ||||
| 		pr = make_purchase_receipt(do_not_save=True) | ||||
| 		self.assertRaises(frappe.ValidationError, make_purchase_invoice, pr.name) | ||||
| @ -121,6 +143,22 @@ class TestPurchaseReceipt(unittest.TestCase): | ||||
| 		rm_supp_cost = sum([d.amount for d in pr.get("supplied_items")]) | ||||
| 		self.assertEqual(pr.get("items")[0].rm_supp_cost, flt(rm_supp_cost, 2)) | ||||
| 
 | ||||
| 	def test_subcontracting_gle_fg_item_rate_zero(self): | ||||
| 		from erpnext.stock.doctype.stock_entry.test_stock_entry import make_stock_entry | ||||
| 		set_perpetual_inventory() | ||||
| 		frappe.db.set_value("Buying Settings", None, "backflush_raw_materials_of_subcontract_based_on", "BOM") | ||||
| 		make_stock_entry(item_code="_Test Item", target="Work In Progress - TCP1", qty=100, basic_rate=100, company="_Test Company with perpetual inventory") | ||||
| 		make_stock_entry(item_code="_Test Item Home Desktop 100", target="Work In Progress - TCP1", | ||||
| 			qty=100, basic_rate=100, company="_Test Company with perpetual inventory") | ||||
| 		pr = make_purchase_receipt(item_code="_Test FG Item", qty=10, rate=0, is_subcontracted="Yes", | ||||
| 			company="_Test Company with perpetual inventory", warehouse='Stores - TCP1', supplier_warehouse='Work In Progress - TCP1') | ||||
| 		 | ||||
| 		gl_entries = get_gl_entries("Purchase Receipt", pr.name) | ||||
| 
 | ||||
| 		self.assertFalse(gl_entries) | ||||
| 
 | ||||
| 		set_perpetual_inventory(0) | ||||
| 
 | ||||
| 	def test_serial_no_supplier(self): | ||||
| 		pr = make_purchase_receipt(item_code="_Test Serialized Item With Series", qty=1) | ||||
| 		self.assertEqual(frappe.db.get_value("Serial No", pr.get("items")[0].serial_no, "supplier"), | ||||
| @ -688,7 +726,7 @@ def make_purchase_receipt(**args): | ||||
| 		"received_qty": received_qty, | ||||
| 		"rejected_qty": rejected_qty, | ||||
| 		"rejected_warehouse": args.rejected_warehouse or "_Test Rejected Warehouse - _TC" if rejected_qty != 0 else "", | ||||
| 		"rate": args.rate or 50, | ||||
| 		"rate": args.rate if args.rate != None else 50, | ||||
| 		"conversion_factor": args.conversion_factor or 1.0, | ||||
| 		"serial_no": args.serial_no, | ||||
| 		"stock_uom": args.stock_uom or "_Test UOM", | ||||
|  | ||||
| @ -31,7 +31,7 @@ def make_sl_entries(sl_entries, allow_negative_stock=False, via_landed_cost_vouc | ||||
| 				sle['posting_time'] = now_datetime().strftime('%H:%M:%S.%f') | ||||
| 
 | ||||
| 				if cancel: | ||||
| 					sle['actual_qty'] = -flt(sle.get('actual_qty'), 0) | ||||
| 					sle['actual_qty'] = -flt(sle.get('actual_qty')) | ||||
| 
 | ||||
| 					if sle['actual_qty'] < 0 and not sle.get('outgoing_rate'): | ||||
| 						sle['outgoing_rate'] = get_incoming_outgoing_rate_for_cancel(sle.item_code, | ||||
|  | ||||
| @ -209,11 +209,11 @@ function set_time_to_resolve_and_response(frm) { | ||||
| 
 | ||||
| 	frm.dashboard.set_headline_alert( | ||||
| 		'<div class="row">' + | ||||
| 			'<div class="col-xs-6">' + | ||||
| 				'<span class="indicator whitespace-nowrap '+ time_to_respond.indicator +'"><span class="hidden-xs">Time to Respond: '+ time_to_respond.diff_display +'</span></span> ' + | ||||
| 			'<div class="col-xs-12 col-sm-6">' + | ||||
| 				'<span class="indicator whitespace-nowrap '+ time_to_respond.indicator +'"><span>Time to Respond: '+ time_to_respond.diff_display +'</span></span> ' + | ||||
| 			'</div>' + | ||||
| 			'<div class="col-xs-6">' + | ||||
| 				'<span class="indicator whitespace-nowrap '+ time_to_resolve.indicator +'"><span class="hidden-xs">Time to Resolve: '+ time_to_resolve.diff_display +'</span></span> ' + | ||||
| 			'<div class="col-xs-12 col-sm-6">' + | ||||
| 				'<span class="indicator whitespace-nowrap '+ time_to_resolve.indicator +'"><span>Time to Resolve: '+ time_to_resolve.diff_display +'</span></span> ' + | ||||
| 			'</div>' + | ||||
| 		'</div>' | ||||
| 	); | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user