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

# -*- 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