Search code examples
javascriptodooodoo-16

Why a JS method in Odoo 16 works only sometimes with similar environments?


I'm working with Odoo 16. I've installed the Enterprise Barcode app, and I've made a generic development over it.

The Barcode app has a menu which shows pickings in a kanban view. What I'm doing with my development is to automatically fill in a picking field each time an user belonging to a specific group clicks on its kanban card to open it.

Example: the barcode app shows three pickings (PICK001, PICK002, PICK003) in a kanban view. I'm the user Administrator and I click on the kanban card of PICK003 and this way I'm redirected to other view which shows the stock move lines of the PICK003. At the same time, my development sets the picking field preparer_id (a Many2one pointing to res.users) to Administrator.

Now, the code of my development:

/** @odoo-module **/
import { StockBarcodeKanbanController } from '@stock_barcode/kanban/stock_barcode_kanban_controller';
import { patch } from 'web.utils';
import { useService } from "@web/core/utils/hooks";
import { session } from "@web/session";

const originalOpenRecord = StockBarcodeKanbanController.prototype.openRecord;

patch(StockBarcodeKanbanController.prototype, 'StockBarcodeKanbanControllerIchExt', {
    setup() {
        this._super.apply(this, arguments);
        this.rpc = useService("rpc");
    },

    async openRecord(record) {
        // Check if preparer_id is empty
        if (!record.data.preparer_id) {
            // Check if current user is preparer (belong to preparers group)
            const isPreparer = await this.rpc("/web/dataset/call_kw", {
                model: 'res.users',
                method: 'has_group',
                args: ['ich_stock_ext.group_stock_preparer'],
                kwargs: {}
            });
            if(isPreparer) {
                // Call RPC to update preparer_id value with the current user
                await this.rpc('/web/dataset/call_kw/stock.picking/write', {
                    model: 'stock.picking',
                    method: 'write',
                    args: [[record.resId], { 'preparer_id': session.uid }],
                    kwargs: {}
                });
                // Refresh view
                this.model.load({reload: true});
            }
        }
        // Call "super" to execute whatever the original method did
        return originalOpenRecord.apply(this, arguments);
    },

});

This code seems to work perfect. However, sometimes, especially when I log in through the Odoo mobile app, if I click on PICK003, then on PICK002, and afterwards on PICK003 again, the field preparer_id remains empty, instead of showing the user again.

I've checked the field during every step. I have another development which empties the field when going back to the kanban view. This other development always works fine, so before clicking on a kanban card, the preparer_id is always empty.

What could be happening? How can I change the code to ensure that the preparer_id is updated in every situation?


Solution

  • Several days getting crazy with this and the answer is a joke.

    This code seems to work perfect. However, sometimes, especially when I log in through the Odoo mobile app, if I click on PICK003, then on PICK002, and afterwards on PICK003 again, the field preparer_id remains empty, instead of showing the user again.

    Espcially when I use the mobile app... because the mobile screen is smaller than the computer one... and because of this the kanban cards are smaller too... and the human fingers are bigger than the mouse pointer... So sometimes, users clicked on the name of the picking instead of the kanban card, and the Enterprise Barcode app has a different behavior for them, however both take you to the same screen, that's the reason why nobody noticed that.

    After clicking probably more than one thousand times on the kanban card these days, I found out the fact that clicking on the name calls a Python method named action_open_picking_client_action which opens the stock move lines screen too, but without going through JS, therefore, my JS code was ignored. But if I click on the kanban card, not on the name, the stock move lines screen is opened through JS code, executing my JS code.

    So I only had to do this:

    def action_open_picking_client_action(self):
        if self.env.user.has_group('ich_stock_ext.group_stock_preparer'):
            self.filtered(lambda p: not p.preparer_id).write({
                'preparer_id': self.env.user.id,
            })
        return super(StockPicking, self).action_open_picking_client_action()
    

    And that's it. There was no asynchronicity problem, which I blamed for days.

    Programmers are undervalued...