You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

344 lines
17 KiB

# -*- coding: utf-8 -*-
##############################################################################
#
# Cybrosys Technologies Pvt. Ltd.
# Copyright (C) 2017-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
# Author: Nikhil krishnan(<https://www.cybrosys.com>)
# you can modify it under the terms of the GNU LESSER
# GENERAL PUBLIC LICENSE (LGPL v3), Version 3.
#
# It is forbidden to publish, distribute, sublicense, or sell copies
# of the Software or modified copies of the Software.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details.
#
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE
# GENERAL PUBLIC LICENSE (LGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
##############################################################################
import datetime
from datetime import datetime
from odoo import models, fields, api, exceptions, _
from odoo.exceptions import Warning
from odoo.exceptions import UserError
import odoo.addons.decimal_precision as dp
class MrpProduction(models.Model):
""" Manufacturing Orders """
_inherit = 'mrp.production'
@api.multi
def _indent_count(self):
for order in self:
indent_count = self.env['mrp.indent'].search([('origin', '=', order.id)])
order.mrp_indent_order_count = len(indent_count)
@api.multi
def mrp_indent_confirm(self):
for order in self:
indent_id = self.env['mrp.indent'].search([('origin', '=', order.id)])
if indent_id:
for indent in indent_id:
if not indent.move_lines:
raise exceptions.Warning(_("Warning "
"You cannot confirm an indent %s which has no line." % indent.name))
else:
indent.write({'state': 'waiting_approval'})
self.indent_state = 'waiting_approval'
@api.multi
def action_before_assign(self):
indent_count = self.env['mrp.indent'].search([('origin', '=', self.id)])
if not indent_count:
self.indent_state = 'indent_created'
vals = {
'origin': self.id,
'required_date': self.date_planned_start,
'item_for': 'mrp',
'company_id': self.company_id.id,
}
indent_obj = self.env['mrp.indent'].create(vals)
for move in self.move_raw_ids:
move.write({'mrp_indent_id': indent_obj.id})
else:
if self.indent_state == 'indent_created':
raise UserError(_("Indent already created, Please Check and Confirm your indent"))
else:
raise UserError(_("Indent already created, Please wait for the store team approval"))
@api.multi
def action_cancel(self):
indent_ids = self.env['mrp.indent'].search([('origin', '=', self.id)])
if indent_ids:
for indent in indent_ids:
indent.write({'state': 'cancel'})
self.indent_state = 'cancel'
return super(MrpProduction, self).action_cancel()
mrp_indent_order_count = fields.Integer(string='# of Indent Orders', compute='_indent_count')
indent_state = fields.Selection(
[('draft', 'Not indented'),
('indent_created', 'Indent Created'),
('waiting_approval', 'Waiting for Approval'),
('done', 'Indent Approved'),
('cancel', 'Indent Canceled'),
('reject', 'Indent Rejected')], string='Indent Status', readonly=True, copy=False, default='draft')
class MrpIndent(models.Model):
_name = "mrp.indent"
@api.multi
def action_assign(self):
for production in self:
move_to_assign = production.move_lines.filtered(lambda x: x.state in ('confirmed', 'waiting', 'assigned'))
move_to_assign.action_assign()
if self.origin.availability == 'assigned':
self.issued_date = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
self.state = 'done'
self.origin.write({'indent_state': 'done'})
return True
@api.model
def create(self, vals):
vals['name'] = self.env['ir.sequence'].next_by_code('mrp.indent') or '/'
return super(MrpIndent, self).create(vals)
name = fields.Char(string='name', readonly=True, copy=False)
indent_date = fields.Datetime(string='Indent Date', required=True, default=fields.Datetime.now, readonly=True,
states={'draft': [('readonly', False)]})
required_date = fields.Datetime(string='Required Date', required=True, readonly=True, default=fields.Datetime.now,
states={'draft': [('readonly', False)]})
origin = fields.Many2one('mrp.production', string='Source Document', readonly=True, copy=False, states={'draft': [('readonly', False)]})
issued_date = fields.Datetime(string='Approve Date', readonly=True)
issued_by = fields.Many2one('res.users', string='Issued by', readonly=True)
requirement = fields.Selection([('1', 'Ordinary'), ('2', 'Urgent')], 'Requirement', readonly=True, default='1',
states={'draft': [('readonly', False)]})
item_for = fields.Selection([('mrp', 'Produce'), ('other', 'Other')], string='Order for', default='other',
readonly=True, states={'draft': [('readonly', False)]})
move_lines = fields.One2many('stock.move', 'mrp_indent_id', string='Moves', copy=False, readonly=True)
product_lines = fields.One2many('mrp.indent.product.lines', 'indent_id', string='Product', copy=False)
description = fields.Text(string='Additional Information', readonly=True,
states={'draft': [('readonly', False)], 'waiting_approval': [('readonly', False)]})
company_id = fields.Many2one('res.company', string='Company', default=lambda self: self.env.user.company_id,
readonly=True, states={'draft': [('readonly', False)]})
state = fields.Selection(
[('draft', 'Draft'),
('waiting_approval', 'Waiting for Approval'),
('inprogress', 'Ready to Transfer'),
('move_created', 'Moves Created'),
('done', 'Done'),
('cancel', 'Cancel'),
('reject', 'Rejected')], string='State', readonly=True, default='draft', track_visibility='onchange')
@api.multi
def mrp_indent_confirm(self):
for indent in self:
if indent.item_for == 'mrp':
if not indent.move_lines:
raise exceptions.Warning(_("Warning "
"You cannot confirm an indent %s which has no line." % indent.name))
else:
indent.write({'state': 'waiting_approval'})
indent.origin.write({'indent_state': 'waiting_approval'})
else:
if not indent.product_lines:
raise exceptions.Warning(_("Warning "
"You cannot confirm an indent %s which has "
"no product line." % indent.name))
else:
indent.write({'state': 'waiting_approval'})
@api.one
def mrp_indent_inprogress(self):
todo = []
for o in self:
if not any(line for line in o.product_lines):
raise exceptions.Warning(_('Error!'),
_('You cannot Approve a order without any order line.'))
for line in o.product_lines:
if line:
todo.append(line.id)
appr_date = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
self.env['mrp.indent.product.lines'].action_confirm(todo)
for id in self.ids:
self.write({'state': 'inprogress', 'issued_date': appr_date})
return True
@api.one
def indent_reject(self):
if self.move_lines:
for line in self.move_lines:
if line.state == 'cancel':
pass
elif line.state == 'done':
pass
else:
line.action_cancel()
self.write({'state': 'reject'})
if self.origin:
self.origin.action_cancel()
@api.multi
def indent_transfer(self):
name = self.name
move_lines_obj = self.env['stock.move']
if self.product_lines:
for line in self.product_lines:
if line.product_id.type != 'service':
if line.location_id:
if line.location_dest_id:
tot_qty = 0
obj_quant = self.env['stock.quant'].search([('product_id', '=', line.product_id.id),
('location_id', '=', line.location_id.id)])
for obj in obj_quant:
tot_qty += obj.qty
move_line = {}
if line.product_id.type == 'consu':
move_line = {
'product_id': line.product_id.id,
'state': "draft",
'product_uom_qty': line.product_uom_qty,
'product_uom': line.product_id.uom_id.id,
'name': line.product_id.name,
'location_id': line.location_id.id,
'location_dest_id': line.location_dest_id.id,
'date': datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
'date_expected': self.required_date,
'invoice_state': "none",
'origin': name,
'mrp_indent_id': self.id
}
move_lines_obj.create(move_line)
else:
move_line = {}
if tot_qty >= line.product_uom_qty:
move_line = {
'product_id': line.product_id.id,
'state': "draft",
'product_uom_qty': line.product_uom_qty,
'product_uom': line.product_id.uom_id.id,
'name': line.product_id.name,
'location_id': line.location_id.id,
'location_dest_id': line.location_dest_id.id,
'date': datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
'date_expected': self.required_date,
'invoice_state': "none",
'origin': name,
'mrp_indent_id': self.id
}
move_lines_obj.create(move_line)
else:
if tot_qty:
raise exceptions.Warning((" No sufficient stock for product ' %s ' in '%s'. "
"Available quantity is %s %s.") %
(line.product_id.name, line.location_id.name, tot_qty,
line.product_uom.name))
else:
raise exceptions.Warning(
(" No stock for product ' %s ' in '%s'."
" Please continue with another location ") % (line.product_id.name,
line.location_id.name))
else:
raise exceptions.Warning((" Destination Location is not set properly for' %s '. "
"So Plese cancel this indent and create a new one please.")
% line.product_id.name)
else:
raise exceptions.Warning(("Source Location is not set properly for ' %s '. "
"Please go and set Source Location.")
% line.product_id.name)
else:
raise exceptions.Warning("This product is a service type product.")
else:
raise exceptions.Warning('You cannot Transfer a order without any product line.')
self.write({'state': 'move_created'})
@api.multi
def indent_transfer_move_confirm(self):
if self.move_lines:
for line in self.move_lines:
if line.state == 'cancel':
pass
elif line.state == 'done':
pass
else:
line.action_done()
else:
raise Warning(_('Error!'),
_('You cannot Confirm a order without any move lines.'))
self.write({'state': 'done'})
class IndentProductLines(models.Model):
_name = 'mrp.indent.product.lines'
_description = 'Indent Product Lines'
@api.one
def action_confirm(self, todo):
self.write({'state': 'inprogress'})
return True
indent_id = fields.Many2one('mrp.indent', string='Indent', required=True, ondelete='cascade')
name = fields.Text(string='Description', required=True, readonly=True,
states={'draft': [('readonly', False)], 'waiting_approval': [('readonly', False)]})
product_id = fields.Many2one('product.product', string='Product', required=True, readonly=True,
states={'draft': [('readonly', False)], 'waiting_approval': [('readonly', False)]})
original_product_id = fields.Many2one('product.product', string='Product to be Manufactured', readonly=True,
states={'draft': [('readonly', False)], 'waiting_approval': [('readonly', False)]})
product_uom_qty = fields.Float(string='Quantity Required', digits_compute=dp.get_precision('Product UoS'), required=True,
readonly=True, states={'draft': [('readonly', False)],
'waiting_approval': [('readonly', False)],
'inprogress': [('readonly', False)]})
product_uom = fields.Many2one('product.uom', string='Unit of Measure', required=True, readonly=True,
states={'draft': [('readonly', False)], 'waiting_approval': [('readonly', False)],
'inprogress': [('readonly', False)]})
location_id = fields.Many2one('stock.location', string='Source Location', readonly=True,
states={'inprogress': [('readonly', False)]})
location_dest_id = fields.Many2one('stock.location', string='Destination Location', required=True, readonly=True,
states={'draft': [('readonly', False)],
'waiting_approval': [('readonly', False)]})
delay = fields.Float(string='Lead Time')
purpose = fields.Text(string='Purpose')
state = fields.Selection(
[('draft', 'Draft'),
('waiting_approval', 'Waiting for Approval'),
('inprogress', 'Ready to Transfer'),
('move_created', 'Moves Created'),
('done', 'Done'),
('cancel', 'Cancel'),
('reject', 'Rejected')], string='State', default='draft', related='indent_id.state')
sequence = fields.Integer('Sequence')
def onchange_product_id(self, product_id=False, product_uom_qty=0.0, product_uom=False, name=''):
product_obj = self.env['product.product']
value = {}
if not product_id:
return {'value': {'product_uom_qty': 1.0, 'product_uom': False,
'name': '', 'specification': '', 'delay': 0.0}}
product = product_obj.browse(product_id)
value['name'] = product.name_get()[0][1]
value['product_uom'] = product.uom_id.id
value['specification'] = product.name_get()[0][1]
return {'value': value}
class StockMove(models.Model):
_inherit = 'stock.move'
mrp_indent_id = fields.Many2one('mrp.indent', 'Indent')