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.
		
		
		
		
		
			
		
			
				
					
					
						
							250 lines
						
					
					
						
							11 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							250 lines
						
					
					
						
							11 KiB
						
					
					
				
								# -*- coding: utf-8 -*-
							 | 
						|
								###############################################################################
							 | 
						|
								#
							 | 
						|
								#    Cybrosys Technologies Pvt. Ltd.
							 | 
						|
								#
							 | 
						|
								#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>)
							 | 
						|
								#    Author: Ranjith R(odoo@cybrosys.com)
							 | 
						|
								#
							 | 
						|
								#    You can modify it under the terms of the GNU AFFERO
							 | 
						|
								#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3.
							 | 
						|
								#
							 | 
						|
								#    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 AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details.
							 | 
						|
								#
							 | 
						|
								#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
							 | 
						|
								#    (AGPL v3) along with this program.
							 | 
						|
								
							 | 
						|
								###############################################################################
							 | 
						|
								from odoo import api, fields, models
							 | 
						|
								from odoo.tools import date_utils
							 | 
						|
								from odoo.tools.safe_eval import datetime
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								class SubscriptionContracts(models.Model):
							 | 
						|
								    """ Model for subscription contracts """
							 | 
						|
								    _name = 'subscription.contracts'
							 | 
						|
								    _description = 'Subscription Contracts'
							 | 
						|
								    _inherit = ['mail.thread', 'mail.activity.mixin']
							 | 
						|
								
							 | 
						|
								    name = fields.Char(string='Contract Name', required=True,
							 | 
						|
								                       help='To add contract name')
							 | 
						|
								    reference = fields.Char(string='Reference', help='To add contract reference'
							 | 
						|
								                            )
							 | 
						|
								    partner_id = fields.Many2one('res.partner', string="Customer",
							 | 
						|
								                                 help='To add Customer')
							 | 
						|
								    recurring_period = fields.Integer(string='Recurring Period',
							 | 
						|
								                                      help='To add recurring period of '
							 | 
						|
								                                           'subscription contract')
							 | 
						|
								    recurring_period_interval = fields.Selection([
							 | 
						|
								        ('Days', 'Days'),
							 | 
						|
								        ('Weeks', 'Weeks'),
							 | 
						|
								        ('Months', 'Months'),
							 | 
						|
								        ('Years', 'Years'),
							 | 
						|
								    ], help='To add recurring interval of subscription contract')
							 | 
						|
								    contract_reminder = fields.Integer(
							 | 
						|
								        string='Contract Expiration Reminder (Days)',
							 | 
						|
								        help='Add expiry reminder of subscription contract in days.')
							 | 
						|
								    recurring_invoice = fields.Integer(
							 | 
						|
								        string='Recurring Invoice Interval (Days)',
							 | 
						|
								        help='Add recurring invoice interval in days')
							 | 
						|
								    next_invoice_date = fields.Date(string='Next Invoice Date', store=True,
							 | 
						|
								                                    compute='_compute_next_invoice_date',
							 | 
						|
								                                    help='Add date of next invoice')
							 | 
						|
								    company_id = fields.Many2one('res.company', string='Company',
							 | 
						|
								                                 default=lambda self: self.env.company,
							 | 
						|
								                                 help='To get company')
							 | 
						|
								    currency_id = fields.Many2one('res.currency', string='Currency',
							 | 
						|
								                                  help='To get currency', required=True,
							 | 
						|
								                                  default=lambda
							 | 
						|
								                                      self: self.env.company.currency_id)
							 | 
						|
								    date_start = fields.Date(string='Start Date', default=fields.Date.today(),
							 | 
						|
								                             help='To Add subscription contract start date')
							 | 
						|
								    invoice_count = fields.Integer(store=True,
							 | 
						|
								                                   compute='_compute_invoice_count',
							 | 
						|
								                                   string='Invoice count',
							 | 
						|
								                                   help='To get subscription contract end date')
							 | 
						|
								    date_end = fields.Date(string='End Date', help='Subscription End Date')
							 | 
						|
								    current_reference = fields.Integer(compute='_compute_sale_order_lines',
							 | 
						|
								                                       string='Current Subscription Id',
							 | 
						|
								                                       help='To add current Subscription id')
							 | 
						|
								    lock = fields.Boolean(string='Lock', default=False,
							 | 
						|
								                          help='To lock subscription contract')
							 | 
						|
								    state = fields.Selection([
							 | 
						|
								        ('New', 'New'),
							 | 
						|
								        ('Ongoing', 'Ongoing'),
							 | 
						|
								        ('Expire Soon', 'Expire Soon'),
							 | 
						|
								        ('Expired', 'Expired'),
							 | 
						|
								        ('Cancelled', 'Cancelled'),
							 | 
						|
								    ], string='Stage', default='New', copy=False, tracking=True,
							 | 
						|
								        readonly=True, help='Status of subscription contract')
							 | 
						|
								    contract_line_ids = fields.One2many(
							 | 
						|
								        'subscription.contracts.line',
							 | 
						|
								        'subscription_contract_id',
							 | 
						|
								        help='To Add Subscription Contract', string='Contract lines')
							 | 
						|
								    amount_total = fields.Monetary(string="Total", store=True,
							 | 
						|
								                                   compute='_compute_amount_total', tracking=4,
							 | 
						|
								                                   help='To get total amount')
							 | 
						|
								    sale_order_line_ids = fields.One2many('sale.order.line',
							 | 
						|
								                                          'contract_id',
							 | 
						|
								                                          help='Sale Order Lines',
							 | 
						|
								                                          string='To add order lines')
							 | 
						|
								    note = fields.Html(string="Terms and conditions",
							 | 
						|
								                       help='To add terms and conditions', translate=True)
							 | 
						|
								    invoices_active = fields.Boolean('Invoice active', default=False,
							 | 
						|
								                                     help='To active invoices smart button',
							 | 
						|
								                                     compute='_compute_invoice_active')
							 | 
						|
								
							 | 
						|
								    def action_to_confirm(self):
							 | 
						|
								        """ Button to confirm """
							 | 
						|
								        self.write({'state': 'Ongoing'})
							 | 
						|
								
							 | 
						|
								    def action_to_cancel(self):
							 | 
						|
								        """ Button to cancel """
							 | 
						|
								        self.write({'state': 'Cancelled'})
							 | 
						|
								
							 | 
						|
								    def action_generate_invoice(self):
							 | 
						|
								        """ Button to generate invoice """
							 | 
						|
								        self.env['account.move'].create(
							 | 
						|
								            {
							 | 
						|
								                'move_type': 'out_invoice',
							 | 
						|
								                'partner_id': self.partner_id.id,
							 | 
						|
								                'invoice_date': fields.date.today(),
							 | 
						|
								                'contract_origin': self.id,
							 | 
						|
								                'invoice_line_ids': [(0, 0, {
							 | 
						|
								                    'product_id': line.product_id.id,
							 | 
						|
								                    'name': line.description,
							 | 
						|
								                    'quantity': line.qty_ordered,
							 | 
						|
								                    'price_unit': line.price_unit,
							 | 
						|
								                    'tax_ids': line.tax_ids,
							 | 
						|
								                    'discount': line.discount,
							 | 
						|
								                }) for line in self.contract_line_ids]
							 | 
						|
								            })
							 | 
						|
								        self.invoice_count = self.env['account.move'].search_count([
							 | 
						|
								            ('contract_origin', '=', self.id)])
							 | 
						|
								
							 | 
						|
								    def action_lock(self):
							 | 
						|
								        """ Button to lock subscription contract """
							 | 
						|
								        self.lock = True
							 | 
						|
								
							 | 
						|
								    def action_to_unlock(self):
							 | 
						|
								        """ Button to unlock subscription contract """
							 | 
						|
								        self.lock = False
							 | 
						|
								
							 | 
						|
								    def action_get_invoice(self):
							 | 
						|
								        """ Invoices smart button """
							 | 
						|
								        self.ensure_one()
							 | 
						|
								        return {
							 | 
						|
								            'type': 'ir.actions.act_window',
							 | 
						|
								            'name': 'Invoices',
							 | 
						|
								            'view_mode': 'tree,form',
							 | 
						|
								            'res_model': 'account.move',
							 | 
						|
								            'domain': [('contract_origin', '=', self.id)],
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								    @api.depends('contract_line_ids.sub_total')
							 | 
						|
								    def _compute_amount_total(self):
							 | 
						|
								        """ Compute total amount """
							 | 
						|
								        for order in self:
							 | 
						|
								            order_lines = order.contract_line_ids
							 | 
						|
								            order.amount_total = sum(order_lines.mapped('sub_total'))
							 | 
						|
								
							 | 
						|
								    @api.depends('partner_id')
							 | 
						|
								    def _compute_invoice_count(self):
							 | 
						|
								        """ Compute invoice count """
							 | 
						|
								        self.invoice_count = self.env['account.move'].search_count([
							 | 
						|
								            ('contract_origin', '=', self.id)
							 | 
						|
								        ])
							 | 
						|
								
							 | 
						|
								    @api.depends('invoices_active')
							 | 
						|
								    def _compute_invoice_active(self):
							 | 
						|
								        """ Check invoice count to show the invoice smart button """
							 | 
						|
								        invoice_count = self.env['account.move'].search_count([
							 | 
						|
								            ('contract_origin', '=', self.id)
							 | 
						|
								        ])
							 | 
						|
								        if invoice_count != 0:
							 | 
						|
								            self.invoices_active = True
							 | 
						|
								        else:
							 | 
						|
								            self.invoices_active = False
							 | 
						|
								
							 | 
						|
								    @api.depends('date_start', 'recurring_invoice', 'recurring_period',
							 | 
						|
								                 'recurring_period_interval')
							 | 
						|
								    def _compute_next_invoice_date(self):
							 | 
						|
								        """ Compute next invoice date """
							 | 
						|
								        self.next_invoice_date = fields.Date.today()
							 | 
						|
								        start_date = self.date_start
							 | 
						|
								        interval = self.recurring_invoice
							 | 
						|
								        recurring_period = self.recurring_period
							 | 
						|
								        recurring_period_interval = self.recurring_period_interval
							 | 
						|
								        self.next_invoice_date = date_utils.add(start_date,
							 | 
						|
								                                                days=int(interval))
							 | 
						|
								        if recurring_period_interval == 'Days':
							 | 
						|
								            next_schedule = date_utils.add(start_date,
							 | 
						|
								                                           days=int(recurring_period))
							 | 
						|
								            self.date_end = next_schedule
							 | 
						|
								        elif recurring_period_interval == 'Weeks':
							 | 
						|
								            next_schedule = date_utils.add(start_date,
							 | 
						|
								                                           weeks=int(recurring_period))
							 | 
						|
								            self.date_end = next_schedule
							 | 
						|
								        elif recurring_period_interval == 'Months':
							 | 
						|
								            next_schedule = date_utils.add(start_date,
							 | 
						|
								                                           months=int(recurring_period))
							 | 
						|
								            self.date_end = next_schedule
							 | 
						|
								        else:
							 | 
						|
								            next_schedule = date_utils.add(start_date,
							 | 
						|
								                                           years=int(recurring_period))
							 | 
						|
								            self.date_end = next_schedule
							 | 
						|
								
							 | 
						|
								    @api.model
							 | 
						|
								    def subscription_contract_state_change(self):
							 | 
						|
								        """ Automatic state change and create invoice """
							 | 
						|
								        records = self.env['subscription.contracts'].search([])
							 | 
						|
								        for rec in records:
							 | 
						|
								            end_date = rec.date_end
							 | 
						|
								            expiry_reminder = rec.contract_reminder
							 | 
						|
								            expiry_warning_date = date_utils.subtract(end_date,
							 | 
						|
								                                                      days=int(
							 | 
						|
								                                                          expiry_reminder))
							 | 
						|
								            current_date = fields.Date.today()
							 | 
						|
								            next_invoice_date = rec.next_invoice_date
							 | 
						|
								            if expiry_warning_date <= current_date <= end_date:
							 | 
						|
								                rec.write({'state': 'Expire Soon'})
							 | 
						|
								            if end_date < current_date:
							 | 
						|
								                rec.write({'state': 'Expired'})
							 | 
						|
								            if next_invoice_date == current_date and rec.state != 'Cancelled':
							 | 
						|
								                data = rec.env['account.move'].create([
							 | 
						|
								                    {
							 | 
						|
								                        'move_type': 'out_invoice',
							 | 
						|
								                        'partner_id': rec.partner_id.id,
							 | 
						|
								                        'invoice_date': fields.date.today(),
							 | 
						|
								                        'contract_origin': rec.id,
							 | 
						|
								                    }])
							 | 
						|
								                for line in rec.contract_line_ids:
							 | 
						|
								                    data.write({
							 | 
						|
								                        'invoice_line_ids': [(0, 0, {
							 | 
						|
								                            'product_id': line.product_id.id,
							 | 
						|
								                            'name': line.description,
							 | 
						|
								                            'quantity': line.qty_ordered,
							 | 
						|
								                            'price_unit': line.price_unit,
							 | 
						|
								                            'tax_ids': line.tax_ids,
							 | 
						|
								                            'discount': line.discount,
							 | 
						|
								                        })],
							 | 
						|
								                    })
							 | 
						|
								                rec.invoice_count = rec.env['account.move'].search_count([
							 | 
						|
								                    ('contract_origin', '=', rec.id)])
							 | 
						|
								
							 | 
						|
								    @api.depends('current_reference')
							 | 
						|
								    def _compute_sale_order_lines(self):
							 | 
						|
								        """ Get sale order line"""
							 | 
						|
								        self.current_reference = self.id
							 | 
						|
								        product_id = self.contract_line_ids.mapped('product_id')
							 | 
						|
								        sale_order_line = self.env['sale.order.line'].search([
							 | 
						|
								            ('order_partner_id', '=', self.partner_id.id)
							 | 
						|
								        ])
							 | 
						|
								        for rec in sale_order_line:
							 | 
						|
								            if self.date_start <= datetime.datetime.date(
							 | 
						|
								                    rec.create_date) <= self.date_end:
							 | 
						|
								                if rec.product_id in product_id:
							 | 
						|
								                    rec.contract_id = self.id
							 | 
						|
								
							 |