Odoo18-Base/addons/mrp/wizard/mrp_consumption_warning.py

103 lines
5.5 KiB
Python
Raw Permalink Normal View History

2025-03-10 11:12:23 +07:00
# -*- 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)