You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
230 lines
8.8 KiB
230 lines
8.8 KiB
odoo.define('pos_barcode_tare.screens', function (require) {
|
|
|
|
"use strict";
|
|
var chrome = require('point_of_sale.chrome');
|
|
var core = require('web.core');
|
|
var devices = require('point_of_sale.devices');
|
|
var gui = require('point_of_sale.gui');
|
|
var models = require('point_of_sale.models');
|
|
var screens = require('point_of_sale.screens');
|
|
var QWeb = core.qweb;
|
|
var _t = core._t;
|
|
|
|
// This configures read action for tare barcode. A tare barcode contains a
|
|
// fake product ID and the weight to be subtracted from the product in the
|
|
// latest order line.
|
|
screens.ScreenWidget.include(
|
|
{
|
|
barcode_weight_action: function (code) {
|
|
var self = this;
|
|
var order = this.pos.get_order();
|
|
// Computes the paid weight
|
|
var last_order_line = order.get_last_orderline();
|
|
var total_weight = last_order_line.get_quantity();
|
|
var tare = code.value;
|
|
var paid_weight = total_weight - tare;
|
|
// Throws a warning popup if the price is negative
|
|
if (paid_weight <= 0) {
|
|
self.gui.show_popup('confirm',
|
|
{'title': _t('Negative weight'),
|
|
'body': _t('The calculated weight is negative. ' +
|
|
'Did you scan the correct tare label?'),
|
|
confirm: function () {
|
|
// Operator can choose to ignore the warning.
|
|
last_order_line.set_quantity(paid_weight);
|
|
}});
|
|
} else {
|
|
last_order_line.set_quantity(paid_weight);
|
|
}
|
|
},
|
|
|
|
// Setup the callback action for the "weight" barcodes.
|
|
show: function () {
|
|
var self = this;
|
|
this._super();
|
|
this.pos.barcode_reader.set_action_callback(
|
|
'weight',
|
|
_.bind(self.barcode_weight_action, self));
|
|
},
|
|
});
|
|
|
|
// This create a new button on top of action widget. This button links to
|
|
// the barcode label printing screen defined below.
|
|
var TareScreenButton = screens.ActionButtonWidget.extend({
|
|
template: 'TareScreenButton',
|
|
|
|
button_click: function () {
|
|
var self = this;
|
|
self.gui.show_screen('tare');
|
|
},
|
|
});
|
|
|
|
screens.define_action_button({
|
|
'name': 'tareScreenButton',
|
|
'widget': TareScreenButton,
|
|
});
|
|
|
|
// This is a new screen that reads weight from the electronic scale and
|
|
// create a barcode label encoding the weight. The screen shows a preview
|
|
// of the label. The user is expected to check if the preview matches what's
|
|
// measured on the scale. The barcode image is generated by the report
|
|
// module.
|
|
var TareScreenWidget = screens.ScreenWidget.extend({
|
|
template: 'TareScreenWidget',
|
|
next_screen: 'products',
|
|
previous_screen: 'products',
|
|
default_tare_value_kg: 0.0,
|
|
weight_barcode_prefix: null,
|
|
|
|
show: function () {
|
|
this._super();
|
|
var self = this;
|
|
// Fetch the barcode prefix from POS barcode parser rules.
|
|
this.weight_barcode_prefix = this.get_barcode_prefix(
|
|
this.pos.config.iface_tare_barcode_sequence_id);
|
|
// Setup the proxy
|
|
var queue = this.pos.proxy_queue;
|
|
// The pooling of the scale starts here.
|
|
queue.schedule(function () {
|
|
return self.pos.proxy.scale_read().then(function (weight) {
|
|
self.set_weight(weight.weight);
|
|
});
|
|
}, {duration:150, repeat: true});
|
|
// Shows a barcode whose weight might be zero, but this is preferred
|
|
// for UI/UX reasons.
|
|
this.render_receipt();
|
|
this.lock_screen(true);
|
|
},
|
|
get_barcode_prefix: function (barcode_sequence_id) {
|
|
var self = this;
|
|
var barcode_pattern = self.get_barcode_pattern(barcode_sequence_id);
|
|
return barcode_pattern.substr(0, 2);
|
|
},
|
|
get_barcode_pattern: function (barcode_sequence_id) {
|
|
var self = this;
|
|
var rules = self.get_barcode_rules();
|
|
var rule = rules.filter(
|
|
function (r) {
|
|
return r.sequence === barcode_sequence_id;
|
|
})[0];
|
|
return rule.pattern;
|
|
},
|
|
get_barcode_rules: function () {
|
|
var self = this;
|
|
return self.pos.barcode_reader.barcode_parser.nomenclature.rules;
|
|
},
|
|
set_weight: function (weight) {
|
|
if (weight > 0) {
|
|
this.weight = weight;
|
|
this.render_receipt();
|
|
this.lock_screen(false);
|
|
}
|
|
},
|
|
get_weight: function () {
|
|
if (typeof this.weight === 'undefined') {
|
|
return this.default_tare_value_kg;
|
|
}
|
|
return this.weight;
|
|
},
|
|
ean13_checksum: function (s) {
|
|
var result = 0;
|
|
for (var counter = s.length-1; counter >=0; counter--) {
|
|
var counterCheckSum = counter % 2;
|
|
counterCheckSum *= 2;
|
|
counterCheckSum += 1;
|
|
result += parseInt(s.charAt(counter), 10) * counterCheckSum;
|
|
}
|
|
var checksum = 10;
|
|
checksum -= result % 10;
|
|
return checksum % 10;
|
|
},
|
|
barcode_data: function (weight) {
|
|
// We use EAN13 barcode, it looks like 21 00000 12345 x. First there
|
|
// is the prefix, here 21, that is used to decide which type of
|
|
// barcode we're dealing with. A weight barcode has then two groups
|
|
// of five digits. The first group encodes the product id. Here the
|
|
// product id is 00000. The second group encodes the weight in
|
|
// grams. Here the weight is 12.345kg. The last digit of the barcode
|
|
// is a checksum, here symbolized by x.
|
|
var padding_size = 5;
|
|
var void_product_id = '0'.repeat(padding_size);
|
|
var weight_in_gram = weight * 10e2;
|
|
// Weight has to be padded with zeros.
|
|
var weight_with_padding = '0'.repeat(padding_size) + weight_in_gram;
|
|
var padded_weight = weight_with_padding.substr(
|
|
weight_with_padding.length - padding_size);
|
|
// Builds the barcode data (ie. all but the checksum).
|
|
var barcode_data = this.weight_barcode_prefix
|
|
.concat(void_product_id, padded_weight);
|
|
// Compute checksum and concat with barcode data to get the actual
|
|
// barcode.
|
|
var checksum = this.ean13_checksum(barcode_data);
|
|
var barcode = barcode_data.concat(checksum);
|
|
|
|
return barcode;
|
|
},
|
|
get_barcode_data: function () {
|
|
return this.barcode_data(this.get_weight());
|
|
},
|
|
should_auto_print: function () {
|
|
return this.pos.config.iface_print_auto &&
|
|
!this.pos.get_order()._printed;
|
|
},
|
|
should_close_immediately: function () {
|
|
return this.pos.config.iface_print_via_proxy &&
|
|
this.pos.config.iface_print_skip_screen;
|
|
},
|
|
lock_screen: function (locked) {
|
|
this._locked = locked;
|
|
if (locked) {
|
|
this.$('.print-label').addClass('disabled');
|
|
} else {
|
|
this.$('.print-label').removeClass('disabled');
|
|
}
|
|
},
|
|
print_web: function () {
|
|
window.print();
|
|
this.pos.get_order()._printed = true;
|
|
},
|
|
print: function () {
|
|
// See comment in print function of ReceiptScreenWidget
|
|
var self = this;
|
|
this.lock_screen(true);
|
|
|
|
setTimeout(function () {
|
|
self.lock_screen(false);
|
|
}, 1000);
|
|
|
|
this.print_web();
|
|
this.click_back();
|
|
},
|
|
click_back: function () {
|
|
this.close();
|
|
this.gui.show_screen(this.previous_screen);
|
|
},
|
|
renderElement: function () {
|
|
var self = this;
|
|
this._super();
|
|
this.$('.back').click(function () {
|
|
self.click_back();
|
|
});
|
|
this.$('.print-label').click(function () {
|
|
if (!self._locked) {
|
|
self.print();
|
|
}
|
|
});
|
|
},
|
|
render_receipt: function () {
|
|
this.$('.pos-tare-label-container').html(
|
|
QWeb.render('PosTareLabel', {widget:this}));
|
|
},
|
|
close: function () {
|
|
this._super();
|
|
delete this.weight;
|
|
this.pos.proxy_queue.clear();
|
|
},
|
|
});
|
|
|
|
gui.define_screen({name:'tare', widget: TareScreenWidget});
|
|
|
|
});
|