odoo/addons/mrp/models/stock_move.py

412 lines
21 KiB
Python

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo import api, exceptions, fields, models, _
from odoo.exceptions import UserError
from odoo.tools import float_compare, float_round
from odoo.addons import decimal_precision as dp
class StockMoveLots(models.Model):
_name = 'stock.move.lots'
_description = "Quantities to Process by lots"
move_id = fields.Many2one('stock.move', 'Move')
workorder_id = fields.Many2one('mrp.workorder', 'Work Order')
production_id = fields.Many2one('mrp.production', 'Production Order')
lot_id = fields.Many2one(
'stock.production.lot', 'Lot',
domain="[('product_id', '=', product_id)]")
lot_produced_id = fields.Many2one('stock.production.lot', 'Finished Lot')
lot_produced_qty = fields.Float(
'Quantity Finished Product', digits=dp.get_precision('Product Unit of Measure'),
help="Informative, not used in matching")
quantity = fields.Float('To Do', default=1.0, digits=dp.get_precision('Product Unit of Measure'))
quantity_done = fields.Float('Done', digits=dp.get_precision('Product Unit of Measure'))
product_id = fields.Many2one(
'product.product', 'Product',
readonly=True, related="move_id.product_id", store=True)
done_wo = fields.Boolean('Done for Work Order', default=True, help="Technical Field which is False when temporarily filled in in work order") # TDE FIXME: naming
done_move = fields.Boolean('Move Done', related='move_id.is_done', store=True) # TDE FIXME: naming
plus_visible = fields.Boolean("Plus Visible", compute='_compute_plus')
@api.one
@api.constrains('lot_id', 'quantity_done')
def _check_lot_id(self):
if self.move_id.product_id.tracking == 'serial':
lots = set([])
for move_lot in self.move_id.active_move_lot_ids.filtered(lambda r: not r.lot_produced_id and r.lot_id):
if move_lot.lot_id in lots:
raise exceptions.UserError(_('You cannot use the same serial number in two different lines.'))
if float_compare(move_lot.quantity_done, 1.0, precision_rounding=move_lot.product_id.uom_id.rounding) == 1:
raise exceptions.UserError(_('You can only produce 1.0 %s for products with unique serial number.') % move_lot.product_id.uom_id.name)
lots.add(move_lot.lot_id)
def _compute_plus(self):
for movelot in self:
if movelot.move_id.product_id.tracking == 'serial':
movelot.plus_visible = (movelot.quantity_done <= 0.0)
else:
movelot.plus_visible = (movelot.quantity == 0.0) or (movelot.quantity_done < movelot.quantity)
@api.multi
def do_plus(self):
self.ensure_one()
self.quantity_done = self.quantity_done + 1
return self.move_id.split_move_lot()
@api.multi
def do_minus(self):
self.ensure_one()
self.quantity_done = self.quantity_done - 1
return self.move_id.split_move_lot()
@api.multi
def write(self, vals):
if 'lot_id' in vals:
for movelot in self:
movelot.move_id.production_id.move_raw_ids.mapped('move_lot_ids')\
.filtered(lambda r: r.done_wo and not r.done_move and r.lot_produced_id == movelot.lot_id)\
.write({'lot_produced_id': vals['lot_id']})
return super(StockMoveLots, self).write(vals)
class StockMove(models.Model):
_inherit = 'stock.move'
production_id = fields.Many2one(
'mrp.production', 'Production Order for finished products')
raw_material_production_id = fields.Many2one(
'mrp.production', 'Production Order for raw materials')
unbuild_id = fields.Many2one(
'mrp.unbuild', 'Unbuild Order')
consume_unbuild_id = fields.Many2one(
'mrp.unbuild', 'Consume Unbuild Order')
operation_id = fields.Many2one(
'mrp.routing.workcenter', 'Operation To Consume') # TDE FIXME: naming
workorder_id = fields.Many2one(
'mrp.workorder', 'Work Order To Consume')
has_tracking = fields.Selection(related='product_id.tracking', string='Product with Tracking') # TDE FIXME: naming ...
# Quantities to process, in normalized UoMs
quantity_available = fields.Float(
'Quantity Available', compute="_qty_available",
digits=dp.get_precision('Product Unit of Measure'))
quantity_done_store = fields.Float('Quantity done store', digits=0)
quantity_done = fields.Float(
'Quantity', compute='_qty_done_compute', inverse='_qty_done_set',
digits=dp.get_precision('Product Unit of Measure'))
move_lot_ids = fields.One2many('stock.move.lots', 'move_id', string='Lots')
active_move_lot_ids = fields.One2many('stock.move.lots', 'move_id', domain=[('done_wo', '=', True)], string='Lots')
bom_line_id = fields.Many2one('mrp.bom.line', 'BoM Line')
unit_factor = fields.Float('Unit Factor')
is_done = fields.Boolean(
'Done', compute='_compute_is_done',
store=True,
help='Technical Field to order moves') # TDE: what ?
@api.depends('state', 'product_uom_qty', 'reserved_availability')
def _qty_available(self):
for move in self:
# For consumables, state is available so availability = qty to do
if move.state == 'assigned':
move.quantity_available = move.product_uom_qty
elif move.product_id.uom_id and move.product_uom:
move.quantity_available = move.product_id.uom_id._compute_quantity(move.reserved_availability, move.product_uom)
@api.multi
@api.depends('move_lot_ids', 'move_lot_ids.quantity_done', 'quantity_done_store')
def _qty_done_compute(self):
for move in self:
if move.has_tracking != 'none':
move.quantity_done = sum(move.move_lot_ids.filtered(lambda x: x.done_wo).mapped('quantity_done')) #TODO: change with active_move_lot_ids?
else:
move.quantity_done = move.quantity_done_store
@api.multi
def _qty_done_set(self):
for move in self:
if move.has_tracking == 'none':
move.quantity_done_store = move.quantity_done
@api.multi
@api.depends('state')
def _compute_is_done(self):
for move in self:
move.is_done = (move.state in ('done', 'cancel'))
@api.multi
def action_assign(self, no_prepare=False):
res = super(StockMove, self).action_assign(no_prepare=no_prepare)
self.check_move_lots()
return res
def _propagate_cancel(self):
self.ensure_one()
if not self.move_dest_id.raw_material_production_id:
super(StockMove, self)._propagate_cancel()
elif self.move_dest_id.state == 'waiting':
# If waiting, the chain will be broken and we are not sure if we can still wait for it (=> could take from stock instead)
self.move_dest_id.write({'state': 'confirmed'})
@api.multi
def action_cancel(self):
if any(move.quantity_done for move in self):
raise exceptions.UserError(_('You cannot cancel a move move having already consumed material'))
return super(StockMove, self).action_cancel()
@api.multi
def check_move_lots(self):
moves_todo = self.filtered(lambda x: x.raw_material_production_id and x.state not in ('done', 'cancel'))
return moves_todo.create_lots()
@api.multi
def create_lots(self):
lots = self.env['stock.move.lots']
for move in self:
unlink_move_lots = move.move_lot_ids.filtered(lambda x : (x.quantity_done == 0) and x.done_wo)
unlink_move_lots.sudo().unlink()
group_new_quant = {}
old_move_lot = {}
for movelot in move.move_lot_ids:
key = (movelot.lot_id.id or False)
old_move_lot.setdefault(key, []).append(movelot)
for quant in move.reserved_quant_ids:
key = (quant.lot_id.id or False)
quantity = move.product_id.uom_id._compute_quantity(quant.qty, move.product_uom)
if group_new_quant.get(key):
group_new_quant[key] += quantity
else:
group_new_quant[key] = quantity
for key in group_new_quant:
quantity = group_new_quant[key]
if old_move_lot.get(key):
if old_move_lot[key][0].quantity == quantity:
continue
else:
old_move_lot[key][0].quantity = quantity
else:
vals = {
'move_id': move.id,
'product_id': move.product_id.id,
'workorder_id': move.workorder_id.id,
'production_id': move.raw_material_production_id.id,
'quantity': quantity,
'lot_id': key,
}
lots.create(vals)
return True
@api.multi
def _create_extra_move(self):
''' Creates an extra move if necessary depending on extra quantities than foreseen or extra moves'''
self.ensure_one()
quantity_to_split = 0
uom_qty_to_split = 0
extra_move = self.env['stock.move']
rounding = self.product_uom.rounding
link_procurement = False
# If more produced than the procurement linked, you should create an extra move
if self.procurement_id and self.production_id and float_compare(self.production_id.qty_produced, self.procurement_id.product_qty, precision_rounding=rounding) > 0:
done_moves_total = sum(self.production_id.move_finished_ids.filtered(lambda x: x.product_id == self.product_id and x.state == 'done').mapped('product_uom_qty'))
# If you depassed the quantity before, you don't need to split anymore, but adapt the quantities
if float_compare(done_moves_total, self.procurement_id.product_qty, precision_rounding=rounding) >= 0:
quantity_to_split = 0
if float_compare(self.product_uom_qty, self.quantity_done, precision_rounding=rounding) < 0:
self.product_uom_qty = self.quantity_done #TODO: could change qty on move_dest_id also (in case of 2-step in/out)
else:
quantity_to_split = done_moves_total + self.quantity_done - self.procurement_id.product_qty
uom_qty_to_split = self.product_uom_qty - (self.quantity_done - quantity_to_split)#self.product_uom_qty - (self.procurement_id.product_qty + done_moves_total)
if float_compare(uom_qty_to_split, quantity_to_split, precision_rounding=rounding) < 0:
uom_qty_to_split = quantity_to_split
self.product_uom_qty = self.quantity_done - quantity_to_split
# You split also simply when the quantity done is bigger than foreseen
elif float_compare(self.quantity_done, self.product_uom_qty, precision_rounding=rounding) > 0:
quantity_to_split = self.quantity_done - self.product_uom_qty
uom_qty_to_split = quantity_to_split # + no need to change existing self.product_uom_qty
link_procurement = True
if quantity_to_split:
extra_move = self.copy(default={'quantity_done': quantity_to_split, 'product_uom_qty': uom_qty_to_split, 'production_id': self.production_id.id,
'raw_material_production_id': self.raw_material_production_id.id,
'procurement_id': link_procurement and self.procurement_id.id or False})
extra_move.action_confirm()
if self.has_tracking != 'none':
qty_todo = self.quantity_done - quantity_to_split
for movelot in self.move_lot_ids.filtered(lambda x: x.done_wo):
if movelot.quantity_done and movelot.done_wo:
if float_compare(qty_todo, movelot.quantity_done, precision_rounding=rounding) >= 0:
qty_todo -= movelot.quantity_done
elif float_compare(qty_todo, 0, precision_rounding=rounding) > 0:
#split
remaining = movelot.quantity_done - qty_todo
movelot.quantity_done = qty_todo
movelot.copy(default={'move_id': extra_move.id, 'quantity_done': remaining})
qty_todo = 0
else:
movelot.move_id = extra_move.id
else:
self.quantity_done -= quantity_to_split
return extra_move
@api.multi
def move_validate(self):
''' Validate moves based on a production order. '''
moves = self._filter_closed_moves()
quant_obj = self.env['stock.quant']
moves_todo = self.env['stock.move']
moves_to_unreserve = self.env['stock.move']
# Create extra moves where necessary
for move in moves:
# Here, the `quantity_done` was already rounded to the product UOM by the `do_produce` wizard. However,
# it is possible that the user changed the value before posting the inventory by a value that should be
# rounded according to the move's UOM. In this specific case, we chose to round up the value, because it
# is what is expected by the user (if i consumed/produced a little more, the whole UOM unit should be
# consumed/produced and the moves are split correctly).
rounding = move.product_uom.rounding
move.quantity_done = float_round(move.quantity_done, precision_rounding=rounding, rounding_method ='UP')
if move.quantity_done <= 0:
continue
moves_todo |= move
moves_todo |= move._create_extra_move()
# Split moves where necessary and move quants
for move in moves_todo:
rounding = move.product_uom.rounding
if float_compare(move.quantity_done, move.product_uom_qty, precision_rounding=rounding) < 0:
# Need to do some kind of conversion here
qty_split = move.product_uom._compute_quantity(move.product_uom_qty - move.quantity_done, move.product_id.uom_id)
new_move = move.split(qty_split)
# If you were already putting stock.move.lots on the next one in the work order, transfer those to the new move
move.move_lot_ids.filtered(lambda x: not x.done_wo or x.quantity_done == 0.0).write({'move_id': new_move})
self.browse(new_move).quantity_done = 0.0
main_domain = [('qty', '>', 0)]
preferred_domain = [('reservation_id', '=', move.id)]
fallback_domain = [('reservation_id', '=', False)]
fallback_domain2 = ['&', ('reservation_id', '!=', move.id), ('reservation_id', '!=', False)]
preferred_domain_list = [preferred_domain] + [fallback_domain] + [fallback_domain2]
if move.has_tracking == 'none':
quants = quant_obj.quants_get_preferred_domain(move.product_qty, move, domain=main_domain, preferred_domain_list=preferred_domain_list)
self.env['stock.quant'].quants_move(quants, move, move.location_dest_id, owner_id=move.restrict_partner_id.id)
else:
for movelot in move.active_move_lot_ids:
if float_compare(movelot.quantity_done, 0, precision_rounding=rounding) > 0:
if not movelot.lot_id:
raise UserError(_('You need to supply a lot/serial number.'))
qty = move.product_uom._compute_quantity(movelot.quantity_done, move.product_id.uom_id)
quants = quant_obj.quants_get_preferred_domain(qty, move, lot_id=movelot.lot_id.id, domain=main_domain, preferred_domain_list=preferred_domain_list)
self.env['stock.quant'].quants_move(quants, move, move.location_dest_id, lot_id = movelot.lot_id.id, owner_id=move.restrict_partner_id.id)
moves_to_unreserve |= move
# Next move in production order
if move.move_dest_id and move.move_dest_id.state not in ('done', 'cancel'):
move.move_dest_id.action_assign()
moves_to_unreserve.quants_unreserve()
moves_todo.write({'state': 'done', 'date': fields.Datetime.now()})
return moves_todo
@api.multi
def action_done(self):
production_moves = self.filtered(lambda move: (move.production_id or move.raw_material_production_id) and not move.scrapped)
production_moves.move_validate()
return super(StockMove, self-production_moves).action_done()
@api.multi
def split_move_lot(self):
ctx = dict(self.env.context)
self.ensure_one()
view = self.env.ref('mrp.view_stock_move_lots')
serial = (self.has_tracking == 'serial')
only_create = False # Check picking type in theory
show_reserved = any([x for x in self.move_lot_ids if x.quantity > 0.0])
ctx.update({
'serial': serial,
'only_create': only_create,
'create_lots': True,
'state_done': self.is_done,
'show_reserved': show_reserved,
})
if ctx.get('w_production'):
action = self.env.ref('mrp.act_mrp_product_produce').read()[0]
action['context'] = ctx
return action
result = {
'name': _('Register Lots'),
'type': 'ir.actions.act_window',
'view_type': 'form',
'view_mode': 'form',
'res_model': 'stock.move',
'views': [(view.id, 'form')],
'view_id': view.id,
'target': 'new',
'res_id': self.id,
'context': ctx,
}
return result
@api.multi
def save(self):
return True
@api.multi
def action_confirm(self):
moves = self.env['stock.move']
for move in self:
moves |= move.action_explode()
# we go further with the list of ids potentially changed by action_explode
return super(StockMove, moves).action_confirm()
def action_explode(self):
""" Explodes pickings """
# in order to explode a move, we must have a picking_type_id on that move because otherwise the move
# won't be assigned to a picking and it would be weird to explode a move into several if they aren't
# all grouped in the same picking.
if not self.picking_type_id:
return self
bom = self.env['mrp.bom'].sudo()._bom_find(product=self.product_id, company_id=self.company_id.id)
if not bom or bom.type != 'phantom':
return self
phantom_moves = self.env['stock.move']
processed_moves = self.env['stock.move']
factor = self.product_uom._compute_quantity(self.product_uom_qty, bom.product_uom_id) / bom.product_qty
boms, lines = bom.sudo().explode(self.product_id, factor, picking_type=bom.picking_type_id)
for bom_line, line_data in lines:
phantom_moves += self._generate_move_phantom(bom_line, line_data['qty'])
for new_move in phantom_moves:
processed_moves |= new_move.action_explode()
if not self.split_from and self.procurement_id:
# Check if procurements have been made to wait for
moves = self.procurement_id.move_ids
if len(moves) == 1:
self.procurement_id.write({'state': 'done'})
if processed_moves and self.state == 'assigned':
# Set the state of resulting moves according to 'assigned' as the original move is assigned
processed_moves.write({'state': 'assigned'})
# delete the move with original product which is not relevant anymore
self.sudo().unlink()
return processed_moves
def _propagate_split(self, new_move, qty):
if not self.move_dest_id.raw_material_production_id:
super(StockMove, self)._propagate_split(new_move, qty)
def _generate_move_phantom(self, bom_line, quantity):
if bom_line.product_id.type in ['product', 'consu']:
return self.copy(default={
'picking_id': self.picking_id.id if self.picking_id else False,
'product_id': bom_line.product_id.id,
'product_uom': bom_line.product_uom_id.id,
'product_uom_qty': quantity,
'state': 'draft', # will be confirmed below
'name': self.name,
'procurement_id': self.procurement_id.id,
'split_from': self.id, # Needed in order to keep sale connection, but will be removed by unlink
})
return self.env['stock.move']
class PushedFlow(models.Model):
_inherit = "stock.location.path"
def _prepare_move_copy_values(self, move_to_copy, new_date):
new_move_vals = super(PushedFlow, self)._prepare_move_copy_values(move_to_copy, new_date)
new_move_vals['production_id'] = False
return new_move_vals