103 lines
5.5 KiB
Python
103 lines
5.5 KiB
Python
|
# -*- 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)
|