flectra.define('barcodes.FormView', function (require) { "use strict"; var BarcodeEvents = require('barcodes.BarcodeEvents'); // handle to trigger barcode on bus var concurrency = require('web.concurrency'); var core = require('web.core'); var Dialog = require('web.Dialog'); var FormController = require('web.FormController'); var FormRenderer = require('web.FormRenderer'); var _t = core._t; FormController.include({ custom_events: _.extend({}, FormController.prototype.custom_events, { activeBarcode: '_barcodeActivated', }), /** * add default barcode commands for from view * * @override */ init: function () { this._super.apply(this, arguments); this.activeBarcode = { form_view: { commands: { 'O-CMD.EDIT': this._barcodeEdit.bind(this), 'O-CMD.DISCARD': this._barcodeDiscard.bind(this), 'O-CMD.SAVE': this._barcodeSave.bind(this), 'O-CMD.PREV': this._barcodePagerPrevious.bind(this), 'O-CMD.NEXT': this._barcodePagerNext.bind(this), 'O-CMD.PAGER-FIRST': this._barcodePagerFirst.bind(this), 'O-CMD.PAGER-LAST': this._barcodePagerLast.bind(this), }, }, }; this.barcodeMutex = new concurrency.Mutex(); this._barcodeStartListening(); }, /** * @override */ destroy: function () { this._barcodeStopListening(); this._super(); }, //-------------------------------------------------------------------------- // Private //-------------------------------------------------------------------------- /** * @private * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {Object} activeBarcode: options sent by the field who use barcode features * @returns {Deferred} */ _barcodeAddX2MQuantity: function (barcode, activeBarcode) { if (this.mode === 'readonly') { this.do_warn(_t('Error : Document not editable'), _t('To modify this document, please first start edition.')); return new $.Deferred().reject(); } var record = this.model.get(this.handle); var candidate = this._getBarCodeRecord(record, barcode, activeBarcode); if (candidate) { return this._barcodeSelectedCandidate(candidate, record, barcode, activeBarcode); } else { return this._barcodeWithoutCandidate(record, barcode, activeBarcode); } }, /** * @private */ _barcodeDiscard: function () { return this.discardChanges(); }, /** * @private */ _barcodeEdit: function () { return this._setMode('edit'); }, /** * @private */ _barcodePagerFirst: function () { var self = this; return this.mutex.exec(function () {}).then(function () { self.pager.updateState({ current_min: 1, }, {notifyChange: true}); }); }, /** * @private */ _barcodePagerLast: function () { var self = this; return this.mutex.exec(function () {}).then(function () { var state = self.model.get(self.handle, {raw: true}); self.pager.updateState({ current_min: state.count, }, {notifyChange: true}); }); }, /** * @private */ _barcodePagerNext: function () { return this.mutex.exec(function () {}).then(this.pager.next.bind(this.pager)); }, /** * @private */ _barcodePagerPrevious: function () { return this.mutex.exec(function () {}).then(this.pager.previous.bind(this.pager)); }, /** * Returns true iff the given barcode matches the given record (candidate). * * @private * @param {Object} candidate: record in the x2m * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {Object} activeBarcode: options sent by the field who use barcode features * @returns {boolean} */ _barcodeRecordFilter: function (candidate, barcode, activeBarcode) { return candidate.data.product_barcode === barcode; }, /** * @private */ _barcodeSave: function () { return this.saveRecord(); }, /** * @private * @param {Object} candidate: record in the x2m * @param {Object} current record * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {Object} activeBarcode: options sent by the field who use barcode features * @returns {Deferred} */ _barcodeSelectedCandidate: function (candidate, record, barcode, activeBarcode, quantity) { var changes = {}; var candidateChanges = {}; candidateChanges[activeBarcode.quantity] = quantity ? quantity : candidate.data[activeBarcode.quantity] + 1; changes[activeBarcode.fieldName] = { operation: 'UPDATE', id: candidate.id, data: candidateChanges, }; return this.model.notifyChanges(this.handle, changes, {notifyChange: activeBarcode.notifyChange}); }, /** * @private */ _barcodeStartListening: function () { core.bus.on('barcode_scanned', this, this._barcodeScanned); core.bus.on('keypress', this, this._quantityListener); }, /** * @private */ _barcodeStopListening: function () { core.bus.off('barcode_scanned', this, this._barcodeScanned); core.bus.off('keypress', this, this._quantityListener); }, /** * @private * @param {Object} current record * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {Object} activeBarcode: options sent by the field who use barcode features * @returns {Deferred} */ _barcodeWithoutCandidate: function (record, barcode, activeBarcode) { var changes = {}; changes[activeBarcode.name] = barcode; return this.model.notifyChanges(record.id, changes); }, /** * @private * @param {Object} current record * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {Object} activeBarcode: options sent by the field who use barcode features * @returns {Object|undefined} */ _getBarCodeRecord: function (record, barcode, activeBarcode) { var self = this; if (!activeBarcode.fieldName || !record.data[activeBarcode.fieldName]) { return; } return _.find(record.data[activeBarcode.fieldName].data, function (record) { return self._barcodeRecordFilter(record, barcode, activeBarcode); }); }, //-------------------------------------------------------------------------- // Handlers //-------------------------------------------------------------------------- /** * The barcode is activate when at least one widget trigger_up 'activeBarcode' event * with the widget option * * @param {FlectraEvent} event * @param {string} event.data.name: the current field name * @param {string} [event.data.fieldName] optional for x2many sub field * @param {boolean} [event.data.notifyChange] optional for x2many sub field * do not trigger on change server side if a candidate has been found * @param {string} [event.data.quantity] optional field to increase quantity * @param {Object} [event.data.commands] optional added methods * can use comand with specific barcode (with ReservedBarcodePrefixes) * or change 'barcode' for all other received barcodes * (e.g.: 'O-CMD.MAIN-MENU': function ..., barcode: function () {...}) */ _barcodeActivated: function (event) { event.stopPropagation(); var name = event.data.name; this.activeBarcode[name] = { name: name, handle: this.handle, target: event.target, widget: event.target.attrs && event.target.attrs.widget, setQuantityWithKeypress: !! event.data.setQuantityWithKeypress, fieldName: event.data.fieldName, notifyChange: (event.data.notifyChange !== undefined) ? event.data.notifyChange : true, quantity: event.data.quantity, commands: event.data.commands || {}, candidate: this.activeBarcode[name] && this.activeBarcode[name].handle === this.handle ? this.activeBarcode[name].candidate : null, }; // we want to disable autofocus when activating the barcode to avoid // putting the scanned value in the focused field this.disableAutofocus = true; }, /** * @private * @param {string|function} method defined by the commands options * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {Object} activeBarcode: options sent by the field who use barcode features * @returns {Deferred} */ _barcodeActiveScanned: function (method, barcode, activeBarcode) { var self = this; var methodDef; var def = new $.Deferred(); if (typeof method === 'string') { methodDef = this[method](barcode, activeBarcode); } else { methodDef = method.call(this, barcode, activeBarcode); } methodDef .done(function () { var record = self.model.get(self.handle); var candidate = self._getBarCodeRecord(record, barcode, activeBarcode); activeBarcode.candidate = candidate; }) .always(function () { def.resolve(); }); return def; }, /** * Method called when a user scan a barcode, call each method in function of the * widget options then update the renderer * * @private * @param {string} barcode sent by the scanner (string generate from keypress series) * @param {DOM Object} target * @returns {Deferred} */ _barcodeScanned: function (barcode, target) { var self = this; return this.barcodeMutex.exec(function () { var prefixed = _.any(BarcodeEvents.ReservedBarcodePrefixes, function (reserved) {return barcode.indexOf(reserved) === 0;}); var hasCommand = false; var defs = []; if (! $.contains(target, self.el)) { return; } for (var k in self.activeBarcode) { var activeBarcode = self.activeBarcode[k]; // Handle the case where there are several barcode widgets on the same page. Since the // event is global on the page, all barcode widgets will be triggered. However, we only // want to keep the event on the target widget. if (! $.contains(target, self.el)) { continue; } var methods = self.activeBarcode[k].commands; var method = prefixed ? methods[barcode] : methods.barcode; if (method) { if (prefixed) { hasCommand = true; } defs.push(self._barcodeActiveScanned(method, barcode, activeBarcode)); } } if (prefixed && !hasCommand) { self.do_warn(_t('Error : Barcode command is undefined'), barcode); } return self.alive($.when.apply($, defs)).then(function () { if (!prefixed) { // remember the barcode scanned for the quantity listener self.current_barcode = barcode; // redraw the view if we scanned a real barcode (required if // we manually apply the change in JS, e.g. incrementing the // quantity) self.update({}, {reload: false}); } }); }); }, /** * @private * @param {KeyEvent} event */ _quantityListener: function (event) { var character = String.fromCharCode(event.which); if (! $.contains(event.target, this.el)) { return; } // only catch the event if we're not focused in // another field and it's a number if (!$(event.target).is('body, .modal') || !/[0-9]/.test(character)) { return; } var barcodeInfos = _.filter(this.activeBarcode, 'setQuantityWithKeypress'); if (!barcodeInfos.length) { return; } if (!_.compact(_.pluck(barcodeInfos, 'candidate')).length) { return this.do_warn(_t('Error : No last scanned barcode'), _t('To set the quantity please scan a barcode first.')); } for (var k in this.activeBarcode) { if (this.activeBarcode[k].candidate) { this._quantityOpenDialog(character, this.activeBarcode[k]); } } }, /** * @private * @param {string} character * @param {Object} activeBarcode: options sent by the field who use barcode features */ _quantityOpenDialog: function (character, activeBarcode) { var self = this; var $content = $('