blob: 9c56189476443f55c8406d1aaf57f361017980fd [file] [log] [blame]
Rushabh Mehtaf0b45622017-03-31 12:53:05 +05301// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
2// License: GNU General Public License v3. See license.txt
3
4frappe.provide("erpnext.buying");
5
6cur_frm.cscript.tax_table = "Purchase Taxes and Charges";
7
8{% include 'erpnext/accounts/doctype/purchase_taxes_and_charges_template/purchase_taxes_and_charges_template.js' %}
9
10cur_frm.email_field = "contact_email";
11
12erpnext.buying.BuyingController = erpnext.TransactionController.extend({
13 setup: function() {
14 this._super();
15 },
16
Rohit Waghchaure3cf24362019-06-02 16:03:05 +053017 onload: function(doc, cdt, cdn) {
18 this.setup_queries(doc, cdt, cdn);
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053019 this._super();
20
Rushabh Mehta30dc9a12017-11-17 14:31:09 +053021 this.frm.set_query('shipping_rule', function() {
22 return {
23 filters: {
24 "shipping_rule_type": "Buying"
25 }
26 };
27 });
28
rohitwaghchaure8356d4b2018-08-09 15:51:11 +053029 if (this.frm.doc.__islocal
30 && frappe.meta.has_field(this.frm.doc.doctype, "disable_rounded_total")) {
Nabin Haita39f3242019-01-21 18:50:50 +053031
32 var df = frappe.meta.get_docfield(this.frm.doc.doctype, "disable_rounded_total");
marinationca6dbad2019-12-11 12:10:05 +053033 var disable = cint(df.default) || cint(frappe.sys_defaults.disable_rounded_total);
Nabin Haita39f3242019-01-21 18:50:50 +053034 this.frm.set_value("disable_rounded_total", disable);
rohitwaghchaureae4ff5a2018-08-01 18:09:51 +053035 }
36
Faris Ansariab74ca72017-05-30 12:54:42 +053037 /* eslint-disable */
38 // no idea where me is coming from
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053039 if(this.frm.get_field('shipping_address')) {
Faris Ansariab74ca72017-05-30 12:54:42 +053040 this.frm.set_query("shipping_address", function() {
41 if(me.frm.doc.customer) {
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053042 return {
KanchanChauhan1dc26b12017-06-13 15:26:35 +053043 query: 'frappe.contacts.doctype.address.address.address_query',
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053044 filters: { link_doctype: 'Customer', link_name: me.frm.doc.customer }
45 };
46 } else
47 return erpnext.queries.company_address_query(me.frm.doc)
48 });
49 }
Faris Ansariab74ca72017-05-30 12:54:42 +053050 /* eslint-enable */
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053051 },
52
Rohit Waghchaure3cf24362019-06-02 16:03:05 +053053 setup_queries: function(doc, cdt, cdn) {
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053054 var me = this;
55
56 if(this.frm.fields_dict.buying_price_list) {
57 this.frm.set_query("buying_price_list", function() {
58 return{
59 filters: { 'buying': 1 }
60 }
61 });
62 }
63
karthikeyan57fc60212019-07-04 22:46:16 +053064 if(this.frm.fields_dict.tc_name) {
65 this.frm.set_query("tc_name", function() {
66 return{
67 filters: { 'buying': 1 }
68 }
69 });
70 }
71
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053072 me.frm.set_query('supplier', erpnext.queries.supplier);
73 me.frm.set_query('contact_person', erpnext.queries.contact_query);
74 me.frm.set_query('supplier_address', erpnext.queries.address_query);
75
76 if(this.frm.fields_dict.supplier) {
77 this.frm.set_query("supplier", function() {
78 return{ query: "erpnext.controllers.queries.supplier_query" }});
79 }
80
81 this.frm.set_query("item_code", "items", function() {
Doridel Cahanap2b14d6a2018-11-13 14:37:16 +080082 if (me.frm.doc.is_subcontracted == "Yes") {
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053083 return{
84 query: "erpnext.controllers.queries.item_query",
85 filters:{ 'is_sub_contracted_item': 1 }
86 }
Doridel Cahanap2b14d6a2018-11-13 14:37:16 +080087 }
Doridel Cahanap2b14d6a2018-11-13 14:37:16 +080088 else {
Rushabh Mehtaf0b45622017-03-31 12:53:05 +053089 return{
90 query: "erpnext.controllers.queries.item_query",
91 filters: {'is_purchase_item': 1}
92 }
93 }
94 });
Rohit Waghchaure3cf24362019-06-02 16:03:05 +053095
96
97 this.frm.set_query("manufacturer", "items", function(doc, cdt, cdn) {
98 const row = locals[cdt][cdn];
99 return {
100 query: "erpnext.controllers.queries.item_manufacturer_query",
101 filters:{ 'item_code': row.item_code }
102 }
103 });
Deepesh Gargef0d26c2020-01-06 15:34:15 +0530104
105 if(this.frm.fields_dict["items"].grid.get_field('item_code')) {
106 this.frm.set_query("item_tax_template", "items", function(doc, cdt, cdn) {
107 return me.set_query_for_item_tax_template(doc, cdt, cdn)
108 });
109 }
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530110 },
111
112 refresh: function(doc) {
113 frappe.dynamic_link = {doc: this.frm.doc, fieldname: 'supplier', doctype: 'Supplier'};
114
115 this.frm.toggle_display("supplier_name",
116 (this.frm.doc.supplier_name && this.frm.doc.supplier_name!==this.frm.doc.supplier));
117
118 if(this.frm.doc.docstatus==0 &&
119 (this.frm.doctype==="Purchase Order" || this.frm.doctype==="Material Request")) {
120 this.set_from_product_bundle();
121 }
122
123 this._super();
124 },
125
126 supplier: function() {
127 var me = this;
Nabin Haite45ec662018-08-01 17:44:34 +0530128 erpnext.utils.get_party_details(this.frm, null, null, function(){
129 me.apply_price_list();
130 });
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530131 },
132
133 supplier_address: function() {
134 erpnext.utils.get_address_display(this.frm);
Saif Ur Rehmanfd531a62018-12-29 01:49:11 +0500135 erpnext.utils.set_taxes_from_address(this.frm, "supplier_address", "supplier_address", "supplier_address");
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530136 },
137
138 buying_price_list: function() {
139 this.apply_price_list();
140 },
141
142 price_list_rate: function(doc, cdt, cdn) {
143 var item = frappe.get_doc(cdt, cdn);
rohitwaghchaure4870e952019-07-11 10:03:35 +0530144
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530145 frappe.model.round_floats_in(item, ["price_list_rate", "discount_percentage"]);
146
Manas Solankie5e87f72018-05-28 20:07:08 +0530147 let item_rate = item.price_list_rate;
148 if (doc.doctype == "Purchase Order" && item.blanket_order_rate) {
149 item_rate = item.blanket_order_rate;
150 }
Rohit Waghchaure8bfe3302019-03-18 14:34:19 +0530151
152 if (item.discount_percentage) {
153 item.discount_amount = flt(item_rate) * flt(item.discount_percentage) / 100;
154 }
155
Aditya Hase48585c92019-06-18 19:11:17 +0530156 if (item.discount_amount) {
157 item.rate = flt((item.price_list_rate) - (item.discount_amount), precision('rate', item));
rohitwaghchaure4870e952019-07-11 10:03:35 +0530158 } else {
159 item.rate = item_rate;
Aditya Hase48585c92019-06-18 19:11:17 +0530160 }
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530161
162 this.calculate_taxes_and_totals();
163 },
164
165 discount_percentage: function(doc, cdt, cdn) {
Rohit Waghchaure8bfe3302019-03-18 14:34:19 +0530166 var item = frappe.get_doc(cdt, cdn);
167 item.discount_amount = 0.0;
168 this.price_list_rate(doc, cdt, cdn);
169 },
170
171 discount_amount: function(doc, cdt, cdn) {
Nabin Hait593242f2019-04-05 19:35:02 +0530172 var item = frappe.get_doc(cdt, cdn);
173 item.discount_percentage = 0.0;
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530174 this.price_list_rate(doc, cdt, cdn);
175 },
176
177 qty: function(doc, cdt, cdn) {
178 var item = frappe.get_doc(cdt, cdn);
179 if ((doc.doctype == "Purchase Receipt") || (doc.doctype == "Purchase Invoice" && (doc.update_stock || doc.is_return))) {
180 frappe.model.round_floats_in(item, ["qty", "received_qty"]);
181
182 if(!doc.is_return && this.validate_negative_quantity(cdt, cdn, item, ["qty", "received_qty"])){ return }
183
184 if(!item.rejected_qty && item.qty) {
185 item.received_qty = item.qty;
186 }
187
188 frappe.model.round_floats_in(item, ["qty", "received_qty"]);
189 item.rejected_qty = flt(item.received_qty - item.qty, precision("rejected_qty", item));
190 }
191
192 this._super(doc, cdt, cdn);
193 },
194
195 received_qty: function(doc, cdt, cdn) {
196 this.calculate_accepted_qty(doc, cdt, cdn)
197 },
198
199 rejected_qty: function(doc, cdt, cdn) {
200 this.calculate_accepted_qty(doc, cdt, cdn)
201 },
202
203 calculate_accepted_qty: function(doc, cdt, cdn){
204 var item = frappe.get_doc(cdt, cdn);
205 frappe.model.round_floats_in(item, ["received_qty", "rejected_qty"]);
206
207 if(!doc.is_return && this.validate_negative_quantity(cdt, cdn, item, ["received_qty", "rejected_qty"])){ return }
208
209 item.qty = flt(item.received_qty - item.rejected_qty, precision("qty", item));
210 this.qty(doc, cdt, cdn);
211 },
212
213 validate_negative_quantity: function(cdt, cdn, item, fieldnames){
214 if(!item || !fieldnames) { return }
215
216 var is_negative_qty = false;
217 for(var i = 0; i<fieldnames.length; i++) {
218 if(item[fieldnames[i]] < 0){
219 frappe.msgprint(__("Row #{0}: {1} can not be negative for item {2}",
220 [item.idx,__(frappe.meta.get_label(cdt, fieldnames[i], cdn)), item.item_code]));
221 is_negative_qty = true;
222 break;
223 }
224 }
225
226 return is_negative_qty
227 },
228
229 warehouse: function(doc, cdt, cdn) {
230 var item = frappe.get_doc(cdt, cdn);
231 if(item.item_code && item.warehouse) {
232 return this.frm.call({
233 method: "erpnext.stock.get_item_details.get_bin_details",
234 child: item,
235 args: {
236 item_code: item.item_code,
237 warehouse: item.warehouse
238 }
239 });
240 }
241 },
242
243 project: function(doc, cdt, cdn) {
244 var item = frappe.get_doc(cdt, cdn);
245 if(item.project) {
246 $.each(this.frm.doc["items"] || [],
247 function(i, other_item) {
248 if(!other_item.project) {
249 other_item.project = item.project;
250 refresh_field("project", other_item.name, other_item.parentfield);
251 }
252 });
253 }
254 },
255
marination8a30cb22020-04-23 00:18:21 +0530256 rejected_warehouse: function(doc, cdt) {
257 // trigger autofill_warehouse only if parent rejected_warehouse field is triggered
258 if (["Purchase Invoice", "Purchase Receipt"].includes(cdt)) {
259 this.autofill_warehouse(doc.items, "rejected_warehouse", doc.rejected_warehouse);
260 }
261 },
262
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530263 category: function(doc, cdt, cdn) {
264 // should be the category field of tax table
265 if(cdt != doc.doctype) {
266 this.calculate_taxes_and_totals();
267 }
268 },
269 add_deduct_tax: function(doc, cdt, cdn) {
270 this.calculate_taxes_and_totals();
271 },
272
273 set_from_product_bundle: function() {
274 var me = this;
275 this.frm.add_custom_button(__("Product Bundle"), function() {
276 erpnext.buying.get_items_from_product_bundle(me.frm);
277 }, __("Get items from"));
278 },
279
280 shipping_address: function(){
281 var me = this;
282 erpnext.utils.get_address_display(this.frm, "shipping_address",
Faris Ansariab74ca72017-05-30 12:54:42 +0530283 "shipping_address_display", true);
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530284 },
285
286 tc_name: function() {
287 this.get_terms();
288 },
Charles-Henri Decultot8b223102018-06-28 10:57:58 +0200289
Shreya Shah95d93912018-10-08 14:17:37 +0530290 link_to_mrs: function() {
291 var my_items = [];
292 for (var i in cur_frm.doc.items) {
293 if(!cur_frm.doc.items[i].material_request){
294 my_items.push(cur_frm.doc.items[i].item_code);
295 }
296 }
297 frappe.call({
298 method: "erpnext.buying.utils.get_linked_material_requests",
299 args:{
300 items: my_items
301 },
302 callback: function(r) {
Rucha Mahabal2818b5b2019-09-16 15:16:38 +0530303 if(!r.message || r.message.length == 0) {
Shreya Shah95d93912018-10-08 14:17:37 +0530304 frappe.throw(__("No pending Material Requests found to link for the given items."))
305 }
306 else {
307 var i = 0;
308 var item_length = cur_frm.doc.items.length;
309 while (i < item_length) {
310 var qty = cur_frm.doc.items[i].qty;
311 (r.message[0] || []).forEach(function(d) {
312 if (d.qty > 0 && qty > 0 && cur_frm.doc.items[i].item_code == d.item_code && !cur_frm.doc.items[i].material_request_item)
313 {
314 cur_frm.doc.items[i].material_request = d.mr_name;
315 cur_frm.doc.items[i].material_request_item = d.mr_item;
316 var my_qty = Math.min(qty, d.qty);
317 qty = qty - my_qty;
318 d.qty = d.qty - my_qty;
319 cur_frm.doc.items[i].stock_qty = my_qty*cur_frm.doc.items[i].conversion_factor;
320 cur_frm.doc.items[i].qty = my_qty;
Nabin Haita39f3242019-01-21 18:50:50 +0530321
Shreya Shah95d93912018-10-08 14:17:37 +0530322 frappe.msgprint("Assigning " + d.mr_name + " to " + d.item_code + " (row " + cur_frm.doc.items[i].idx + ")");
323 if (qty > 0)
324 {
325 frappe.msgprint("Splitting " + qty + " units of " + d.item_code);
326 var newrow = frappe.model.add_child(cur_frm.doc, cur_frm.doc.items[i].doctype, "items");
327 item_length++;
Nabin Haita39f3242019-01-21 18:50:50 +0530328
Shreya Shah95d93912018-10-08 14:17:37 +0530329 for (var key in cur_frm.doc.items[i])
330 {
331 newrow[key] = cur_frm.doc.items[i][key];
332 }
Nabin Haita39f3242019-01-21 18:50:50 +0530333
Shreya Shah95d93912018-10-08 14:17:37 +0530334 newrow.idx = item_length;
335 newrow["stock_qty"] = newrow.conversion_factor*qty;
336 newrow["qty"] = qty;
Nabin Haita39f3242019-01-21 18:50:50 +0530337
Shreya Shah95d93912018-10-08 14:17:37 +0530338 newrow["material_request"] = "";
339 newrow["material_request_item"] = "";
Nabin Haita39f3242019-01-21 18:50:50 +0530340
Shreya Shah95d93912018-10-08 14:17:37 +0530341 }
342 }
343 });
344 i++;
345 }
346 refresh_field("items");
347 //cur_frm.save();
348 }
349 }
350 });
351 },
352
Charles-Henri Decultot8b223102018-06-28 10:57:58 +0200353 update_auto_repeat_reference: function(doc) {
354 if (doc.auto_repeat) {
355 frappe.call({
Rucha Mahabal65a627c2019-07-17 13:50:32 +0530356 method:"frappe.automation.doctype.auto_repeat.auto_repeat.update_reference",
Nabin Haita39f3242019-01-21 18:50:50 +0530357 args:{
Charles-Henri Decultot8b223102018-06-28 10:57:58 +0200358 docname: doc.auto_repeat,
359 reference:doc.name
360 },
361 callback: function(r){
362 if (r.message=="success") {
363 frappe.show_alert({message:__("Auto repeat document updated"), indicator:'green'});
364 } else {
365 frappe.show_alert({message:__("An error occurred during the update process"), indicator:'red'});
366 }
367 }
368 })
369 }
Rohit Waghchaure3cf24362019-06-02 16:03:05 +0530370 },
371
372 manufacturer: function(doc, cdt, cdn) {
373 const row = locals[cdt][cdn];
374
375 if(row.manufacturer) {
376 frappe.call({
377 method: "erpnext.stock.doctype.item_manufacturer.item_manufacturer.get_item_manufacturer_part_no",
378 args: {
379 'item_code': row.item_code,
380 'manufacturer': row.manufacturer
381 },
382 callback: function(r) {
383 if (r.message) {
384 frappe.model.set_value(cdt, cdn, 'manufacturer_part_no', r.message);
385 }
386 }
387 });
388 }
marination3e7a8ab2020-04-07 21:00:57 +0530389 },
390
391 manufacturer_part_no: function(doc, cdt, cdn) {
392 const row = locals[cdt][cdn];
393
394 if (row.manufacturer_part_no) {
395 frappe.model.get_value('Item Manufacturer',
396 {
397 'item_code': row.item_code,
398 'manufacturer': row.manufacturer,
399 'manufacturer_part_no': row.manufacturer_part_no
400 },
401 'name',
402 function(data) {
403 if (!data) {
404 let msg = {
405 message: __("Manufacturer Part Number <b>{0}</b> is invalid", [row.manufacturer_part_no]),
406 title: __("Invalid Part Number")
407 }
408 frappe.throw(msg);
409 }
410 });
411
412 }
413 }
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530414});
415
416cur_frm.add_fetch('project', 'cost_center', 'cost_center');
417
418erpnext.buying.get_default_bom = function(frm) {
419 $.each(frm.doc["items"] || [], function(i, d) {
420 if (d.item_code && d.bom === "") {
421 return frappe.call({
422 type: "GET",
423 method: "erpnext.stock.get_item_details.get_default_bom",
424 args: {
425 "item_code": d.item_code,
426 },
427 callback: function(r) {
428 if(r) {
429 frappe.model.set_value(d.doctype, d.name, "bom", r.message);
430 }
431 }
432 })
433 }
434 });
435}
436
437erpnext.buying.get_items_from_product_bundle = function(frm) {
438 var dialog = new frappe.ui.Dialog({
439 title: __("Get Items from Product Bundle"),
440 fields: [
441 {
442 "fieldtype": "Link",
443 "label": __("Product Bundle"),
444 "fieldname": "product_bundle",
445 "options":"Product Bundle",
446 "reqd": 1
447 },
448 {
449 "fieldtype": "Currency",
450 "label": __("Quantity"),
451 "fieldname": "quantity",
452 "reqd": 1,
453 "default": 1
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530454 }
Marica9942acd2020-04-21 13:13:39 +0530455 ],
456 primary_action_label: 'Get Items',
457 primary_action(args){
458 if(!args) return;
459 dialog.hide();
460 return frappe.call({
461 type: "GET",
462 method: "erpnext.stock.doctype.packed_item.packed_item.get_items_from_product_bundle",
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530463 args: {
Marica9942acd2020-04-21 13:13:39 +0530464 args: {
465 item_code: args.product_bundle,
466 quantity: args.quantity,
467 parenttype: frm.doc.doctype,
468 parent: frm.doc.name,
469 supplier: frm.doc.supplier,
470 currency: frm.doc.currency,
471 conversion_rate: frm.doc.conversion_rate,
472 price_list: frm.doc.buying_price_list,
473 price_list_currency: frm.doc.price_list_currency,
474 plc_conversion_rate: frm.doc.plc_conversion_rate,
475 company: frm.doc.company,
476 is_subcontracted: frm.doc.is_subcontracted,
477 transaction_date: frm.doc.transaction_date || frm.doc.posting_date,
478 ignore_pricing_rule: frm.doc.ignore_pricing_rule,
479 doctype: frm.doc.doctype
tundebabzyf35710d2017-08-16 08:55:18 +0100480 }
Marica9942acd2020-04-21 13:13:39 +0530481 },
482 freeze: true,
483 callback: function(r) {
484 const first_row_is_empty = function(child_table){
485 if($.isArray(child_table) && child_table.length > 0) {
486 return !child_table[0].item_code;
487 }
488 return false;
489 };
tundebabzyf35710d2017-08-16 08:55:18 +0100490
Marica9942acd2020-04-21 13:13:39 +0530491 const remove_empty_first_row = function(frm){
492 if (first_row_is_empty(frm.doc.items)){
493 frm.doc.items = frm.doc.items.splice(1);
494 }
495 };
tundebabzyf35710d2017-08-16 08:55:18 +0100496
Marica9942acd2020-04-21 13:13:39 +0530497 if(!r.exc && r.message) {
498 remove_empty_first_row(frm);
499 for ( var i=0; i< r.message.length; i++ ) {
500 var d = frm.add_child("items");
501 var item = r.message[i];
502 for ( var key in item) {
503 if ( !is_null(item[key]) ) {
504 d[key] = item[key];
505 }
506 }
507 if(frappe.meta.get_docfield(d.doctype, "price_list_rate", d.name)) {
508 frm.script_manager.trigger("price_list_rate", d.doctype, d.name);
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530509 }
510 }
Marica9942acd2020-04-21 13:13:39 +0530511 frm.refresh_field("items");
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530512 }
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530513 }
Marica9942acd2020-04-21 13:13:39 +0530514 })
515 }
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530516 });
Marica9942acd2020-04-21 13:13:39 +0530517
Rushabh Mehtaf0b45622017-03-31 12:53:05 +0530518 dialog.show();
519}