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.
 
 
 
 
 

254 lines
11 KiB

# -*- coding: utf-8 -*-
##############################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>)
# Author: Cybrosys Techno Solutions(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.
# If not, see <http://www.gnu.org/licenses/>.
#
##############################################################################
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='Name of Contract')
reference = fields.Char(string='Reference', help='Contract reference')
partner_id = fields.Many2one('res.partner', string="Customer",
help='Customer for this contract')
recurring_period = fields.Integer(string='Recurring Period',
help='Recurring period of '
'subscription contract')
recurring_period_interval = fields.Selection([
('Days', 'Days'),
('Weeks', 'Weeks'),
('Months', 'Months'),
('Years', 'Years'),
], help='Recurring interval of subscription contract')
contract_reminder = fields.Integer(
string='Contract Expiration Reminder (Days)',
help='Expiry reminder of subscription contract in days.')
recurring_invoice = fields.Integer(
string='Recurring Invoice Interval (Days)',
help='Recurring invoice interval in days')
next_invoice_date = fields.Date(string='Next Invoice Date', store=True,
compute='_compute_next_invoice_date',
help='Date of next invoice')
company_id = fields.Many2one('res.company', string='Company',
default=lambda self: self.env.company)
currency_id = fields.Many2one(
'res.currency', string='Currency',
required=True, default=lambda self: self.env.company.currency_id)
date_start = fields.Date(string='Start Date', default=fields.Date.today(),
help='Subscription contract start date')
invoice_count = fields.Integer(store=True,
compute='_compute_invoice_count',
string='Invoice count',
help='Number of invoices generated')
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='Current Subscription id')
lock = fields.Boolean(string='Lock', default=False,
help='Lock subscription contract so that further'
' modifications are not possible.')
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',
string='Contract lines', help='Products to be added in the contract')
amount_total = fields.Monetary(string="Total", store=True,
compute='_compute_amount_total', tracking=4,
help='Total amount')
sale_order_line_ids = fields.One2many(
'sale.order.line', 'contract_id',
string='Sale Order Lines',
help='Order lines of Sale Orders which belongs to this contract')
note = fields.Html(string="Terms and conditions",
help='Add any notes', translate=True)
invoices_active = fields.Boolean(
'Invoice active', default=False,
compute='_compute_invoice_active',
help='Compute invoices are active or not')
def action_to_confirm(self):
""" Confirm the Contract """
self.write({'state': 'Ongoing'})
def action_to_cancel(self):
""" Cancel the Contract """
self.write({'state': 'Cancelled'})
def action_generate_invoice(self):
""" 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):
""" Lock subscription contract """
self.lock = True
def action_to_unlock(self):
""" Unlock subscription contract """
self.lock = False
def action_get_invoice(self):
""" Access generated invoices """
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 of Contract """
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 the count of invoices generated """
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 display 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 of contract """
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 of contract lines """
print("sale order line compute",self.current_reference)
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)
])
print(sale_order_line)
print("products",product_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