2016-04-25 17:53:42 +05:30
|
|
|
frappe.provide('erpnext.stock');
|
|
|
|
|
2021-04-23 08:04:00 +05:30
|
|
|
erpnext.stock.ItemDashboard = class ItemDashboard {
|
|
|
|
constructor(opts) {
|
2016-04-25 17:53:42 +05:30
|
|
|
$.extend(this, opts);
|
|
|
|
this.make();
|
2021-04-23 08:04:00 +05:30
|
|
|
}
|
|
|
|
make() {
|
2016-04-25 17:53:42 +05:30
|
|
|
var me = this;
|
|
|
|
this.start = 0;
|
2021-03-17 18:40:21 +05:30
|
|
|
if (!this.sort_by) {
|
2016-04-25 17:53:42 +05:30
|
|
|
this.sort_by = 'projected_qty';
|
|
|
|
this.sort_order = 'asc';
|
|
|
|
}
|
|
|
|
|
|
|
|
this.content = $(frappe.render_template('item_dashboard')).appendTo(this.parent);
|
|
|
|
this.result = this.content.find('.result');
|
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
this.content.on('click', '.btn-move', function () {
|
|
|
|
handle_move_add($(this), "Move");
|
2019-07-05 10:38:48 +05:30
|
|
|
});
|
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
this.content.on('click', '.btn-add', function () {
|
|
|
|
handle_move_add($(this), "Add");
|
2019-07-16 12:52:19 +05:30
|
|
|
});
|
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
this.content.on('click', '.btn-edit', function () {
|
2020-11-26 10:45:44 +05:30
|
|
|
let item = unescape($(this).attr('data-item'));
|
|
|
|
let warehouse = unescape($(this).attr('data-warehouse'));
|
|
|
|
let company = unescape($(this).attr('data-company'));
|
2021-03-17 18:40:21 +05:30
|
|
|
frappe.db.get_value('Putaway Rule', {
|
|
|
|
'item_code': item,
|
|
|
|
'warehouse': warehouse,
|
|
|
|
'company': company
|
|
|
|
}, 'name', (r) => {
|
|
|
|
frappe.set_route("Form", "Putaway Rule", r.name);
|
|
|
|
});
|
2020-11-26 10:45:44 +05:30
|
|
|
});
|
|
|
|
|
2019-07-16 12:52:19 +05:30
|
|
|
function handle_move_add(element, action) {
|
|
|
|
let item = unescape(element.attr('data-item'));
|
|
|
|
let warehouse = unescape(element.attr('data-warehouse'));
|
|
|
|
let actual_qty = unescape(element.attr('data-actual_qty'));
|
|
|
|
let disable_quick_entry = Number(unescape(element.attr('data-disable_quick_entry')));
|
2023-02-03 18:08:34 +05:30
|
|
|
let entry_type = action === "Move" ? "Material Transfer" : "Material Receipt";
|
2019-07-10 10:22:08 +05:30
|
|
|
|
2019-07-16 12:32:50 +05:30
|
|
|
if (disable_quick_entry) {
|
2019-07-16 12:52:19 +05:30
|
|
|
open_stock_entry(item, warehouse, entry_type);
|
2019-07-16 12:32:50 +05:30
|
|
|
} else {
|
2019-07-16 12:52:19 +05:30
|
|
|
if (action === "Add") {
|
|
|
|
let rate = unescape($(this).attr('data-rate'));
|
2021-03-17 18:40:21 +05:30
|
|
|
erpnext.stock.move_item(item, null, warehouse, actual_qty, rate, function () {
|
|
|
|
me.refresh();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
erpnext.stock.move_item(item, warehouse, null, actual_qty, null, function () {
|
|
|
|
me.refresh();
|
|
|
|
});
|
2019-07-16 12:52:19 +05:30
|
|
|
}
|
2019-07-16 12:32:50 +05:30
|
|
|
}
|
2019-07-16 12:52:19 +05:30
|
|
|
}
|
2019-07-05 10:38:48 +05:30
|
|
|
|
|
|
|
function open_stock_entry(item, warehouse, entry_type) {
|
2021-03-17 18:40:21 +05:30
|
|
|
frappe.model.with_doctype('Stock Entry', function () {
|
2019-07-05 10:38:48 +05:30
|
|
|
var doc = frappe.model.get_new_doc('Stock Entry');
|
2023-02-03 18:08:34 +05:30
|
|
|
if (entry_type) {
|
|
|
|
doc.stock_entry_type = entry_type;
|
|
|
|
}
|
2019-07-05 10:38:48 +05:30
|
|
|
|
|
|
|
var row = frappe.model.add_child(doc, 'items');
|
|
|
|
row.item_code = item;
|
2023-02-03 18:08:34 +05:30
|
|
|
|
|
|
|
if (entry_type === "Material Transfer") {
|
|
|
|
row.s_warehouse = warehouse;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
row.t_warehouse = warehouse;
|
|
|
|
}
|
2019-07-05 10:38:48 +05:30
|
|
|
|
|
|
|
frappe.set_route('Form', doc.doctype, doc.name);
|
2021-03-17 18:40:21 +05:30
|
|
|
});
|
2019-07-05 10:38:48 +05:30
|
|
|
}
|
|
|
|
|
2016-04-25 17:53:42 +05:30
|
|
|
// more
|
2021-03-17 18:40:21 +05:30
|
|
|
this.content.find('.btn-more').on('click', function () {
|
2020-12-09 16:27:18 +05:30
|
|
|
me.start += me.page_length;
|
2016-04-25 17:53:42 +05:30
|
|
|
me.refresh();
|
|
|
|
});
|
|
|
|
|
2021-04-23 08:04:00 +05:30
|
|
|
}
|
|
|
|
refresh() {
|
2016-04-25 17:53:42 +05:30
|
|
|
if(this.before_refresh) {
|
|
|
|
this.before_refresh();
|
|
|
|
}
|
|
|
|
|
2020-11-26 10:45:44 +05:30
|
|
|
let args = {
|
|
|
|
item_code: this.item_code,
|
|
|
|
warehouse: this.warehouse,
|
|
|
|
parent_warehouse: this.parent_warehouse,
|
|
|
|
item_group: this.item_group,
|
|
|
|
company: this.company,
|
|
|
|
start: this.start,
|
|
|
|
sort_by: this.sort_by,
|
|
|
|
sort_order: this.sort_order
|
2020-12-08 19:11:51 +05:30
|
|
|
};
|
2020-11-26 10:45:44 +05:30
|
|
|
|
2016-04-25 17:53:42 +05:30
|
|
|
var me = this;
|
|
|
|
frappe.call({
|
2020-11-26 10:45:44 +05:30
|
|
|
method: this.method,
|
|
|
|
args: args,
|
2021-03-17 18:40:21 +05:30
|
|
|
callback: function (r) {
|
2016-04-25 17:53:42 +05:30
|
|
|
me.render(r.message);
|
2022-12-29 00:01:26 -05:00
|
|
|
if(me.after_refresh) {
|
|
|
|
me.after_refresh();
|
|
|
|
}
|
2016-04-25 17:53:42 +05:30
|
|
|
}
|
|
|
|
});
|
2021-04-23 08:04:00 +05:30
|
|
|
}
|
|
|
|
render(data) {
|
2020-11-26 10:45:44 +05:30
|
|
|
if (this.start===0) {
|
2016-04-25 17:53:42 +05:30
|
|
|
this.max_count = 0;
|
|
|
|
this.result.empty();
|
|
|
|
}
|
|
|
|
|
2020-12-08 19:11:51 +05:30
|
|
|
let context = "";
|
2020-11-26 10:45:44 +05:30
|
|
|
if (this.page_name === "warehouse-capacity-summary") {
|
2020-12-08 19:11:51 +05:30
|
|
|
context = this.get_capacity_dashboard_data(data);
|
2020-11-26 10:45:44 +05:30
|
|
|
} else {
|
2020-12-08 19:11:51 +05:30
|
|
|
context = this.get_item_dashboard_data(data, this.max_count, true);
|
2020-11-26 10:45:44 +05:30
|
|
|
}
|
2016-04-25 17:53:42 +05:30
|
|
|
|
|
|
|
// show more button
|
2021-03-17 18:40:21 +05:30
|
|
|
if (data && data.length === (this.page_length + 1)) {
|
2016-04-25 17:53:42 +05:30
|
|
|
this.content.find('.more').removeClass('hidden');
|
|
|
|
|
|
|
|
// remove the last element
|
|
|
|
data.splice(-1);
|
|
|
|
} else {
|
|
|
|
this.content.find('.more').addClass('hidden');
|
|
|
|
}
|
|
|
|
|
2018-01-24 13:51:26 +05:30
|
|
|
// If not any stock in any warehouses provide a message to end user
|
2017-10-05 16:14:50 +03:00
|
|
|
if (context.data.length > 0) {
|
2020-11-26 10:45:44 +05:30
|
|
|
this.content.find('.result').css('text-align', 'unset');
|
|
|
|
$(frappe.render_template(this.template, context)).appendTo(this.result);
|
2017-10-05 16:14:50 +03:00
|
|
|
} else {
|
2020-11-26 10:45:44 +05:30
|
|
|
var message = __("No Stock Available Currently");
|
|
|
|
this.content.find('.result').css('text-align', 'center');
|
|
|
|
|
2021-02-07 22:27:40 +05:30
|
|
|
$(`<div class='text-muted' style='margin: 20px 5px;'>
|
2020-11-26 10:45:44 +05:30
|
|
|
${message} </div>`).appendTo(this.result);
|
2017-10-05 16:14:50 +03:00
|
|
|
}
|
2021-04-23 08:04:00 +05:30
|
|
|
}
|
2020-11-26 10:45:44 +05:30
|
|
|
|
2021-04-23 08:04:00 +05:30
|
|
|
get_item_dashboard_data(data, max_count, show_item) {
|
2016-04-25 17:53:42 +05:30
|
|
|
if(!max_count) max_count = 0;
|
2016-05-31 07:22:37 +05:30
|
|
|
if(!data) data = [];
|
2018-01-24 19:47:59 +05:30
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
data.forEach(function (d) {
|
2018-03-28 11:16:00 +05:30
|
|
|
d.actual_or_pending = d.projected_qty + d.reserved_qty + d.reserved_qty_for_production + d.reserved_qty_for_sub_contract;
|
2016-04-25 17:53:42 +05:30
|
|
|
d.pending_qty = 0;
|
2018-03-12 14:12:12 +05:30
|
|
|
d.total_reserved = d.reserved_qty + d.reserved_qty_for_production + d.reserved_qty_for_sub_contract;
|
2021-03-17 18:40:21 +05:30
|
|
|
if (d.actual_or_pending > d.actual_qty) {
|
2016-04-25 17:53:42 +05:30
|
|
|
d.pending_qty = d.actual_or_pending - d.actual_qty;
|
|
|
|
}
|
|
|
|
|
|
|
|
max_count = Math.max(d.actual_or_pending, d.actual_qty,
|
|
|
|
d.total_reserved, max_count);
|
|
|
|
});
|
2018-01-24 19:47:59 +05:30
|
|
|
|
2020-11-26 10:45:44 +05:30
|
|
|
let can_write = 0;
|
2020-12-08 19:11:51 +05:30
|
|
|
if (frappe.boot.user.can_write.indexOf("Stock Entry") >= 0) {
|
2018-01-24 19:47:59 +05:30
|
|
|
can_write = 1;
|
|
|
|
}
|
|
|
|
|
2016-04-25 17:53:42 +05:30
|
|
|
return {
|
|
|
|
data: data,
|
|
|
|
max_count: max_count,
|
2021-03-17 18:40:21 +05:30
|
|
|
can_write: can_write,
|
2016-04-25 17:53:42 +05:30
|
|
|
show_item: show_item || false
|
2021-01-05 12:16:25 +05:30
|
|
|
};
|
2021-04-23 08:04:00 +05:30
|
|
|
}
|
2020-11-26 10:45:44 +05:30
|
|
|
|
2021-04-23 08:04:00 +05:30
|
|
|
get_capacity_dashboard_data(data) {
|
2020-12-08 19:11:51 +05:30
|
|
|
if (!data) data = [];
|
2020-11-26 10:45:44 +05:30
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
data.forEach(function (d) {
|
|
|
|
d.color = d.percent_occupied >= 80 ? "#f8814f" : "#2490ef";
|
2020-11-26 10:45:44 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
let can_write = 0;
|
2020-12-08 19:11:51 +05:30
|
|
|
if (frappe.boot.user.can_write.indexOf("Putaway Rule") >= 0) {
|
2020-11-26 10:45:44 +05:30
|
|
|
can_write = 1;
|
2016-04-25 17:53:42 +05:30
|
|
|
}
|
2020-11-26 10:45:44 +05:30
|
|
|
|
|
|
|
return {
|
|
|
|
data: data,
|
|
|
|
can_write: can_write,
|
2020-12-08 19:11:51 +05:30
|
|
|
};
|
2016-04-25 17:53:42 +05:30
|
|
|
}
|
2021-04-23 08:04:00 +05:30
|
|
|
};
|
2019-07-10 10:22:08 +05:30
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
erpnext.stock.move_item = function (item, source, target, actual_qty, rate, callback) {
|
2019-07-10 10:22:08 +05:30
|
|
|
var dialog = new frappe.ui.Dialog({
|
|
|
|
title: target ? __('Add Item') : __('Move Item'),
|
2021-03-17 18:40:21 +05:30
|
|
|
fields: [{
|
|
|
|
fieldname: 'item_code',
|
|
|
|
label: __('Item'),
|
|
|
|
fieldtype: 'Link',
|
|
|
|
options: 'Item',
|
|
|
|
read_only: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldname: 'source',
|
|
|
|
label: __('Source Warehouse'),
|
|
|
|
fieldtype: 'Link',
|
|
|
|
options: 'Warehouse',
|
|
|
|
read_only: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldname: 'target',
|
|
|
|
label: __('Target Warehouse'),
|
|
|
|
fieldtype: 'Link',
|
|
|
|
options: 'Warehouse',
|
2022-01-26 01:01:10 +05:30
|
|
|
reqd: 1,
|
|
|
|
get_query() {
|
|
|
|
return {
|
|
|
|
filters: {
|
|
|
|
is_group: 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-17 18:40:21 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldname: 'qty',
|
|
|
|
label: __('Quantity'),
|
|
|
|
reqd: 1,
|
|
|
|
fieldtype: 'Float',
|
|
|
|
description: __('Available {0}', [actual_qty])
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldname: 'rate',
|
|
|
|
label: __('Rate'),
|
|
|
|
fieldtype: 'Currency',
|
|
|
|
hidden: 1
|
|
|
|
},
|
2019-07-10 10:22:08 +05:30
|
|
|
],
|
2021-03-17 18:40:21 +05:30
|
|
|
});
|
2019-07-10 10:22:08 +05:30
|
|
|
dialog.show();
|
|
|
|
dialog.get_field('item_code').set_input(item);
|
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
if (source) {
|
2019-07-10 10:22:08 +05:30
|
|
|
dialog.get_field('source').set_input(source);
|
|
|
|
} else {
|
|
|
|
dialog.get_field('source').df.hidden = 1;
|
|
|
|
dialog.get_field('source').refresh();
|
|
|
|
}
|
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
if (rate) {
|
2019-07-10 10:22:08 +05:30
|
|
|
dialog.get_field('rate').set_value(rate);
|
|
|
|
dialog.get_field('rate').df.hidden = 0;
|
|
|
|
dialog.get_field('rate').refresh();
|
|
|
|
}
|
|
|
|
|
2021-03-17 18:40:21 +05:30
|
|
|
if (target) {
|
2019-07-10 10:22:08 +05:30
|
|
|
dialog.get_field('target').df.read_only = 1;
|
|
|
|
dialog.get_field('target').value = target;
|
|
|
|
dialog.get_field('target').refresh();
|
|
|
|
}
|
|
|
|
|
2022-01-26 01:01:33 +05:30
|
|
|
dialog.set_primary_action(__('Create Stock Entry'), function () {
|
2022-06-29 02:36:47 -04:00
|
|
|
if (source && (dialog.get_value("qty") == 0 || dialog.get_value("qty") > actual_qty)) {
|
|
|
|
frappe.msgprint(__("Quantity must be greater than zero, and less or equal to {0}", [actual_qty]));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dialog.get_value("source") === dialog.get_value("target")) {
|
|
|
|
frappe.msgprint(__("Source and target warehouse must be different"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-26 01:01:33 +05:30
|
|
|
frappe.model.with_doctype('Stock Entry', function () {
|
|
|
|
let doc = frappe.model.get_new_doc('Stock Entry');
|
|
|
|
doc.from_warehouse = dialog.get_value('source');
|
|
|
|
doc.to_warehouse = dialog.get_value('target');
|
|
|
|
doc.stock_entry_type = doc.from_warehouse ? "Material Transfer" : "Material Receipt";
|
|
|
|
let row = frappe.model.add_child(doc, 'items');
|
|
|
|
row.item_code = dialog.get_value('item_code');
|
2022-02-28 13:04:24 +05:30
|
|
|
row.s_warehouse = dialog.get_value('source');
|
2022-01-26 01:01:33 +05:30
|
|
|
row.t_warehouse = dialog.get_value('target');
|
|
|
|
row.qty = dialog.get_value('qty');
|
|
|
|
row.conversion_factor = 1;
|
|
|
|
row.transfer_qty = dialog.get_value('qty');
|
|
|
|
row.basic_rate = dialog.get_value('rate');
|
|
|
|
frappe.set_route('Form', doc.doctype, doc.name);
|
2019-07-10 10:22:08 +05:30
|
|
|
});
|
|
|
|
});
|
2021-03-17 18:40:21 +05:30
|
|
|
};
|