| // Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors |
| // License: GNU General Public License v3. See license.txt |
| |
| frappe.provide("erpnext.buying"); |
| |
| cur_frm.cscript.tax_table = "Purchase Taxes and Charges"; |
| |
| {% include 'erpnext/accounts/doctype/purchase_taxes_and_charges_template/purchase_taxes_and_charges_template.js' %} |
| |
| cur_frm.email_field = "contact_email"; |
| |
| erpnext.buying.BuyingController = erpnext.TransactionController.extend({ |
| setup: function() { |
| this._super(); |
| }, |
| |
| onload: function(doc, cdt, cdn) { |
| this.setup_queries(doc, cdt, cdn); |
| this._super(); |
| |
| this.frm.set_query('shipping_rule', function() { |
| return { |
| filters: { |
| "shipping_rule_type": "Buying" |
| } |
| }; |
| }); |
| |
| if (this.frm.doc.__islocal |
| && frappe.meta.has_field(this.frm.doc.doctype, "disable_rounded_total")) { |
| |
| var df = frappe.meta.get_docfield(this.frm.doc.doctype, "disable_rounded_total"); |
| var disable = cint(df.default) || cint(frappe.sys_defaults.disable_rounded_total); |
| this.frm.set_value("disable_rounded_total", disable); |
| } |
| |
| /* eslint-disable */ |
| // no idea where me is coming from |
| if(this.frm.get_field('shipping_address')) { |
| this.frm.set_query("shipping_address", function() { |
| if(me.frm.doc.customer) { |
| return { |
| query: 'frappe.contacts.doctype.address.address.address_query', |
| filters: { link_doctype: 'Customer', link_name: me.frm.doc.customer } |
| }; |
| } else |
| return erpnext.queries.company_address_query(me.frm.doc) |
| }); |
| } |
| /* eslint-enable */ |
| }, |
| |
| setup_queries: function(doc, cdt, cdn) { |
| var me = this; |
| |
| if(this.frm.fields_dict.buying_price_list) { |
| this.frm.set_query("buying_price_list", function() { |
| return{ |
| filters: { 'buying': 1 } |
| } |
| }); |
| } |
| |
| if(this.frm.fields_dict.tc_name) { |
| this.frm.set_query("tc_name", function() { |
| return{ |
| filters: { 'buying': 1 } |
| } |
| }); |
| } |
| |
| me.frm.set_query('supplier', erpnext.queries.supplier); |
| me.frm.set_query('contact_person', erpnext.queries.contact_query); |
| me.frm.set_query('supplier_address', erpnext.queries.address_query); |
| |
| me.frm.set_query('billing_address', erpnext.queries.company_address_query); |
| |
| if(this.frm.fields_dict.supplier) { |
| this.frm.set_query("supplier", function() { |
| return{ query: "erpnext.controllers.queries.supplier_query" }}); |
| } |
| |
| this.frm.set_query("item_code", "items", function() { |
| if (me.frm.doc.is_subcontracted == "Yes") { |
| return{ |
| query: "erpnext.controllers.queries.item_query", |
| filters:{ 'supplier': me.frm.doc.supplier, 'is_sub_contracted_item': 1 } |
| } |
| } |
| else { |
| return{ |
| query: "erpnext.controllers.queries.item_query", |
| filters: { 'supplier': me.frm.doc.supplier, 'is_purchase_item': 1 } |
| } |
| } |
| }); |
| |
| |
| this.frm.set_query("manufacturer", "items", function(doc, cdt, cdn) { |
| const row = locals[cdt][cdn]; |
| return { |
| query: "erpnext.controllers.queries.item_manufacturer_query", |
| filters:{ 'item_code': row.item_code } |
| } |
| }); |
| |
| if(this.frm.fields_dict["items"].grid.get_field('item_code')) { |
| this.frm.set_query("item_tax_template", "items", function(doc, cdt, cdn) { |
| return me.set_query_for_item_tax_template(doc, cdt, cdn) |
| }); |
| } |
| }, |
| |
| refresh: function(doc) { |
| frappe.dynamic_link = {doc: this.frm.doc, fieldname: 'supplier', doctype: 'Supplier'}; |
| |
| this.frm.toggle_display("supplier_name", |
| (this.frm.doc.supplier_name && this.frm.doc.supplier_name!==this.frm.doc.supplier)); |
| |
| if(this.frm.doc.docstatus==0 && |
| (this.frm.doctype==="Purchase Order" || this.frm.doctype==="Material Request")) { |
| this.set_from_product_bundle(); |
| } |
| |
| this.toggle_subcontracting_fields(); |
| this._super(); |
| }, |
| |
| toggle_subcontracting_fields: function() { |
| if (in_list(['Purchase Receipt', 'Purchase Invoice'], this.frm.doc.doctype)) { |
| this.frm.fields_dict.supplied_items.grid.update_docfield_property('consumed_qty', |
| 'read_only', this.frm.doc.__onload && this.frm.doc.__onload.backflush_based_on === 'BOM'); |
| |
| this.frm.set_df_property('supplied_items', 'cannot_add_rows', 1); |
| this.frm.set_df_property('supplied_items', 'cannot_delete_rows', 1); |
| } |
| }, |
| |
| supplier: function() { |
| var me = this; |
| erpnext.utils.get_party_details(this.frm, null, null, function(){ |
| me.apply_price_list(); |
| }); |
| }, |
| |
| supplier_address: function() { |
| erpnext.utils.get_address_display(this.frm); |
| erpnext.utils.set_taxes_from_address(this.frm, "supplier_address", "supplier_address", "supplier_address"); |
| }, |
| |
| buying_price_list: function() { |
| this.apply_price_list(); |
| }, |
| |
| discount_percentage: function(doc, cdt, cdn) { |
| var item = frappe.get_doc(cdt, cdn); |
| item.discount_amount = 0.0; |
| this.price_list_rate(doc, cdt, cdn); |
| }, |
| |
| discount_amount: function(doc, cdt, cdn) { |
| var item = frappe.get_doc(cdt, cdn); |
| item.discount_percentage = 0.0; |
| this.price_list_rate(doc, cdt, cdn); |
| }, |
| |
| qty: function(doc, cdt, cdn) { |
| var item = frappe.get_doc(cdt, cdn); |
| if ((doc.doctype == "Purchase Receipt") || (doc.doctype == "Purchase Invoice" && (doc.update_stock || doc.is_return))) { |
| frappe.model.round_floats_in(item, ["qty", "received_qty"]); |
| |
| if(!doc.is_return && this.validate_negative_quantity(cdt, cdn, item, ["qty", "received_qty"])){ return } |
| |
| if(!item.rejected_qty && item.qty) { |
| item.received_qty = item.qty; |
| } |
| |
| frappe.model.round_floats_in(item, ["qty", "received_qty"]); |
| item.rejected_qty = flt(item.received_qty - item.qty, precision("rejected_qty", item)); |
| item.received_stock_qty = flt(item.conversion_factor, precision("conversion_factor", item)) * flt(item.received_qty); |
| } |
| this._super(doc, cdt, cdn); |
| }, |
| |
| batch_no: function(doc, cdt, cdn) { |
| this._super(doc, cdt, cdn); |
| }, |
| |
| received_qty: function(doc, cdt, cdn) { |
| this.calculate_accepted_qty(doc, cdt, cdn) |
| }, |
| |
| rejected_qty: function(doc, cdt, cdn) { |
| this.calculate_accepted_qty(doc, cdt, cdn) |
| }, |
| |
| calculate_accepted_qty: function(doc, cdt, cdn){ |
| var item = frappe.get_doc(cdt, cdn); |
| frappe.model.round_floats_in(item, ["received_qty", "rejected_qty"]); |
| |
| if(!doc.is_return && this.validate_negative_quantity(cdt, cdn, item, ["received_qty", "rejected_qty"])){ return } |
| |
| item.qty = flt(item.received_qty - item.rejected_qty, precision("qty", item)); |
| this.qty(doc, cdt, cdn); |
| }, |
| |
| validate_negative_quantity: function(cdt, cdn, item, fieldnames){ |
| if(!item || !fieldnames) { return } |
| |
| var is_negative_qty = false; |
| for(var i = 0; i<fieldnames.length; i++) { |
| if(item[fieldnames[i]] < 0){ |
| frappe.msgprint(__("Row #{0}: {1} can not be negative for item {2}", [item.idx,__(frappe.meta.get_label(cdt, fieldnames[i], cdn)), item.item_code])); |
| is_negative_qty = true; |
| break; |
| } |
| } |
| |
| return is_negative_qty |
| }, |
| |
| warehouse: function(doc, cdt, cdn) { |
| var item = frappe.get_doc(cdt, cdn); |
| if(item.item_code && item.warehouse) { |
| return this.frm.call({ |
| method: "erpnext.stock.get_item_details.get_bin_details", |
| child: item, |
| args: { |
| item_code: item.item_code, |
| warehouse: item.warehouse, |
| company: doc.company |
| } |
| }); |
| } |
| }, |
| |
| project: function(doc, cdt, cdn) { |
| var item = frappe.get_doc(cdt, cdn); |
| if(item.project) { |
| $.each(this.frm.doc["items"] || [], |
| function(i, other_item) { |
| if(!other_item.project) { |
| other_item.project = item.project; |
| refresh_field("project", other_item.name, other_item.parentfield); |
| } |
| }); |
| } |
| }, |
| |
| rejected_warehouse: function(doc, cdt) { |
| // trigger autofill_warehouse only if parent rejected_warehouse field is triggered |
| if (["Purchase Invoice", "Purchase Receipt"].includes(cdt)) { |
| this.autofill_warehouse(doc.items, "rejected_warehouse", doc.rejected_warehouse); |
| } |
| }, |
| |
| category: function(doc, cdt, cdn) { |
| // should be the category field of tax table |
| if(cdt != doc.doctype) { |
| this.calculate_taxes_and_totals(); |
| } |
| }, |
| add_deduct_tax: function(doc, cdt, cdn) { |
| this.calculate_taxes_and_totals(); |
| }, |
| |
| set_from_product_bundle: function() { |
| var me = this; |
| this.frm.add_custom_button(__("Product Bundle"), function() { |
| erpnext.buying.get_items_from_product_bundle(me.frm); |
| }, __("Get Items From")); |
| }, |
| |
| shipping_address: function(){ |
| var me = this; |
| erpnext.utils.get_address_display(this.frm, "shipping_address", |
| "shipping_address_display", true); |
| }, |
| |
| billing_address: function() { |
| erpnext.utils.get_address_display(this.frm, "billing_address", |
| "billing_address_display", true); |
| }, |
| |
| tc_name: function() { |
| this.get_terms(); |
| }, |
| |
| update_auto_repeat_reference: function(doc) { |
| if (doc.auto_repeat) { |
| frappe.call({ |
| method:"frappe.automation.doctype.auto_repeat.auto_repeat.update_reference", |
| args:{ |
| docname: doc.auto_repeat, |
| reference:doc.name |
| }, |
| callback: function(r){ |
| if (r.message=="success") { |
| frappe.show_alert({message:__("Auto repeat document updated"), indicator:'green'}); |
| } else { |
| frappe.show_alert({message:__("An error occurred during the update process"), indicator:'red'}); |
| } |
| } |
| }) |
| } |
| }, |
| |
| manufacturer: function(doc, cdt, cdn) { |
| const row = locals[cdt][cdn]; |
| |
| if(row.manufacturer) { |
| frappe.call({ |
| method: "erpnext.stock.doctype.item_manufacturer.item_manufacturer.get_item_manufacturer_part_no", |
| args: { |
| 'item_code': row.item_code, |
| 'manufacturer': row.manufacturer |
| }, |
| callback: function(r) { |
| if (r.message) { |
| frappe.model.set_value(cdt, cdn, 'manufacturer_part_no', r.message); |
| } |
| } |
| }); |
| } |
| }, |
| |
| manufacturer_part_no: function(doc, cdt, cdn) { |
| const row = locals[cdt][cdn]; |
| |
| if (row.manufacturer_part_no) { |
| frappe.model.get_value('Item Manufacturer', |
| { |
| 'item_code': row.item_code, |
| 'manufacturer': row.manufacturer, |
| 'manufacturer_part_no': row.manufacturer_part_no |
| }, |
| 'name', |
| function(data) { |
| if (!data) { |
| let msg = { |
| message: __("Manufacturer Part Number <b>{0}</b> is invalid", [row.manufacturer_part_no]), |
| title: __("Invalid Part Number") |
| } |
| frappe.throw(msg); |
| } |
| }); |
| |
| } |
| } |
| }); |
| |
| cur_frm.add_fetch('project', 'cost_center', 'cost_center'); |
| |
| erpnext.buying.link_to_mrs = function(frm) { |
| frappe.call({ |
| method: "erpnext.buying.utils.get_linked_material_requests", |
| args:{ |
| items: frm.doc.items.map((item) => item.item_code) |
| }, |
| callback: function(r) { |
| if (!r.message || r.message.length == 0) { |
| frappe.throw({ |
| message: __("No pending Material Requests found to link for the given items."), |
| title: __("Note") |
| }); |
| } |
| |
| var item_length = frm.doc.items.length; |
| for (let item of frm.doc.items) { |
| var qty = item.qty; |
| (r.message[0] || []).forEach(function(d) { |
| if (d.qty > 0 && qty > 0 && item.item_code == d.item_code && !item.material_request_item) |
| { |
| item.material_request = d.mr_name; |
| item.material_request_item = d.mr_item; |
| var my_qty = Math.min(qty, d.qty); |
| qty = qty - my_qty; |
| d.qty = d.qty - my_qty; |
| item.stock_qty = my_qty*item.conversion_factor; |
| item.qty = my_qty; |
| |
| frappe.msgprint("Assigning " + d.mr_name + " to " + d.item_code + " (row " + item.idx + ")"); |
| if (qty > 0) |
| { |
| frappe.msgprint("Splitting " + qty + " units of " + d.item_code); |
| var newrow = frappe.model.add_child(frm.doc, item.doctype, "items"); |
| item_length++; |
| |
| for (var key in item) |
| { |
| newrow[key] = item[key]; |
| } |
| |
| newrow.idx = item_length; |
| newrow["stock_qty"] = newrow.conversion_factor*qty; |
| newrow["qty"] = qty; |
| |
| newrow["material_request"] = ""; |
| newrow["material_request_item"] = ""; |
| |
| } |
| } |
| }); |
| } |
| refresh_field("items"); |
| } |
| }); |
| } |
| |
| erpnext.buying.get_default_bom = function(frm) { |
| $.each(frm.doc["items"] || [], function(i, d) { |
| if (d.item_code && d.bom === "") { |
| return frappe.call({ |
| type: "GET", |
| method: "erpnext.stock.get_item_details.get_default_bom", |
| args: { |
| "item_code": d.item_code, |
| }, |
| callback: function(r) { |
| if(r) { |
| frappe.model.set_value(d.doctype, d.name, "bom", r.message); |
| } |
| } |
| }) |
| } |
| }); |
| } |
| |
| erpnext.buying.get_items_from_product_bundle = function(frm) { |
| var dialog = new frappe.ui.Dialog({ |
| title: __("Get Items from Product Bundle"), |
| fields: [ |
| { |
| "fieldtype": "Link", |
| "label": __("Product Bundle"), |
| "fieldname": "product_bundle", |
| "options":"Product Bundle", |
| "reqd": 1 |
| }, |
| { |
| "fieldtype": "Currency", |
| "label": __("Quantity"), |
| "fieldname": "quantity", |
| "reqd": 1, |
| "default": 1 |
| } |
| ], |
| primary_action_label: 'Get Items', |
| primary_action(args){ |
| if(!args) return; |
| dialog.hide(); |
| return frappe.call({ |
| type: "GET", |
| method: "erpnext.stock.doctype.packed_item.packed_item.get_items_from_product_bundle", |
| args: { |
| args: { |
| item_code: args.product_bundle, |
| quantity: args.quantity, |
| parenttype: frm.doc.doctype, |
| parent: frm.doc.name, |
| supplier: frm.doc.supplier, |
| currency: frm.doc.currency, |
| conversion_rate: frm.doc.conversion_rate, |
| price_list: frm.doc.buying_price_list, |
| price_list_currency: frm.doc.price_list_currency, |
| plc_conversion_rate: frm.doc.plc_conversion_rate, |
| company: frm.doc.company, |
| is_subcontracted: frm.doc.is_subcontracted, |
| transaction_date: frm.doc.transaction_date || frm.doc.posting_date, |
| ignore_pricing_rule: frm.doc.ignore_pricing_rule, |
| doctype: frm.doc.doctype |
| } |
| }, |
| freeze: true, |
| callback: function(r) { |
| const first_row_is_empty = function(child_table){ |
| if($.isArray(child_table) && child_table.length > 0) { |
| return !child_table[0].item_code; |
| } |
| return false; |
| }; |
| |
| const remove_empty_first_row = function(frm){ |
| if (first_row_is_empty(frm.doc.items)){ |
| frm.doc.items = frm.doc.items.splice(1); |
| } |
| }; |
| |
| if(!r.exc && r.message) { |
| remove_empty_first_row(frm); |
| for (var i=0; i< r.message.length; i++) { |
| var d = frm.add_child("items"); |
| var item = r.message[i]; |
| for (var key in item) { |
| if (!is_null(item[key]) && key !== "doctype") { |
| d[key] = item[key]; |
| } |
| } |
| if(frappe.meta.get_docfield(d.doctype, "price_list_rate", d.name)) { |
| frm.script_manager.trigger("price_list_rate", d.doctype, d.name); |
| } |
| } |
| frm.refresh_field("items"); |
| } |
| } |
| }) |
| } |
| }); |
| |
| dialog.show(); |
| } |