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.
		
		
		
		
		
			
		
			
				
					
					
						
							156 lines
						
					
					
						
							7.2 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							156 lines
						
					
					
						
							7.2 KiB
						
					
					
				
								# -*- coding: utf-8 -*-
							 | 
						|
								"""
							 | 
						|
								Can use only selected products to invoice as well as bills.
							 | 
						|
								"""
							 | 
						|
								###############################################################################
							 | 
						|
								#
							 | 
						|
								#    Cybrosys Technologies Pvt. Ltd.
							 | 
						|
								#    Copyright (C) 2023-TODAY Cybrosys Technologies(<https://www.cybrosys.com>).
							 | 
						|
								#    Author: Saneen K (odoo@cybrosys.com)
							 | 
						|
								#
							 | 
						|
								#    This program is free software: you can modify
							 | 
						|
								#    it under the terms of the GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) as
							 | 
						|
								#    published by the Free Software Foundation, either version 3 of the
							 | 
						|
								#    License, or (at your option) any later version.
							 | 
						|
								#
							 | 
						|
								#    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 for more details.
							 | 
						|
								#
							 | 
						|
								#    You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE
							 | 
						|
								#    along with this program.  If not, see <https://www.gnu.org/licenses/>.
							 | 
						|
								#
							 | 
						|
								###############################################################################
							 | 
						|
								from odoo import fields, models, _
							 | 
						|
								from odoo.exceptions import UserError
							 | 
						|
								from odoo.tools import float_is_zero, groupby
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								class PurchaseOrderLine(models.Model):
							 | 
						|
								    """Inherits the purchase order line"""
							 | 
						|
								    _inherit = 'purchase.order.line'
							 | 
						|
								
							 | 
						|
								    is_product_select = fields.Boolean(string="Select",
							 | 
						|
								                                       help="Select products from order line",
							 | 
						|
								                                       copy=False)
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								class PurchaseOrder(models.Model):
							 | 
						|
								    """Inherits the purchase order"""
							 | 
						|
								    _inherit = 'purchase.order'
							 | 
						|
								
							 | 
						|
								    def action_select_all(self):
							 | 
						|
								        """Select all the products from the sale order line for invoice"""
							 | 
						|
								        self.order_line.filtered(lambda rec: rec.qty_to_invoice > 0).write(
							 | 
						|
								            {'is_product_select': True})
							 | 
						|
								
							 | 
						|
								    def action_deselect_all(self):
							 | 
						|
								        """Deselect all the products from the sale order line for invoice"""
							 | 
						|
								        self.order_line.filtered(lambda rec: rec.qty_to_invoice > 0).write(
							 | 
						|
								            {'is_product_select': False})
							 | 
						|
								
							 | 
						|
								    def action_create_invoice(self):
							 | 
						|
								        """Create the invoice associated to the PO and Update the invoiceable
							 | 
						|
								         lines on the basis of selected lines"""
							 | 
						|
								        if self.order_line.filtered(
							 | 
						|
								                lambda selected: selected.is_product_select):
							 | 
						|
								            # 1) Prepare invoice vals and clean-up the section lines
							 | 
						|
								            invoice_vals_list = []
							 | 
						|
								            sequence = 10
							 | 
						|
								            for order in self:
							 | 
						|
								                if order.invoice_status != 'to invoice':
							 | 
						|
								                    continue
							 | 
						|
								                order = order.with_company(order.company_id)
							 | 
						|
								                pending_section = None
							 | 
						|
								                # Invoice values.
							 | 
						|
								                invoice_vals = order._prepare_invoice()
							 | 
						|
								                # Invoice line values (keep only necessary sections).
							 | 
						|
								                for line in order.order_line.filtered(lambda line_select: (
							 | 
						|
								                                  line_select.is_product_select) or True not in
							 | 
						|
								                          self.order_line.mapped(
							 | 
						|
								                              'is_product_select') or
							 | 
						|
								                          line_select.display_type == 'line_section'):
							 | 
						|
								                    if line.display_type == 'line_section':
							 | 
						|
								                        pending_section = line
							 | 
						|
								                        continue
							 | 
						|
								                    if not float_is_zero(line.qty_to_invoice,
							 | 
						|
								                                         precision_digits=self.env[
							 | 
						|
								                                             'decimal.precision'].precision_get(
							 | 
						|
								                                             'Product Unit of Measure')):
							 | 
						|
								                        if pending_section:
							 | 
						|
								                            line_vals = pending_section._prepare_account_move_line()
							 | 
						|
								                            line_vals.update({'sequence': sequence})
							 | 
						|
								                            invoice_vals['invoice_line_ids'].append(
							 | 
						|
								                                (0, 0, line_vals))
							 | 
						|
								                            sequence += 1
							 | 
						|
								                            pending_section = None
							 | 
						|
								                        line_vals = line._prepare_account_move_line()
							 | 
						|
								                        line_vals.update({'sequence': sequence})
							 | 
						|
								                        invoice_vals['invoice_line_ids'].append(
							 | 
						|
								                            (0, 0, line_vals))
							 | 
						|
								                        sequence += 1
							 | 
						|
								                invoice_vals_list.append(invoice_vals)
							 | 
						|
								            if not invoice_vals_list:
							 | 
						|
								                raise UserError(
							 | 
						|
								                    _('There is no invoiceable line. If a product has a'
							 | 
						|
								                      ' control policy based on received quantity, '
							 | 
						|
								                      'please make sure that a quantity has been '
							 | 
						|
								                      'received.'))
							 | 
						|
								            # 2) group by (company_id, partner_id, currency_id) for batch creation
							 | 
						|
								            new_invoice_vals_list = []
							 | 
						|
								            for grouping_keys, invoices in groupby(
							 | 
						|
								                    invoice_vals_list,
							 | 
						|
								                    key=lambda x: (x.get('company_id'), x.get(
							 | 
						|
								                        'partner_id'), x.get('currency_id'))):
							 | 
						|
								                ref_invoice_vals = self.get_ref_invoice_vals(invoices=invoices)
							 | 
						|
								                new_invoice_vals_list.append(ref_invoice_vals)
							 | 
						|
								            invoice_vals_list = new_invoice_vals_list
							 | 
						|
								            # 3) Create invoices.
							 | 
						|
								            moves = self.env['account.move']
							 | 
						|
								            if invoice_vals_list[0].get('invoice_line_ids'):
							 | 
						|
								                for vals in invoice_vals_list:
							 | 
						|
								                    moves |= self.env['account.move'].with_context(
							 | 
						|
								                        default_move_type='in_invoice').with_company(
							 | 
						|
								                        vals['company_id']).create(
							 | 
						|
								                        vals)
							 | 
						|
								                # 4) Some moves might actually be refunds: convert them if the
							 | 
						|
								                # total amount is negative
							 | 
						|
								                # We do this after the moves have been created since we need taxes,
							 | 
						|
								                # 'etc. to know if the total
							 | 
						|
								                # is actually negative or not
							 | 
						|
								                moves.filtered(
							 | 
						|
								                    lambda m: m.currency_id.round(m.amount_total) < 0) \
							 | 
						|
								                    .action_switch_invoice_into_refund_credit_note()
							 | 
						|
								                return self.action_view_invoice(moves)
							 | 
						|
								            raise UserError(
							 | 
						|
								                _('There is no invoiceable line. Please make sure '
							 | 
						|
								                  'that a order line is selected'))
							 | 
						|
								        raise UserError(_('There is no invoiceable line. Please make sure '
							 | 
						|
								                          'that a order line is selected'))
							 | 
						|
								
							 | 
						|
								    def get_ref_invoice_vals(self, invoices):
							 | 
						|
								        """
							 | 
						|
								        This is the method get_ref_invoice_vals which will return the invoice
							 | 
						|
								        vals to the method
							 | 
						|
								        """
							 | 
						|
								        origins = set()
							 | 
						|
								        payment_refs = set()
							 | 
						|
								        refs = set()
							 | 
						|
								        ref_invoice_vals = {}
							 | 
						|
								        for invoice_vals in invoices:
							 | 
						|
								            if not ref_invoice_vals:
							 | 
						|
								                ref_invoice_vals = invoice_vals
							 | 
						|
								            else:
							 | 
						|
								                ref_invoice_vals['invoice_line_ids'] += invoice_vals[
							 | 
						|
								                    'invoice_line_ids']
							 | 
						|
								            origins.add(invoice_vals['invoice_origin'])
							 | 
						|
								            payment_refs.add(invoice_vals['payment_reference'])
							 | 
						|
								            refs.add(invoice_vals['ref'])
							 | 
						|
								        ref_invoice_vals.update({
							 | 
						|
								            'ref': ', '.join(refs)[:2000],
							 | 
						|
								            'invoice_origin': ', '.join(origins),
							 | 
						|
								            'payment_reference': len(
							 | 
						|
								                payment_refs) == 1 and payment_refs.pop() or False,
							 | 
						|
								        })
							 | 
						|
								        return ref_invoice_vals
							 | 
						|
								
							 |