# -*- coding: utf-8 -*- # Part of Odoo. See LICENSE file for full copyright and licensing details. from odoo import _, fields, models, api from odoo.exceptions import UserError from odoo.tools import float_compare, float_is_zero class MrpConsumptionWarning(models.TransientModel): _name = 'mrp.consumption.warning' _description = "Wizard in case of consumption in warning/strict and more component has been used for a MO (related to the bom)" mrp_production_ids = fields.Many2many('mrp.production') mrp_production_count = fields.Integer(compute="_compute_mrp_production_count") consumption = fields.Selection([ ('flexible', 'Allowed'), ('warning', 'Allowed with warning'), ('strict', 'Blocked')], compute="_compute_consumption") mrp_consumption_warning_line_ids = fields.One2many('mrp.consumption.warning.line', 'mrp_consumption_warning_id') @api.depends("mrp_production_ids") def _compute_mrp_production_count(self): for wizard in self: wizard.mrp_production_count = len(wizard.mrp_production_ids) @api.depends("mrp_consumption_warning_line_ids.consumption") def _compute_consumption(self): for wizard in self: consumption_map = set(wizard.mrp_consumption_warning_line_ids.mapped("consumption")) wizard.consumption = "strict" in consumption_map and "strict" or "warning" in consumption_map and "warning" or "flexible" def action_confirm(self): ctx = dict(self.env.context) ctx.pop('default_mrp_production_ids', None) return self.mrp_production_ids.with_context(ctx, skip_consumption=True).button_mark_done() def action_set_qty(self): missing_move_vals = [] problem_tracked_products = self.env['product.product'] for production in self.mrp_production_ids: for line in self.mrp_consumption_warning_line_ids: if line.mrp_production_id != production: continue for move in production.move_raw_ids: if line.product_id != move.product_id: continue qty_expected = line.product_uom_id._compute_quantity(line.product_expected_qty_uom, move.product_uom) qty_compare_result = float_compare(qty_expected, move.quantity_done, precision_rounding=move.product_uom.rounding) if qty_compare_result != 0: if (move.has_tracking in ('lot', 'serial') and not production.use_auto_consume_components_lots and qty_compare_result > 0): problem_tracked_products |= line.product_id break move.quantity_done = qty_expected # in case multiple lines with same product => set others to 0 since we have no way to know how to distribute the qty done line.product_expected_qty_uom = 0 # move was deleted before confirming MO or force deleted somehow if not float_is_zero(line.product_expected_qty_uom, precision_rounding=line.product_uom_id.rounding): if line.product_id.tracking in ('lot', 'serial') and not line.mrp_production_id.use_auto_consume_components_lots: problem_tracked_products |= line.product_id continue missing_move_vals.append({ 'product_id': line.product_id.id, 'product_uom': line.product_uom_id.id, 'product_uom_qty': line.product_expected_qty_uom, 'quantity_done': line.product_expected_qty_uom, 'raw_material_production_id': line.mrp_production_id.id, 'additional': True, }) if problem_tracked_products: raise UserError( _("Values cannot be set and validated because a Lot/Serial Number needs to be specified for a tracked product that is having its consumed amount increased:\n- ") + "\n- ".join(problem_tracked_products.mapped('name')) ) if missing_move_vals: self.env['stock.move'].create(missing_move_vals) return self.action_confirm() def action_cancel(self): if self.env.context.get('from_workorder') and len(self.mrp_production_ids) == 1: return { 'type': 'ir.actions.act_window', 'res_model': 'mrp.production', 'views': [[self.env.ref('mrp.mrp_production_form_view').id, 'form']], 'res_id': self.mrp_production_ids.id, 'target': 'main', } class MrpConsumptionWarningLine(models.TransientModel): _name = 'mrp.consumption.warning.line' _description = "Line of issue consumption" mrp_consumption_warning_id = fields.Many2one('mrp.consumption.warning', "Parent Wizard", readonly=True, required=True, ondelete="cascade") mrp_production_id = fields.Many2one('mrp.production', "Manufacturing Order", readonly=True, required=True, ondelete="cascade") consumption = fields.Selection(related="mrp_production_id.consumption") product_id = fields.Many2one('product.product', "Product", readonly=True, required=True) product_uom_id = fields.Many2one('uom.uom', "Unit of Measure", related="product_id.uom_id", readonly=True) product_consumed_qty_uom = fields.Float("Consumed", readonly=True) product_expected_qty_uom = fields.Float("To Consume", readonly=True)