2017-06-22 10:19:42 +00:00
|
|
|
|
|
|
|
erpnext.SerialNoBatchSelector = class SerialNoBatchSelector {
|
2017-08-28 11:49:28 +00:00
|
|
|
constructor(opts, show_dialog) {
|
2017-06-22 10:19:42 +00:00
|
|
|
$.extend(this, opts);
|
2019-04-16 12:55:53 +00:00
|
|
|
this.show_dialog = show_dialog;
|
2017-06-22 10:19:42 +00:00
|
|
|
// frm, item, warehouse_details, has_batch, oldest
|
2017-06-23 06:09:03 +00:00
|
|
|
let d = this.item;
|
2020-03-26 07:48:57 +00:00
|
|
|
this.has_batch = 0; this.has_serial_no = 0;
|
|
|
|
|
|
|
|
if (d && d.has_batch_no && (!d.batch_no || this.show_dialog)) this.has_batch = 1;
|
2019-04-17 10:16:15 +00:00
|
|
|
// !(this.show_dialog == false) ensures that show_dialog is implictly true, even when undefined
|
2020-03-26 07:48:57 +00:00
|
|
|
if(d && d.has_serial_no && !(this.show_dialog == false)) this.has_serial_no = 1;
|
|
|
|
|
|
|
|
this.setup();
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
setup() {
|
|
|
|
this.item_code = this.item.item_code;
|
|
|
|
this.qty = this.item.qty;
|
|
|
|
this.make_dialog();
|
2017-12-14 10:36:23 +00:00
|
|
|
this.on_close_dialog();
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
make_dialog() {
|
|
|
|
var me = this;
|
|
|
|
|
|
|
|
this.data = this.oldest ? this.oldest : [];
|
|
|
|
let title = "";
|
|
|
|
let fields = [
|
2017-06-26 10:01:46 +00:00
|
|
|
{
|
|
|
|
fieldname: 'item_code',
|
|
|
|
read_only: 1,
|
|
|
|
fieldtype:'Link',
|
|
|
|
options: 'Item',
|
|
|
|
label: __('Item Code'),
|
|
|
|
default: me.item_code
|
|
|
|
},
|
2017-06-22 10:19:42 +00:00
|
|
|
{
|
|
|
|
fieldname: 'warehouse',
|
|
|
|
fieldtype:'Link',
|
|
|
|
options: 'Warehouse',
|
2020-03-26 07:48:57 +00:00
|
|
|
reqd: me.has_batch && !me.has_serial_no ? 0 : 1,
|
2017-06-22 10:19:42 +00:00
|
|
|
label: __(me.warehouse_details.type),
|
2020-03-26 07:48:57 +00:00
|
|
|
default: typeof me.warehouse_details.name == "string" ? me.warehouse_details.name : '',
|
2017-06-22 10:19:42 +00:00
|
|
|
onchange: function(e) {
|
2020-07-23 13:21:26 +00:00
|
|
|
me.warehouse_details.name = this.get_value();
|
2019-06-28 08:38:08 +00:00
|
|
|
|
2020-03-26 07:48:57 +00:00
|
|
|
if(me.has_batch && !me.has_serial_no) {
|
2019-06-28 08:38:08 +00:00
|
|
|
fields = fields.concat(me.get_batch_fields());
|
|
|
|
} else {
|
|
|
|
fields = fields.concat(me.get_serial_no_fields());
|
|
|
|
}
|
|
|
|
|
2017-06-22 10:19:42 +00:00
|
|
|
var batches = this.layout.fields_dict.batches;
|
|
|
|
if(batches) {
|
|
|
|
batches.grid.df.data = [];
|
|
|
|
batches.grid.refresh();
|
|
|
|
batches.grid.add_new_row(null, null, null);
|
|
|
|
}
|
2017-07-12 08:40:43 +00:00
|
|
|
},
|
|
|
|
get_query: function() {
|
|
|
|
return {
|
2018-05-31 03:59:18 +00:00
|
|
|
query: "erpnext.controllers.queries.warehouse_query",
|
|
|
|
filters: [
|
|
|
|
["Bin", "item_code", "=", me.item_code],
|
|
|
|
["Warehouse", "is_group", "=", 0],
|
|
|
|
["Warehouse", "company", "=", me.frm.doc.company]
|
|
|
|
]
|
|
|
|
}
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
{fieldtype:'Column Break'},
|
2017-06-26 10:01:46 +00:00
|
|
|
{
|
|
|
|
fieldname: 'qty',
|
|
|
|
fieldtype:'Float',
|
2020-03-26 07:48:57 +00:00
|
|
|
read_only: me.has_batch && !me.has_serial_no,
|
2021-05-14 07:06:41 +00:00
|
|
|
label: __(me.has_batch && !me.has_serial_no ? 'Selected Qty' : 'Qty'),
|
2022-03-19 12:17:18 +00:00
|
|
|
default: flt(me.item.stock_qty) || flt(me.item.transfer_qty),
|
2017-06-26 10:01:46 +00:00
|
|
|
},
|
2021-05-14 07:06:41 +00:00
|
|
|
...get_pending_qty_fields(me),
|
2021-05-14 06:51:38 +00:00
|
|
|
{
|
|
|
|
fieldname: 'uom',
|
|
|
|
read_only: 1,
|
|
|
|
fieldtype: 'Link',
|
|
|
|
options: 'UOM',
|
|
|
|
label: __('UOM'),
|
|
|
|
default: me.item.uom
|
|
|
|
},
|
2019-04-16 07:17:39 +00:00
|
|
|
{
|
|
|
|
fieldname: 'auto_fetch_button',
|
|
|
|
fieldtype:'Button',
|
2020-03-26 07:48:57 +00:00
|
|
|
hidden: me.has_batch && !me.has_serial_no,
|
|
|
|
label: __('Auto Fetch'),
|
|
|
|
description: __('Fetch Serial Numbers based on FIFO'),
|
2019-04-17 04:56:21 +00:00
|
|
|
click: () => {
|
2019-04-16 12:55:53 +00:00
|
|
|
let qty = this.dialog.fields_dict.qty.get_value();
|
2022-03-19 12:19:19 +00:00
|
|
|
let already_selected_serial_nos = get_selected_serial_nos(me);
|
2019-04-16 07:17:39 +00:00
|
|
|
let numbers = frappe.call({
|
|
|
|
method: "erpnext.stock.doctype.serial_no.serial_no.auto_fetch_serial_number",
|
|
|
|
args: {
|
2019-04-16 11:31:01 +00:00
|
|
|
qty: qty,
|
2019-04-16 07:17:39 +00:00
|
|
|
item_code: me.item_code,
|
2020-03-26 07:48:57 +00:00
|
|
|
warehouse: typeof me.warehouse_details.name == "string" ? me.warehouse_details.name : '',
|
2022-03-19 12:18:43 +00:00
|
|
|
batch_nos: me.item.batch_no || null,
|
2022-03-19 12:19:19 +00:00
|
|
|
posting_date: me.frm.doc.posting_date || me.frm.doc.transaction_date,
|
|
|
|
exclude_sr_nos: already_selected_serial_nos
|
2019-04-16 07:17:39 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
numbers.then((data) => {
|
|
|
|
let auto_fetched_serial_numbers = data.message;
|
2019-04-16 12:55:53 +00:00
|
|
|
let records_length = auto_fetched_serial_numbers.length;
|
2020-07-23 13:21:26 +00:00
|
|
|
if (!records_length) {
|
|
|
|
const warehouse = me.dialog.fields_dict.warehouse.get_value().bold();
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.msgprint(
|
|
|
|
__('Serial numbers unavailable for Item {0} under warehouse {1}. Please try changing warehouse.', [me.item.item_code.bold(), warehouse])
|
|
|
|
);
|
2020-07-23 13:21:26 +00:00
|
|
|
}
|
2019-04-17 10:16:15 +00:00
|
|
|
if (records_length < qty) {
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.msgprint(__('Fetched only {0} available serial numbers.', [records_length]));
|
2019-04-16 11:31:01 +00:00
|
|
|
}
|
2019-04-16 07:17:39 +00:00
|
|
|
let serial_no_list_field = this.dialog.fields_dict.serial_no;
|
|
|
|
numbers = auto_fetched_serial_numbers.join('\n');
|
|
|
|
serial_no_list_field.set_value(numbers);
|
2019-04-16 11:33:25 +00:00
|
|
|
});
|
2019-04-16 07:17:39 +00:00
|
|
|
}
|
|
|
|
}
|
2017-06-22 10:19:42 +00:00
|
|
|
];
|
|
|
|
|
2020-03-26 07:48:57 +00:00
|
|
|
if (this.has_batch && !this.has_serial_no) {
|
2017-06-22 10:19:42 +00:00
|
|
|
title = __("Select Batch Numbers");
|
|
|
|
fields = fields.concat(this.get_batch_fields());
|
|
|
|
} else {
|
2020-03-26 07:48:57 +00:00
|
|
|
// if only serial no OR
|
|
|
|
// if both batch_no & serial_no then only select serial_no and auto set batches nos
|
2017-06-22 10:19:42 +00:00
|
|
|
title = __("Select Serial Numbers");
|
|
|
|
fields = fields.concat(this.get_serial_no_fields());
|
|
|
|
}
|
|
|
|
|
|
|
|
this.dialog = new frappe.ui.Dialog({
|
|
|
|
title: title,
|
|
|
|
fields: fields
|
|
|
|
});
|
|
|
|
|
|
|
|
this.dialog.set_primary_action(__('Insert'), function() {
|
|
|
|
me.values = me.dialog.get_values();
|
2017-06-22 10:24:18 +00:00
|
|
|
if(me.validate()) {
|
2020-03-26 07:48:57 +00:00
|
|
|
frappe.run_serially([
|
|
|
|
() => me.update_batch_items(),
|
|
|
|
() => me.update_serial_no_item(),
|
|
|
|
() => me.update_batch_serial_no_items(),
|
|
|
|
() => {
|
|
|
|
refresh_field("items");
|
2021-02-12 05:59:24 +00:00
|
|
|
refresh_field("packed_items");
|
2020-03-26 07:48:57 +00:00
|
|
|
if (me.callback) {
|
|
|
|
return me.callback(me.item);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
() => me.dialog.hide()
|
|
|
|
])
|
2017-06-22 10:24:18 +00:00
|
|
|
}
|
2017-06-22 10:19:42 +00:00
|
|
|
});
|
|
|
|
|
2017-08-28 11:49:28 +00:00
|
|
|
if(this.show_dialog) {
|
|
|
|
let d = this.item;
|
2020-03-26 07:48:57 +00:00
|
|
|
if (this.item.serial_no) {
|
|
|
|
this.dialog.fields_dict.serial_no.set_value(this.item.serial_no);
|
2017-11-29 05:23:09 +00:00
|
|
|
}
|
2021-02-12 05:59:24 +00:00
|
|
|
|
2020-03-26 07:48:57 +00:00
|
|
|
if (this.has_batch && !this.has_serial_no && d.batch_no) {
|
2018-03-28 07:03:15 +00:00
|
|
|
this.frm.doc.items.forEach(data => {
|
|
|
|
if(data.item_code == d.item_code) {
|
|
|
|
this.dialog.fields_dict.batches.df.data.push({
|
|
|
|
'batch_no': data.batch_no,
|
|
|
|
'actual_qty': data.actual_qty,
|
|
|
|
'selected_qty': data.qty,
|
|
|
|
'available_qty': data.actual_batch_qty
|
|
|
|
});
|
|
|
|
}
|
2017-10-02 06:43:36 +00:00
|
|
|
});
|
|
|
|
this.dialog.fields_dict.batches.grid.refresh();
|
|
|
|
}
|
2017-08-28 11:49:28 +00:00
|
|
|
}
|
|
|
|
|
2020-03-26 07:48:57 +00:00
|
|
|
if (this.has_batch && !this.has_serial_no) {
|
2018-03-28 07:03:15 +00:00
|
|
|
this.update_total_qty();
|
2021-05-14 07:06:41 +00:00
|
|
|
this.update_pending_qtys();
|
2018-03-28 07:03:15 +00:00
|
|
|
}
|
|
|
|
|
2017-06-22 10:19:42 +00:00
|
|
|
this.dialog.show();
|
|
|
|
}
|
|
|
|
|
2017-12-14 10:36:23 +00:00
|
|
|
on_close_dialog() {
|
|
|
|
this.dialog.get_close_btn().on('click', () => {
|
|
|
|
this.on_close && this.on_close(this.item);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-06-22 10:19:42 +00:00
|
|
|
validate() {
|
|
|
|
let values = this.values;
|
|
|
|
if(!values.warehouse) {
|
|
|
|
frappe.throw(__("Please select a warehouse"));
|
|
|
|
return false;
|
|
|
|
}
|
2020-03-26 07:48:57 +00:00
|
|
|
if(this.has_batch && !this.has_serial_no) {
|
2017-06-22 10:19:42 +00:00
|
|
|
if(values.batches.length === 0 || !values.batches) {
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.throw(__("Please select batches for batched item {0}", [values.item_code]));
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
values.batches.map((batch, i) => {
|
|
|
|
if(!batch.selected_qty || batch.selected_qty === 0 ) {
|
2017-10-02 06:43:36 +00:00
|
|
|
if (!this.show_dialog) {
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.throw(__("Please select quantity on row {0}", [i+1]));
|
2017-10-02 06:43:36 +00:00
|
|
|
}
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
let serial_nos = values.serial_no || '';
|
|
|
|
if (!serial_nos || !serial_nos.replace(/\s/g, '').length) {
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.throw(__("Please enter serial numbers for serialized item {0}", [values.item_code]));
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-26 07:48:57 +00:00
|
|
|
update_batch_items() {
|
|
|
|
// clones an items if muliple batches are selected.
|
|
|
|
if(this.has_batch && !this.has_serial_no) {
|
2017-06-22 10:19:42 +00:00
|
|
|
this.values.batches.map((batch, i) => {
|
2018-03-28 07:03:15 +00:00
|
|
|
let batch_no = batch.batch_no;
|
|
|
|
let row = '';
|
|
|
|
|
|
|
|
if (i !== 0 && !this.batch_exists(batch_no)) {
|
2020-03-26 07:48:57 +00:00
|
|
|
row = this.frm.add_child("items", { ...this.item });
|
2018-03-28 07:03:15 +00:00
|
|
|
} else {
|
|
|
|
row = this.frm.doc.items.find(i => i.batch_no === batch_no);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!row) {
|
|
|
|
row = this.item;
|
|
|
|
}
|
2020-03-26 07:48:57 +00:00
|
|
|
// this ensures that qty & batch no is set
|
2017-06-27 08:33:56 +00:00
|
|
|
this.map_row_values(row, batch, 'batch_no',
|
2017-06-23 06:09:03 +00:00
|
|
|
'selected_qty', this.values.warehouse);
|
2017-06-22 10:19:42 +00:00
|
|
|
});
|
2021-02-12 05:59:24 +00:00
|
|
|
}
|
2020-03-26 07:48:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
update_serial_no_item() {
|
|
|
|
// just updates serial no for the item
|
|
|
|
if(this.has_serial_no && !this.has_batch) {
|
2017-06-26 08:08:24 +00:00
|
|
|
this.map_row_values(this.item, this.values, 'serial_no', 'qty');
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
2020-03-26 07:48:57 +00:00
|
|
|
}
|
2018-03-28 07:03:15 +00:00
|
|
|
|
2020-03-26 07:48:57 +00:00
|
|
|
update_batch_serial_no_items() {
|
|
|
|
// if serial no selected is from different batches, adds new rows for each batch.
|
|
|
|
if(this.has_batch && this.has_serial_no) {
|
|
|
|
const selected_serial_nos = this.values.serial_no.split(/\n/g).filter(s => s);
|
|
|
|
|
|
|
|
return frappe.db.get_list("Serial No", {
|
|
|
|
filters: { 'name': ["in", selected_serial_nos]},
|
|
|
|
fields: ["batch_no", "name"]
|
|
|
|
}).then((data) => {
|
2021-02-12 05:59:24 +00:00
|
|
|
// data = [{batch_no: 'batch-1', name: "SR-001"},
|
2020-03-26 07:48:57 +00:00
|
|
|
// {batch_no: 'batch-2', name: "SR-003"}, {batch_no: 'batch-2', name: "SR-004"}]
|
|
|
|
const batch_serial_map = data.reduce((acc, d) => {
|
|
|
|
if (!acc[d['batch_no']]) acc[d['batch_no']] = [];
|
|
|
|
acc[d['batch_no']].push(d['name'])
|
|
|
|
return acc
|
|
|
|
}, {})
|
|
|
|
// batch_serial_map = { "batch-1": ['SR-001'], "batch-2": ["SR-003", "SR-004"]}
|
|
|
|
Object.keys(batch_serial_map).map((batch_no, i) => {
|
|
|
|
let row = '';
|
|
|
|
const serial_no = batch_serial_map[batch_no];
|
|
|
|
if (i == 0) {
|
|
|
|
row = this.item;
|
|
|
|
this.map_row_values(row, {qty: serial_no.length, batch_no: batch_no}, 'batch_no',
|
|
|
|
'qty', this.values.warehouse);
|
|
|
|
} else if (!this.batch_exists(batch_no)) {
|
|
|
|
row = this.frm.add_child("items", { ...this.item });
|
|
|
|
row.batch_no = batch_no;
|
|
|
|
} else {
|
|
|
|
row = this.frm.doc.items.find(i => i.batch_no === batch_no);
|
|
|
|
}
|
|
|
|
const values = {
|
|
|
|
'qty': serial_no.length,
|
|
|
|
'serial_no': serial_no.join('\n')
|
|
|
|
}
|
|
|
|
this.map_row_values(row, values, 'serial_no',
|
|
|
|
'qty', this.values.warehouse);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
|
2018-03-28 07:03:15 +00:00
|
|
|
batch_exists(batch) {
|
|
|
|
const batches = this.frm.doc.items.map(data => data.batch_no);
|
|
|
|
return (batches && in_list(batches, batch)) ? true : false;
|
|
|
|
}
|
|
|
|
|
2017-06-27 08:33:56 +00:00
|
|
|
map_row_values(row, values, number, qty_field, warehouse) {
|
2017-06-23 06:09:03 +00:00
|
|
|
row.qty = values[qty_field];
|
2019-01-24 08:00:07 +00:00
|
|
|
row.transfer_qty = flt(values[qty_field]) * flt(row.conversion_factor);
|
2017-06-26 08:08:24 +00:00
|
|
|
row[number] = values[number];
|
2017-06-22 10:19:42 +00:00
|
|
|
if(this.warehouse_details.type === 'Source Warehouse') {
|
2017-06-23 06:09:03 +00:00
|
|
|
row.s_warehouse = values.warehouse || warehouse;
|
2017-06-22 10:24:18 +00:00
|
|
|
} else if(this.warehouse_details.type === 'Target Warehouse') {
|
2017-06-23 06:09:03 +00:00
|
|
|
row.t_warehouse = values.warehouse || warehouse;
|
2017-06-22 10:24:18 +00:00
|
|
|
} else {
|
2017-06-23 06:09:03 +00:00
|
|
|
row.warehouse = values.warehouse || warehouse;
|
|
|
|
}
|
2021-02-12 05:59:24 +00:00
|
|
|
|
|
|
|
this.frm.dirty();
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
|
2018-03-28 07:03:15 +00:00
|
|
|
update_total_qty() {
|
2017-06-22 10:19:42 +00:00
|
|
|
let qty_field = this.dialog.fields_dict.qty;
|
2018-03-28 07:03:15 +00:00
|
|
|
let total_qty = 0;
|
2017-06-22 10:19:42 +00:00
|
|
|
|
2018-03-28 07:03:15 +00:00
|
|
|
this.dialog.fields_dict.batches.df.data.forEach(data => {
|
|
|
|
total_qty += flt(data.selected_qty);
|
|
|
|
});
|
|
|
|
|
|
|
|
qty_field.set_input(total_qty);
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
|
2021-06-11 15:31:30 +00:00
|
|
|
update_pending_qtys() {
|
2021-05-14 07:06:41 +00:00
|
|
|
const pending_qty_field = this.dialog.fields_dict.pending_qty;
|
|
|
|
const total_selected_qty_field = this.dialog.fields_dict.total_selected_qty;
|
2017-06-22 10:19:42 +00:00
|
|
|
|
2021-05-14 07:06:41 +00:00
|
|
|
if (!pending_qty_field || !total_selected_qty_field) return;
|
|
|
|
|
|
|
|
const me = this;
|
|
|
|
const required_qty = this.dialog.fields_dict.required_qty.value;
|
|
|
|
const selected_qty = this.dialog.fields_dict.qty.value;
|
|
|
|
const total_selected_qty = selected_qty + calc_total_selected_qty(me);
|
|
|
|
const pending_qty = required_qty - total_selected_qty;
|
|
|
|
|
|
|
|
pending_qty_field.set_input(pending_qty);
|
|
|
|
total_selected_qty_field.set_input(total_selected_qty);
|
2021-05-31 05:51:18 +00:00
|
|
|
}
|
|
|
|
|
2021-06-11 15:31:30 +00:00
|
|
|
get_batch_fields() {
|
2017-06-22 10:19:42 +00:00
|
|
|
var me = this;
|
2019-06-28 08:38:08 +00:00
|
|
|
|
2017-06-22 10:19:42 +00:00
|
|
|
return [
|
|
|
|
{fieldtype:'Section Break', label: __('Batches')},
|
2019-04-16 07:17:39 +00:00
|
|
|
{fieldname: 'batches', fieldtype: 'Table', label: __('Batch Entries'),
|
2017-06-22 10:19:42 +00:00
|
|
|
fields: [
|
|
|
|
{
|
2019-06-10 12:03:36 +00:00
|
|
|
'fieldtype': 'Link',
|
|
|
|
'read_only': 0,
|
|
|
|
'fieldname': 'batch_no',
|
|
|
|
'options': 'Batch',
|
|
|
|
'label': __('Select Batch'),
|
|
|
|
'in_list_view': 1,
|
|
|
|
get_query: function () {
|
2017-11-29 05:23:09 +00:00
|
|
|
return {
|
2019-07-02 04:40:55 +00:00
|
|
|
filters: {
|
|
|
|
item_code: me.item_code,
|
2020-03-26 07:48:57 +00:00
|
|
|
warehouse: me.warehouse || typeof me.warehouse_details.name == "string" ? me.warehouse_details.name : ''
|
2019-07-02 04:40:55 +00:00
|
|
|
},
|
2019-06-28 08:38:08 +00:00
|
|
|
query: 'erpnext.controllers.queries.get_batch_no'
|
2019-06-10 12:03:36 +00:00
|
|
|
};
|
2017-06-22 10:19:42 +00:00
|
|
|
},
|
2019-06-10 12:20:52 +00:00
|
|
|
change: function () {
|
2020-07-23 14:36:26 +00:00
|
|
|
const batch_no = this.get_value();
|
|
|
|
if (!batch_no) {
|
2017-06-22 10:19:42 +00:00
|
|
|
this.grid_row.on_grid_fields_dict
|
|
|
|
.available_qty.set_value(0);
|
|
|
|
return;
|
|
|
|
}
|
2017-06-26 10:01:46 +00:00
|
|
|
let selected_batches = this.grid.grid_rows.map((row) => {
|
2019-06-10 12:03:36 +00:00
|
|
|
if (row === this.grid_row) {
|
2017-06-26 10:01:46 +00:00
|
|
|
return "";
|
|
|
|
}
|
2018-03-28 07:03:15 +00:00
|
|
|
|
|
|
|
if (row.on_grid_fields_dict.batch_no) {
|
|
|
|
return row.on_grid_fields_dict.batch_no.get_value();
|
|
|
|
}
|
2017-06-26 10:01:46 +00:00
|
|
|
});
|
2021-04-15 09:22:13 +00:00
|
|
|
if (selected_batches.includes(batch_no)) {
|
2017-06-26 10:01:46 +00:00
|
|
|
this.set_value("");
|
2021-04-15 09:22:13 +00:00
|
|
|
frappe.throw(__('Batch {0} already selected.', [batch_no]));
|
2017-06-26 10:01:46 +00:00
|
|
|
}
|
2020-03-24 06:04:44 +00:00
|
|
|
|
2019-06-10 12:03:36 +00:00
|
|
|
if (me.warehouse_details.name) {
|
2017-06-22 10:19:42 +00:00
|
|
|
frappe.call({
|
|
|
|
method: 'erpnext.stock.doctype.batch.batch.get_batch_qty',
|
|
|
|
args: {
|
2020-07-23 14:36:26 +00:00
|
|
|
batch_no,
|
2017-06-22 10:19:42 +00:00
|
|
|
warehouse: me.warehouse_details.name,
|
|
|
|
item_code: me.item_code
|
|
|
|
},
|
|
|
|
callback: (r) => {
|
|
|
|
this.grid_row.on_grid_fields_dict
|
|
|
|
.available_qty.set_value(r.message || 0);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
} else {
|
2017-06-26 10:01:46 +00:00
|
|
|
this.set_value("");
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.throw(__('Please select a warehouse to get available quantities'));
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
// e.stopImmediatePropagation();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
2019-06-10 12:03:36 +00:00
|
|
|
'fieldtype': 'Float',
|
|
|
|
'read_only': 1,
|
|
|
|
'fieldname': 'available_qty',
|
|
|
|
'label': __('Available'),
|
|
|
|
'in_list_view': 1,
|
|
|
|
'default': 0,
|
|
|
|
change: function () {
|
2017-06-22 10:19:42 +00:00
|
|
|
this.grid_row.on_grid_fields_dict.selected_qty.set_value('0');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
2019-06-10 12:03:36 +00:00
|
|
|
'fieldtype': 'Float',
|
|
|
|
'read_only': 0,
|
|
|
|
'fieldname': 'selected_qty',
|
|
|
|
'label': __('Qty'),
|
|
|
|
'in_list_view': 1,
|
2017-06-22 10:19:42 +00:00
|
|
|
'default': 0,
|
2019-06-10 12:20:52 +00:00
|
|
|
change: function () {
|
2017-06-22 10:19:42 +00:00
|
|
|
var batch_no = this.grid_row.on_grid_fields_dict.batch_no.get_value();
|
|
|
|
var available_qty = this.grid_row.on_grid_fields_dict.available_qty.get_value();
|
|
|
|
var selected_qty = this.grid_row.on_grid_fields_dict.selected_qty.get_value();
|
|
|
|
|
2019-06-10 12:03:36 +00:00
|
|
|
if (batch_no.length === 0 && parseInt(selected_qty) !== 0) {
|
2017-06-22 10:19:42 +00:00
|
|
|
frappe.throw(__("Please select a batch"));
|
|
|
|
}
|
2019-06-10 12:03:36 +00:00
|
|
|
if (me.warehouse_details.type === 'Source Warehouse' &&
|
2017-06-22 10:19:42 +00:00
|
|
|
parseFloat(available_qty) < parseFloat(selected_qty)) {
|
2017-06-26 12:15:49 +00:00
|
|
|
|
|
|
|
this.set_value('0');
|
2020-10-26 05:47:04 +00:00
|
|
|
frappe.throw(__('For transfer from source, selected quantity cannot be greater than available quantity'));
|
2017-06-22 10:19:42 +00:00
|
|
|
} else {
|
|
|
|
this.grid.refresh();
|
|
|
|
}
|
2018-03-28 07:03:15 +00:00
|
|
|
|
|
|
|
me.update_total_qty();
|
2021-05-14 07:06:41 +00:00
|
|
|
me.update_pending_qtys();
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
],
|
|
|
|
in_place_edit: true,
|
|
|
|
data: this.data,
|
2019-06-10 12:03:36 +00:00
|
|
|
get_data: function () {
|
2017-06-22 10:19:42 +00:00
|
|
|
return this.data;
|
|
|
|
},
|
|
|
|
}
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
get_serial_no_fields() {
|
|
|
|
var me = this;
|
2017-07-16 16:34:16 +00:00
|
|
|
this.serial_list = [];
|
2019-06-03 06:27:00 +00:00
|
|
|
|
|
|
|
let serial_no_filters = {
|
|
|
|
item_code: me.item_code,
|
2019-06-03 07:34:57 +00:00
|
|
|
delivery_document_no: ""
|
2019-06-03 06:27:00 +00:00
|
|
|
}
|
|
|
|
|
2020-02-11 06:14:24 +00:00
|
|
|
if (this.item.batch_no) {
|
2020-01-23 10:19:39 +00:00
|
|
|
serial_no_filters["batch_no"] = this.item.batch_no;
|
|
|
|
}
|
|
|
|
|
2019-06-03 06:27:00 +00:00
|
|
|
if (me.warehouse_details.name) {
|
|
|
|
serial_no_filters['warehouse'] = me.warehouse_details.name;
|
|
|
|
}
|
2019-12-24 12:49:58 +00:00
|
|
|
|
2020-07-23 13:21:26 +00:00
|
|
|
if (me.frm.doc.doctype === 'POS Invoice' && !this.showing_reserved_serial_nos_error) {
|
|
|
|
frappe.call({
|
|
|
|
method: "erpnext.stock.doctype.serial_no.serial_no.get_pos_reserved_serial_nos",
|
|
|
|
args: {
|
2020-10-26 05:47:04 +00:00
|
|
|
filters: {
|
|
|
|
item_code: me.item_code,
|
|
|
|
warehouse: typeof me.warehouse_details.name == "string" ? me.warehouse_details.name : '',
|
|
|
|
}
|
2020-07-23 13:21:26 +00:00
|
|
|
}
|
|
|
|
}).then((data) => {
|
|
|
|
serial_no_filters['name'] = ["not in", data.message[0]]
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-06-22 10:19:42 +00:00
|
|
|
return [
|
2019-04-16 08:02:41 +00:00
|
|
|
{fieldtype: 'Section Break', label: __('Serial Numbers')},
|
2017-06-22 10:19:42 +00:00
|
|
|
{
|
|
|
|
fieldtype: 'Link', fieldname: 'serial_no_select', options: 'Serial No',
|
2019-04-16 08:02:41 +00:00
|
|
|
label: __('Select to add Serial Number.'),
|
2017-06-22 10:19:42 +00:00
|
|
|
get_query: function() {
|
2019-06-03 06:27:00 +00:00
|
|
|
return {
|
|
|
|
filters: serial_no_filters
|
|
|
|
};
|
2017-07-16 16:34:16 +00:00
|
|
|
},
|
|
|
|
onchange: function(e) {
|
|
|
|
if(this.in_local_change) return;
|
|
|
|
this.in_local_change = 1;
|
|
|
|
|
|
|
|
let serial_no_list_field = this.layout.fields_dict.serial_no;
|
|
|
|
let qty_field = this.layout.fields_dict.qty;
|
|
|
|
|
|
|
|
let new_number = this.get_value();
|
|
|
|
let list_value = serial_no_list_field.get_value();
|
|
|
|
let new_line = '\n';
|
|
|
|
if(!list_value) {
|
|
|
|
new_line = '';
|
|
|
|
} else {
|
|
|
|
me.serial_list = list_value.replace(/\n/g, ' ').match(/\S+/g) || [];
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!me.serial_list.includes(new_number)) {
|
|
|
|
this.set_new_description('');
|
|
|
|
serial_no_list_field.set_value(me.serial_list.join('\n') + new_line + new_number);
|
|
|
|
me.serial_list = serial_no_list_field.get_value().replace(/\n/g, ' ').match(/\S+/g) || [];
|
|
|
|
} else {
|
|
|
|
this.set_new_description(new_number + ' is already selected.');
|
|
|
|
}
|
|
|
|
|
|
|
|
qty_field.set_input(me.serial_list.length);
|
|
|
|
this.$input.val("");
|
|
|
|
this.in_local_change = 0;
|
2017-06-22 10:19:42 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
{fieldtype: 'Column Break'},
|
2017-07-16 16:34:16 +00:00
|
|
|
{
|
|
|
|
fieldname: 'serial_no',
|
|
|
|
fieldtype: 'Small Text',
|
2020-03-26 07:48:57 +00:00
|
|
|
label: __(me.has_batch && !me.has_serial_no ? 'Selected Batch Numbers' : 'Selected Serial Numbers'),
|
2017-07-16 16:34:16 +00:00
|
|
|
onchange: function() {
|
|
|
|
me.serial_list = this.get_value()
|
|
|
|
.replace(/\n/g, ' ').match(/\S+/g) || [];
|
|
|
|
this.layout.fields_dict.qty.set_input(me.serial_list.length);
|
|
|
|
}
|
|
|
|
}
|
2017-06-22 10:19:42 +00:00
|
|
|
];
|
|
|
|
}
|
2021-06-11 15:31:30 +00:00
|
|
|
};
|
2021-05-14 07:06:41 +00:00
|
|
|
|
|
|
|
function get_pending_qty_fields(me) {
|
|
|
|
if (!check_can_calculate_pending_qty(me)) return [];
|
|
|
|
const { frm: { doc: { fg_completed_qty }}, item: { item_code, stock_qty }} = me;
|
|
|
|
const { qty_consumed_per_unit } = erpnext.stock.bom.items[item_code];
|
|
|
|
|
|
|
|
const total_selected_qty = calc_total_selected_qty(me);
|
|
|
|
const required_qty = flt(fg_completed_qty) * flt(qty_consumed_per_unit);
|
|
|
|
const pending_qty = required_qty - (flt(stock_qty) + total_selected_qty);
|
|
|
|
|
|
|
|
const pending_qty_fields = [
|
|
|
|
{ fieldtype: 'Section Break', label: __('Pending Quantity') },
|
|
|
|
{
|
|
|
|
fieldname: 'required_qty',
|
|
|
|
read_only: 1,
|
|
|
|
fieldtype: 'Float',
|
|
|
|
label: __('Required Qty'),
|
|
|
|
default: required_qty
|
|
|
|
},
|
|
|
|
{ fieldtype: 'Column Break' },
|
|
|
|
{
|
|
|
|
fieldname: 'total_selected_qty',
|
|
|
|
read_only: 1,
|
|
|
|
fieldtype: 'Float',
|
|
|
|
label: __('Total Selected Qty'),
|
|
|
|
default: total_selected_qty
|
|
|
|
},
|
|
|
|
{ fieldtype: 'Column Break' },
|
|
|
|
{
|
|
|
|
fieldname: 'pending_qty',
|
|
|
|
read_only: 1,
|
|
|
|
fieldtype: 'Float',
|
|
|
|
label: __('Pending Qty'),
|
|
|
|
default: pending_qty
|
|
|
|
},
|
|
|
|
];
|
|
|
|
return pending_qty_fields;
|
|
|
|
}
|
|
|
|
|
2022-03-19 12:19:19 +00:00
|
|
|
// get all items with same item code except row for which selector is open.
|
|
|
|
function get_rows_with_same_item_code(me) {
|
2021-05-14 07:06:41 +00:00
|
|
|
const { frm: { doc: { items }}, item: { name, item_code }} = me;
|
2022-03-19 12:19:19 +00:00
|
|
|
return items.filter(item => (item.name !== name) && (item.item_code === item_code))
|
|
|
|
}
|
|
|
|
|
|
|
|
function calc_total_selected_qty(me) {
|
|
|
|
const totalSelectedQty = get_rows_with_same_item_code(me)
|
|
|
|
.map(item => flt(item.qty))
|
|
|
|
.reduce((i, j) => i + j, 0);
|
2021-05-14 07:06:41 +00:00
|
|
|
return totalSelectedQty;
|
|
|
|
}
|
|
|
|
|
2022-03-19 12:19:19 +00:00
|
|
|
function get_selected_serial_nos(me) {
|
|
|
|
const selected_serial_nos = get_rows_with_same_item_code(me)
|
|
|
|
.map(item => item.serial_no)
|
|
|
|
.filter(serial => serial)
|
|
|
|
.map(sr_no_string => sr_no_string.split('\n'))
|
|
|
|
.reduce((acc, arr) => acc.concat(arr), [])
|
|
|
|
.filter(serial => serial);
|
|
|
|
return selected_serial_nos;
|
|
|
|
};
|
|
|
|
|
2021-05-14 07:06:41 +00:00
|
|
|
function check_can_calculate_pending_qty(me) {
|
|
|
|
const { frm: { doc }, item } = me;
|
|
|
|
const docChecks = doc.bom_no
|
|
|
|
&& doc.fg_completed_qty
|
|
|
|
&& erpnext.stock.bom
|
|
|
|
&& erpnext.stock.bom.name === doc.bom_no;
|
2022-03-23 09:59:50 +00:00
|
|
|
const itemChecks = !!item
|
2022-04-06 11:05:36 +00:00
|
|
|
&& !item.original_item
|
|
|
|
&& erpnext.stock.bom && erpnext.stock.bom.items
|
2022-03-23 09:59:50 +00:00
|
|
|
&& (item.item_code in erpnext.stock.bom.items);
|
2021-05-14 07:06:41 +00:00
|
|
|
return docChecks && itemChecks;
|
|
|
|
}
|
2022-03-23 09:53:26 +00:00
|
|
|
|
|
|
|
//# sourceURL=serial_no_batch_selector.js
|
2022-10-10 07:58:19 +00:00
|
|
|
|
|
|
|
|
|
|
|
erpnext.SerialNoBatchBundleUpdate = class SerialNoBatchBundleUpdate {
|
|
|
|
constructor(frm, item, callback) {
|
|
|
|
this.frm = frm;
|
|
|
|
this.item = item;
|
|
|
|
this.qty = item.qty;
|
|
|
|
this.callback = callback;
|
|
|
|
this.make();
|
|
|
|
this.render_data();
|
|
|
|
}
|
|
|
|
|
|
|
|
make() {
|
|
|
|
this.dialog = new frappe.ui.Dialog({
|
|
|
|
title: __('Update Serial No / Batch No'),
|
|
|
|
fields: this.get_dialog_fields(),
|
|
|
|
primary_action_label: __('Update'),
|
|
|
|
primary_action: () => this.update_ledgers()
|
|
|
|
});
|
|
|
|
this.dialog.show();
|
|
|
|
}
|
|
|
|
|
|
|
|
get_serial_no_filters() {
|
|
|
|
return {
|
|
|
|
'item_code': this.item.item_code,
|
|
|
|
'warehouse': ["=", ""],
|
|
|
|
'delivery_document_no': ["=", ""],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
get_dialog_fields() {
|
|
|
|
let fields = [];
|
|
|
|
|
|
|
|
if (this.item.has_serial_no) {
|
|
|
|
fields.push({
|
|
|
|
fieldtype: 'Link',
|
|
|
|
fieldname: 'scan_serial_no',
|
|
|
|
label: __('Scan Serial No'),
|
|
|
|
options: 'Serial No',
|
|
|
|
get_query: () => {
|
|
|
|
return {
|
|
|
|
filters: this.get_serial_no_filters()
|
|
|
|
};
|
|
|
|
},
|
|
|
|
onchange: () => this.update_serial_batch_no()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.item.has_batch_no && this.item.has_serial_no) {
|
|
|
|
fields.push({
|
|
|
|
fieldtype: 'Column Break',
|
|
|
|
label: __('Batch No')
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.item.has_batch_no) {
|
|
|
|
fields.push({
|
|
|
|
fieldtype: 'Link',
|
|
|
|
fieldname: 'scan_batch_no',
|
|
|
|
label: __('Scan Batch No'),
|
|
|
|
options: 'Batch',
|
|
|
|
onchange: () => this.update_serial_batch_no()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.item.has_batch_no && this.item.has_serial_no) {
|
|
|
|
fields.push({
|
|
|
|
fieldtype: 'Section Break',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
fields.push({
|
|
|
|
fieldname: 'ledgers',
|
|
|
|
fieldtype: 'Table',
|
|
|
|
allow_bulk_edit: true,
|
|
|
|
data: [],
|
|
|
|
fields: this.get_dialog_table_fields(),
|
|
|
|
});
|
|
|
|
|
|
|
|
return fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_dialog_table_fields() {
|
|
|
|
let fields = []
|
|
|
|
|
|
|
|
if (this.item.has_serial_no) {
|
|
|
|
fields.push({
|
|
|
|
fieldtype: 'Link',
|
|
|
|
options: 'Serial No',
|
|
|
|
fieldname: 'serial_no',
|
|
|
|
label: __('Serial No'),
|
|
|
|
in_list_view: 1,
|
|
|
|
get_query: () => {
|
|
|
|
return {
|
|
|
|
filters: this.get_serial_no_filters()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else if (this.item.has_batch_no) {
|
|
|
|
fields = [
|
|
|
|
{
|
|
|
|
fieldtype: 'Link',
|
|
|
|
options: 'Batch',
|
|
|
|
fieldname: 'batch_no',
|
|
|
|
label: __('Batch No'),
|
|
|
|
in_list_view: 1,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldtype: 'Float',
|
|
|
|
fieldname: 'qty',
|
|
|
|
label: __('Quantity'),
|
|
|
|
in_list_view: 1,
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
fields.push({
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'name',
|
|
|
|
label: __('Name'),
|
|
|
|
hidden: 1,
|
|
|
|
})
|
|
|
|
|
|
|
|
return fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
update_serial_batch_no() {
|
|
|
|
const { scan_serial_no, scan_batch_no } = this.dialog.get_values();
|
|
|
|
|
|
|
|
if (scan_serial_no) {
|
|
|
|
this.dialog.fields_dict.ledgers.df.data.push({
|
|
|
|
serial_no: scan_serial_no
|
|
|
|
});
|
|
|
|
|
|
|
|
this.dialog.fields_dict.scan_serial_no.set_value('');
|
|
|
|
} else if (scan_batch_no) {
|
|
|
|
this.dialog.fields_dict.ledgers.df.data.push({
|
|
|
|
batch_no: scan_batch_no
|
|
|
|
});
|
|
|
|
|
|
|
|
this.dialog.fields_dict.scan_batch_no.set_value('');
|
|
|
|
}
|
|
|
|
|
|
|
|
this.dialog.fields_dict.ledgers.grid.refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
update_ledgers() {
|
|
|
|
if (!this.frm.is_new()) {
|
|
|
|
let ledgers = this.dialog.get_values().ledgers;
|
|
|
|
|
|
|
|
if (ledgers && !ledgers.length) {
|
|
|
|
frappe.throw(__('Please add atleast one Serial No / Batch No'));
|
|
|
|
}
|
|
|
|
|
|
|
|
frappe.call({
|
2022-11-24 11:46:21 +00:00
|
|
|
method: 'erpnext.stock.doctype.serial_and_batch_bundle.serial_and_batch_bundle.add_serial_batch_ledgers',
|
2022-10-10 07:58:19 +00:00
|
|
|
args: {
|
|
|
|
ledgers: ledgers,
|
|
|
|
child_row: this.item
|
|
|
|
}
|
|
|
|
}).then(r => {
|
|
|
|
this.callback && this.callback(r.message);
|
|
|
|
this.dialog.hide();
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render_data() {
|
|
|
|
if (!this.frm.is_new() && this.item.serial_and_batch_bundle) {
|
|
|
|
frappe.call({
|
2022-11-24 11:46:21 +00:00
|
|
|
method: 'erpnext.stock.doctype.serial_and_batch_bundle.serial_and_batch_bundle.get_serial_batch_ledgers',
|
2022-10-10 07:58:19 +00:00
|
|
|
args: {
|
|
|
|
item_code: this.item.item_code,
|
|
|
|
name: this.item.serial_and_batch_bundle,
|
|
|
|
voucher_no: this.item.parent,
|
|
|
|
}
|
|
|
|
}).then(r => {
|
|
|
|
if (r.message) {
|
|
|
|
this.set_data(r.message);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
set_data(data) {
|
|
|
|
data.forEach(d => {
|
|
|
|
this.dialog.fields_dict.ledgers.df.data.push(d);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.dialog.fields_dict.ledgers.grid.refresh();
|
|
|
|
}
|
|
|
|
}
|