brotherton-erpnext/erpnext/public/js/pos/pos.js

507 lines
15 KiB
JavaScript
Raw Normal View History

2013-11-20 07:29:58 +00:00
// Copyright (c) 2013, Web Notes Technologies Pvt. Ltd. and Contributors
// License: GNU General Public License v3. See license.txt
erpnext.POS = Class.extend({
init: function(wrapper, frm) {
this.wrapper = wrapper;
this.frm = frm;
2014-12-19 10:50:32 +00:00
this.wrapper.html(frappe.render_template("pos", {}));
2014-05-05 05:31:32 +00:00
2013-11-27 10:46:32 +00:00
this.check_transaction_type();
this.make();
var me = this;
$(this.frm.wrapper).on("refresh-fields", function() {
me.refresh();
});
this.wrapper.find('input.discount-amount').on("change", function() {
2014-02-14 10:17:51 +00:00
frappe.model.set_value(me.frm.doctype, me.frm.docname, "discount_amount", this.value);
});
2014-01-10 09:35:44 +00:00
this.call_function("remove-items", function() {me.remove_selected_items();});
2013-11-27 10:46:32 +00:00
this.call_function("make-payment", function() {me.make_payment();});
},
check_transaction_type: function() {
var me = this;
2013-08-22 11:56:43 +00:00
2013-11-27 10:46:32 +00:00
// Check whether the transaction is "Sales" or "Purchase"
2014-02-14 10:17:51 +00:00
if (frappe.meta.has_field(cur_frm.doc.doctype, "customer")) {
2013-11-27 10:46:32 +00:00
this.set_transaction_defaults("Customer", "export");
}
2014-02-14 10:17:51 +00:00
else if (frappe.meta.has_field(cur_frm.doc.doctype, "supplier")) {
2013-11-27 10:46:32 +00:00
this.set_transaction_defaults("Supplier", "import");
}
},
set_transaction_defaults: function(party, export_or_import) {
var me = this;
this.party = party;
2014-05-05 05:31:32 +00:00
this.price_list = (party == "Customer" ?
2013-11-27 10:46:32 +00:00
this.frm.doc.selling_price_list : this.frm.doc.buying_price_list);
this.price_list_field = (party == "Customer" ? "selling_price_list" : "buying_price_list");
2013-11-27 10:46:32 +00:00
this.sales_or_purchase = (party == "Customer" ? "Sales" : "Purchase");
this.net_total = "net_total_" + export_or_import;
this.grand_total = "grand_total_" + export_or_import;
2014-02-10 13:50:15 +00:00
// this.amount = export_or_import + "_amount";
// this.rate = export_or_import + "_rate";
2013-11-27 10:46:32 +00:00
},
call_function: function(class_name, fn, event_name) {
this.wrapper.find("." + class_name).on(event_name || "click", fn);
},
make: function() {
this.make_party();
2013-08-22 11:56:43 +00:00
this.make_barcode();
this.make_search();
this.make_item_group();
2013-08-22 11:56:43 +00:00
this.make_item_list();
},
make_party: function() {
var me = this;
2014-02-14 10:17:51 +00:00
this.party_field = frappe.ui.form.make_control({
df: {
"fieldtype": "Link",
"options": this.party,
"label": this.party,
"fieldname": "pos_party",
"placeholder": this.party
},
2013-10-28 12:51:07 +00:00
parent: this.wrapper.find(".party-area"),
only_input: true,
});
this.party_field.make_input();
this.party_field.$input.on("change", function() {
if(!me.party_field.autocomplete_open)
2014-05-05 05:31:32 +00:00
frappe.model.set_value(me.frm.doctype, me.frm.docname,
me.party.toLowerCase(), this.value);
2013-08-22 11:56:43 +00:00
});
},
make_barcode: function() {
var me = this;
2014-02-14 10:17:51 +00:00
this.barcode = frappe.ui.form.make_control({
2013-08-22 11:56:43 +00:00
df: {
"fieldtype": "Data",
"label": "Barcode",
"fieldname": "pos_barcode",
"placeholder": "Barcode / Serial No"
2013-08-22 11:56:43 +00:00
},
parent: this.wrapper.find(".barcode-area"),
2013-10-28 12:51:07 +00:00
only_input: true,
2013-08-22 11:56:43 +00:00
});
this.barcode.make_input();
this.barcode.$input.on("keypress", function() {
if(me.barcode_timeout)
clearTimeout(me.barcode_timeout);
me.barcode_timeout = setTimeout(function() { me.add_item_thru_barcode(); }, 1000);
2013-08-22 11:56:43 +00:00
});
},
make_search: function() {
var me = this;
2014-02-14 10:17:51 +00:00
this.search = frappe.ui.form.make_control({
2013-08-22 11:56:43 +00:00
df: {
2013-11-27 10:46:32 +00:00
"fieldtype": "Data",
2013-08-22 11:56:43 +00:00
"label": "Item",
2013-08-23 07:46:22 +00:00
"fieldname": "pos_item",
2013-11-27 10:46:32 +00:00
"placeholder": "Search Item"
2013-08-22 11:56:43 +00:00
},
2013-10-28 12:51:07 +00:00
parent: this.wrapper.find(".search-area"),
only_input: true,
2013-08-22 11:56:43 +00:00
});
this.search.make_input();
2013-11-27 10:46:32 +00:00
this.search.$input.on("keypress", function() {
2013-08-22 11:56:43 +00:00
if(!me.search.autocomplete_open)
2013-11-27 10:46:32 +00:00
if(me.item_timeout)
clearTimeout(me.item_timeout);
me.item_timeout = setTimeout(function() { me.make_item_list(); }, 1000);
2013-08-22 11:56:43 +00:00
});
},
make_item_group: function() {
2013-08-22 11:56:43 +00:00
var me = this;
2014-02-14 10:17:51 +00:00
this.item_group = frappe.ui.form.make_control({
2013-08-22 11:56:43 +00:00
df: {
"fieldtype": "Link",
"options": "Item Group",
"label": "Item Group",
"fieldname": "pos_item_group",
"placeholder": "Item Group"
2013-08-22 11:56:43 +00:00
},
parent: this.wrapper.find(".item-group-area"),
2013-10-28 12:51:07 +00:00
only_input: true,
2013-08-22 11:56:43 +00:00
});
this.item_group.make_input();
this.item_group.$input.on("change", function() {
if(!me.item_group.autocomplete_open)
me.make_item_list();
2013-08-22 11:56:43 +00:00
});
},
make_item_list: function() {
var me = this;
2014-11-28 09:26:10 +00:00
if(!this.price_list) {
msgprint(__("Price List not found or disabled"));
return;
}
2013-11-27 10:46:32 +00:00
me.item_timeout = null;
2014-02-14 10:17:51 +00:00
frappe.call({
method: 'erpnext.accounts.doctype.sales_invoice.pos.get_items',
2013-08-22 11:56:43 +00:00
args: {
sales_or_purchase: this.sales_or_purchase,
price_list: this.price_list,
2013-08-22 11:56:43 +00:00
item_group: this.item_group.$input.val(),
item: this.search.$input.val()
},
callback: function(r) {
var $wrap = me.wrapper.find(".item-list");
me.wrapper.find(".item-list").empty();
2013-11-27 10:46:32 +00:00
if (r.message) {
$.each(r.message, function(index, obj) {
if (obj.image)
image = '<img src="' + obj.image + '" class="img-responsive" \
style="border:1px solid #eee; max-height: 140px;">';
else
2015-01-01 07:19:18 +00:00
image = '<div class="missing-image"><i class="octicon octicon-circle-slash"></i></div>';
2013-08-22 11:56:43 +00:00
2013-11-27 10:46:32 +00:00
$(repl('<div class="col-xs-3 pos-item" data-item_code="%(item_code)s">\
<div style="height: 140px; overflow: hidden;">%(item_image)s</div>\
<div class="small">%(item_code)s</div>\
<div class="small">%(item_name)s</div>\
<div class="small">%(item_price)s</div>\
2014-05-05 05:31:32 +00:00
</div>',
2013-11-27 10:46:32 +00:00
{
item_code: obj.name,
2014-02-10 12:24:04 +00:00
item_price: format_currency(obj.price_list_rate, obj.currency),
2013-11-27 10:46:32 +00:00
item_name: obj.name===obj.item_name ? "" : obj.item_name,
item_image: image
})).appendTo($wrap);
});
}
2013-08-22 11:56:43 +00:00
2013-09-02 13:18:39 +00:00
// if form is local then allow this function
$(me.wrapper).find("div.pos-item").on("click", function() {
if(me.frm.doc.docstatus==0) {
2014-12-19 10:50:32 +00:00
console.log($(this).attr("data-item_code"));
me.add_to_cart($(this).attr("data-item_code"));
}
});
2013-08-22 11:56:43 +00:00
}
});
},
add_to_cart: function(item_code, serial_no) {
2013-08-22 11:56:43 +00:00
var me = this;
var caught = false;
2014-12-19 10:50:32 +00:00
if(!me.frm.doc[me.party.toLowerCase()] && ((me.frm.doctype == "Quotation" &&
me.frm.doc.quotation_to == "Customer")
|| me.frm.doctype != "Quotation")) {
msgprint(__("Please select {0} first.", [me.party]));
return;
}
2013-08-22 11:56:43 +00:00
// get no_of_items
2014-12-19 10:50:32 +00:00
var no_of_items = me.wrapper.find(".pos-bill-item").length;
2014-05-05 05:31:32 +00:00
2013-08-22 11:56:43 +00:00
// check whether the item is already added
if (no_of_items != 0) {
2014-12-26 07:45:21 +00:00
$.each(this.frm.doc["items"] || [], function(i, d) {
if (d.item_code == item_code) {
2013-08-22 11:56:43 +00:00
caught = true;
if (serial_no)
2014-02-14 10:17:51 +00:00
frappe.model.set_value(d.doctype, d.name, "serial_no", d.serial_no + '\n' + serial_no);
else
2014-02-14 10:17:51 +00:00
frappe.model.set_value(d.doctype, d.name, "qty", d.qty + 1);
}
2013-08-22 11:56:43 +00:00
});
}
2014-05-05 05:31:32 +00:00
// if item not found then add new item
if (!caught)
2013-11-27 10:46:32 +00:00
this.add_new_item_to_grid(item_code, serial_no);
this.refresh();
this.refresh_search_box();
},
add_new_item_to_grid: function(item_code, serial_no) {
var me = this;
2014-12-26 07:45:21 +00:00
var child = frappe.model.add_child(me.frm.doc, this.frm.doctype + " Item", "items");
2013-11-27 10:46:32 +00:00
child.item_code = item_code;
2013-11-27 10:46:32 +00:00
if (serial_no)
child.serial_no = serial_no;
2013-11-27 10:46:32 +00:00
this.frm.script_manager.trigger("item_code", child.doctype, child.name);
},
refresh_search_box: function() {
var me = this;
// Clear Item Box and remake item list
if (this.search.$input.val()) {
this.search.set_input("");
this.make_item_list();
2013-08-22 11:56:43 +00:00
}
},
update_qty: function(item_code, qty) {
2014-12-19 10:50:32 +00:00
console.log([item_code, qty]);
2013-08-22 11:56:43 +00:00
var me = this;
2014-12-26 07:45:21 +00:00
$.each(this.frm.doc["items"] || [], function(i, d) {
2013-08-22 11:56:43 +00:00
if (d.item_code == item_code) {
2013-11-27 10:46:32 +00:00
if (qty == 0) {
2014-02-14 10:17:51 +00:00
frappe.model.clear_doc(d.doctype, d.name);
2013-11-27 10:46:32 +00:00
me.refresh_grid();
} else {
2014-02-14 10:17:51 +00:00
frappe.model.set_value(d.doctype, d.name, "qty", qty);
}
2013-08-22 11:56:43 +00:00
}
});
this.refresh();
},
refresh: function() {
var me = this;
this.refresh_item_list();
this.party_field.set_input(this.frm.doc[this.party.toLowerCase()]);
this.wrapper.find('input.discount-amount').val(this.frm.doc.discount_amount);
2013-08-22 11:56:43 +00:00
this.barcode.set_input("");
2013-11-27 10:46:32 +00:00
this.show_items_in_item_cart();
this.show_taxes();
this.set_totals();
// if form is local then only run all these functions
if (this.frm.doc.docstatus===0) {
this.call_when_local();
}
this.disable_text_box_and_button();
2014-01-10 09:35:44 +00:00
this.hide_payment_button();
2013-11-27 10:46:32 +00:00
// If quotation to is not Customer then remove party
2014-08-08 10:00:49 +00:00
if (this.frm.doctype == "Quotation" && this.frm.doc.quotation_to!="Customer") {
2014-12-19 10:50:32 +00:00
this.party_field.$input.prop("disabled", true);
2013-11-27 10:46:32 +00:00
}
},
refresh_item_list: function() {
var me = this;
// refresh item list on change of price list
if (this.frm.doc[this.price_list_field] != this.price_list) {
this.price_list = this.frm.doc[this.price_list_field];
this.make_item_list();
}
},
2013-11-27 10:46:32 +00:00
show_items_in_item_cart: function() {
var me = this;
2014-12-19 10:50:32 +00:00
var $items = this.wrapper.find(".items").empty();
2013-08-22 11:56:43 +00:00
2014-12-31 09:33:14 +00:00
$.each(this.frm.doc.items|| [], function(i, d) {
2014-12-19 10:50:32 +00:00
$(frappe.render_template("pos_bill_item", {
item_code: d.item_code,
item_name: d.item_name===d.item_code ? "" : ("<br>" + d.item_name),
qty: d.qty,
actual_qty: d.actual_qty,
rate: format_currency(d.rate, me.frm.doc.currency),
amount: format_currency(d.amount, me.frm.doc.currency)
})).appendTo($items);
2013-08-22 11:56:43 +00:00
});
2014-12-19 10:50:32 +00:00
this.wrapper.find("input.pos-item-qty").on("focus", function() {
$(this).select();
});
2013-11-27 10:46:32 +00:00
},
show_taxes: function() {
var me = this;
2014-12-26 07:45:21 +00:00
var taxes = this.frm.doc["taxes"] || [];
2013-09-30 12:48:19 +00:00
$(this.wrapper).find(".tax-table")
2014-01-10 09:35:44 +00:00
.toggle((taxes && taxes.length) ? true : false)
2013-08-23 07:46:22 +00:00
.find("tbody").empty();
2014-05-05 05:31:32 +00:00
2013-08-23 07:46:22 +00:00
$.each(taxes, function(i, d) {
2014-01-10 09:35:44 +00:00
if (d.tax_amount) {
$(repl('<tr>\
<td>%(description)s %(rate)s</td>\
<td style="text-align: right;">%(tax_amount)s</td>\
<tr>', {
description: d.description,
rate: ((d.charge_type == "Actual") ? '' : ("(" + d.rate + "%)")),
2014-05-05 05:31:32 +00:00
tax_amount: format_currency(flt(d.tax_amount)/flt(me.frm.doc.conversion_rate),
2014-01-10 09:35:44 +00:00
me.frm.doc.currency)
})).appendTo(".tax-table tbody");
}
2013-08-23 07:46:22 +00:00
});
2013-11-27 10:46:32 +00:00
},
set_totals: function() {
var me = this;
2014-05-05 05:31:32 +00:00
this.wrapper.find(".net-total").text(format_currency(this.frm.doc[this.net_total],
2013-11-27 10:46:32 +00:00
me.frm.doc.currency));
2014-05-05 05:31:32 +00:00
this.wrapper.find(".grand-total").text(format_currency(this.frm.doc[this.grand_total],
2013-11-27 10:46:32 +00:00
me.frm.doc.currency));
2014-05-05 05:31:32 +00:00
$(".paid-amount-area").toggle(!!this.frm.doc.paid_amount);
if(this.frm.doc.paid_amount) {
2014-05-05 05:31:32 +00:00
this.wrapper.find(".paid-amount").text(format_currency(this.frm.doc.paid_amount,
me.frm.doc.currency));
}
2013-11-27 10:46:32 +00:00
},
call_when_local: function() {
var me = this;
2013-08-23 07:46:22 +00:00
2013-11-27 10:46:32 +00:00
// append quantity to the respective item after change from input box
2014-12-19 10:50:32 +00:00
$(this.wrapper).find("input.pos-item-qty").on("change", function() {
var item_code = $(this).closest("tr").attr("id");
2013-11-27 10:46:32 +00:00
me.update_qty(item_code, $(this).val());
});
2013-08-22 11:56:43 +00:00
// increase/decrease qty on plus/minus button
2014-12-19 10:50:32 +00:00
$(this.wrapper).find(".pos-qty-btn").on("click", function() {
var $item = $(this).parents(".pos-bill-item:first");
me.increase_decrease_qty($item, $(this).attr("data-action"));
});
2013-11-27 10:46:32 +00:00
// on td click toggle the highlighting of row
$(this.wrapper).find("#cart tbody tr td").on("click", function() {
var row = $(this).closest("tr");
if (row.attr("data-selected") == "false") {
row.attr("class", "warning");
row.attr("data-selected", "true");
}
else {
row.prop("class", null);
row.attr("data-selected", "false");
}
2013-08-23 07:46:22 +00:00
me.refresh_delete_btn();
2013-11-27 10:46:32 +00:00
});
2013-09-02 13:18:39 +00:00
2013-11-27 10:46:32 +00:00
me.refresh_delete_btn();
2014-12-19 10:50:32 +00:00
//me.focus();
},
focus: function() {
2014-12-31 10:04:37 +00:00
if(me.frm.doc[this.party].toLowerCase()) {
this.barcode.$input.focus();
} else {
2014-12-19 10:50:32 +00:00
if(!(this.frm.doctype == "Quotation" && this.frm.doc.quotation_to!="Customer"))
this.party_field.$input.focus();
}
2013-11-27 10:46:32 +00:00
},
2014-12-19 10:50:32 +00:00
increase_decrease_qty: function($item, operation) {
var item_code = $item.attr("data-item-code");
var item_qty = cint($item.find("input.pos-item-qty").val());
if (operation == "increase-qty")
this.update_qty(item_code, item_qty + 1);
2014-08-08 10:00:49 +00:00
else if (operation == "decrease-qty" && item_qty != 0)
this.update_qty(item_code, item_qty - 1);
},
2013-11-27 10:46:32 +00:00
disable_text_box_and_button: function() {
var me = this;
2013-09-02 13:18:39 +00:00
// if form is submitted & cancelled then disable all input box & buttons
$(this.wrapper)
2014-12-19 10:50:32 +00:00
.find(".remove-items, .make-payment, .pos-qty-btn")
.toggle(this.frm.doc.docstatus===0);
2014-05-05 05:31:32 +00:00
$(this.wrapper).find('input, button').prop("disabled", !(this.frm.doc.docstatus===0));
2013-11-27 10:46:32 +00:00
},
2014-01-10 09:35:44 +00:00
hide_payment_button: function() {
$(this.wrapper)
.find(".make-payment")
.toggle(this.frm.doctype == "Sales Invoice" && this.frm.doc.is_pos);
2013-08-23 07:46:22 +00:00
},
refresh_delete_btn: function() {
2014-01-10 09:35:44 +00:00
$(this.wrapper).find(".remove-items").toggle($(".item-cart .warning").length ? true : false);
2013-08-22 11:56:43 +00:00
},
add_item_thru_barcode: function() {
var me = this;
me.barcode_timeout = null;
2014-02-14 10:17:51 +00:00
frappe.call({
method: 'erpnext.accounts.doctype.sales_invoice.pos.get_item_code',
args: {barcode_serial_no: this.barcode.$input.val()},
2013-08-22 11:56:43 +00:00
callback: function(r) {
if (r.message) {
if (r.message[1] == "serial_no")
me.add_to_cart(r.message[0][0].item_code, r.message[0][0].name);
else
me.add_to_cart(r.message[0][0].name);
2013-08-22 11:56:43 +00:00
}
else
2014-04-14 10:55:30 +00:00
msgprint(__("Invalid Barcode"));
2013-08-23 12:52:32 +00:00
me.refresh();
2013-08-22 11:56:43 +00:00
}
});
},
2014-01-10 09:35:44 +00:00
remove_selected_items: function() {
2013-08-22 11:56:43 +00:00
var me = this;
var selected_items = [];
2013-09-30 12:48:19 +00:00
var no_of_items = $(this.wrapper).find("#cart tbody tr").length;
2013-08-22 11:56:43 +00:00
for(var x=0; x<=no_of_items - 1; x++) {
2013-09-30 12:48:19 +00:00
var row = $(this.wrapper).find("#cart tbody tr:eq(" + x + ")");
2013-08-22 11:56:43 +00:00
if(row.attr("data-selected") == "true") {
selected_items.push(row.attr("id"));
}
}
2014-12-26 07:45:21 +00:00
var child = this.frm.doc["items"] || [];
2013-08-22 11:56:43 +00:00
$.each(child, function(i, d) {
for (var i in selected_items) {
if (d.item_code == selected_items[i]) {
2014-02-14 10:17:51 +00:00
frappe.model.clear_doc(d.doctype, d.name);
2013-08-22 11:56:43 +00:00
}
}
});
2014-01-10 09:35:44 +00:00
2013-11-27 10:46:32 +00:00
this.refresh_grid();
},
refresh_grid: function() {
this.frm.dirty();
2014-12-26 07:45:21 +00:00
this.frm.fields_dict["items"].grid.refresh();
2013-09-27 07:02:26 +00:00
this.frm.script_manager.trigger("calculate_taxes_and_totals");
2013-11-27 10:46:32 +00:00
this.refresh();
2013-08-22 11:56:43 +00:00
},
make_payment: function() {
var me = this;
2013-09-30 12:48:19 +00:00
var no_of_items = $(this.wrapper).find("#cart tbody tr").length;
2013-08-22 11:56:43 +00:00
var mode_of_payment = [];
2014-05-05 05:31:32 +00:00
2013-08-22 11:56:43 +00:00
if (no_of_items == 0)
2014-04-14 10:55:30 +00:00
msgprint(__("Payment cannot be made for empty cart"));
2013-08-22 11:56:43 +00:00
else {
2014-02-14 10:17:51 +00:00
frappe.call({
method: 'erpnext.accounts.doctype.sales_invoice.pos.get_mode_of_payment',
2013-08-22 11:56:43 +00:00
callback: function(r) {
2014-05-05 05:31:32 +00:00
if(!r.message) {
msgprint(__("Please add to Modes of Payment from Setup."))
2014-05-05 05:31:32 +00:00
return;
}
2013-08-22 11:56:43 +00:00
for (x=0; x<=r.message.length - 1; x++) {
mode_of_payment.push(r.message[x].name);
}
// show payment wizard
2014-02-14 10:17:51 +00:00
var dialog = new frappe.ui.Dialog({
2013-08-22 11:56:43 +00:00
width: 400,
2014-05-05 05:31:32 +00:00
title: 'Payment',
2013-08-22 11:56:43 +00:00
fields: [
{fieldtype:'Data', fieldname:'total_amount', label:'Total Amount', read_only:1},
2014-05-05 05:31:32 +00:00
{fieldtype:'Select', fieldname:'mode_of_payment', label:'Mode of Payment',
2013-08-22 11:56:43 +00:00
options:mode_of_payment.join('\n'), reqd: 1},
{fieldtype:'Button', fieldname:'pay', label:'Pay'}
]
});
dialog.set_values({
"total_amount": $(".grand-total").text()
});
dialog.show();
dialog.get_input("total_amount").prop("disabled", true);
2014-05-05 05:31:32 +00:00
2013-08-22 11:56:43 +00:00
dialog.fields_dict.pay.input.onclick = function() {
2013-09-27 07:02:26 +00:00
me.frm.set_value("mode_of_payment", dialog.get_values().mode_of_payment);
me.frm.set_value("paid_amount", dialog.get_values().total_amount);
me.frm.cscript.mode_of_payment(me.frm.doc);
me.frm.save();
2013-08-22 11:56:43 +00:00
dialog.hide();
me.refresh();
};
}
});
2013-08-22 11:56:43 +00:00
}
},
2014-04-14 11:44:23 +00:00
});