@ -0,0 +1,46 @@ |
|||||
|
.. image:: https://img.shields.io/badge/license-AGPL--3-blue.svg |
||||
|
:target: https://www.gnu.org/licenses/agpl-3.0-standalone.html |
||||
|
:alt: License: AGPL-3 |
||||
|
|
||||
|
Subscription Management |
||||
|
======================= |
||||
|
* Subscription Package for Odoo 18 community edition |
||||
|
|
||||
|
License |
||||
|
------- |
||||
|
General Public License, Version 3 (AGPL v3). |
||||
|
(https://www.gnu.org/licenses/agpl-3.0-standalone.html) |
||||
|
|
||||
|
Company |
||||
|
------- |
||||
|
* `Cybrosys Techno Solutions <https://cybrosys.com/>`__ |
||||
|
|
||||
|
Credits |
||||
|
------- |
||||
|
Developers: (V15) Amal Prasad |
||||
|
(V16) Archana V |
||||
|
(V17) Janish Babu EK |
||||
|
(V18) Sreerag PM |
||||
|
Contact: odoo@cybrosys.com |
||||
|
|
||||
|
Contacts |
||||
|
-------- |
||||
|
* Mail Contact : odoo@cybrosys.com |
||||
|
* Website : https://cybrosys.com |
||||
|
|
||||
|
Bug Tracker |
||||
|
----------- |
||||
|
Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported. |
||||
|
|
||||
|
Maintainer |
||||
|
========== |
||||
|
.. image:: https://cybrosys.com/images/logo.png |
||||
|
:target: https://cybrosys.com |
||||
|
|
||||
|
This module is maintained by Cybrosys Technologies. |
||||
|
|
||||
|
For support and more information, please visit `Our Website <https://cybrosys.com/>`__ |
||||
|
|
||||
|
Further information |
||||
|
=================== |
||||
|
HTML Description: `<static/description/index.html>`__ |
@ -0,0 +1,24 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 . import models |
||||
|
from . import report |
||||
|
from . import wizard |
@ -0,0 +1,66 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
{ |
||||
|
'name': 'Subscription Management', |
||||
|
'version': '18.0.1.0.0', |
||||
|
'category': 'Sales', |
||||
|
'summary': 'Subscription Package Management Module For Odoo18 Community', |
||||
|
'description': 'Subscription Package Management Module specifically ' |
||||
|
'designed for Odoo 18 Community edition. ' |
||||
|
'This module aims to enhance the subscription ' |
||||
|
'management capabilities within the Odoo platform, ' |
||||
|
'providing users with advanced features and ' |
||||
|
'functionalities for efficiently handling subscription ' |
||||
|
'packages in the community version of Odoo 18.', |
||||
|
'author': 'Cybrosys Techno solutions', |
||||
|
'company': 'Cybrosys Techno Solutions', |
||||
|
'maintainer': 'Cybrosys Techno Solutions', |
||||
|
'website': "https://www.cybrosys.com", |
||||
|
'depends': ['base', 'sale_management'], |
||||
|
'data': [ |
||||
|
'security/subscription_package_groups.xml', |
||||
|
'security/ir.model.access.csv', |
||||
|
'data/uom_demo_data.xml', |
||||
|
'data/subscription_package_stop_data.xml', |
||||
|
'data/subscription_stage_data.xml', |
||||
|
'data/mail_subscription_renew_data.xml', |
||||
|
'data/ir_cron_data.xml', |
||||
|
'data/ir_sequence.xml', |
||||
|
'views/subscription_package_views.xml', |
||||
|
'views/product_template_views.xml', |
||||
|
'views/subscription_package_plan_views.xml', |
||||
|
'views/subscription_stage_views.xml', |
||||
|
'views/subscription_package_stop_views.xml', |
||||
|
'views/mail_activity_views.xml', |
||||
|
'views/res_partner_views.xml', |
||||
|
'views/recurrence_period_views.xml', |
||||
|
'views/sale_order_views.xml', |
||||
|
'views/product_product_views.xml', |
||||
|
'report/subscription_report_view.xml', |
||||
|
'wizard/subscription_close_views.xml', |
||||
|
], |
||||
|
'images': ['static/description/banner.jpg'], |
||||
|
'license': 'AGPL-3', |
||||
|
'installable': True, |
||||
|
'auto_install': False, |
||||
|
'application': True, |
||||
|
} |
@ -0,0 +1,12 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Cron Job: To check close date --> |
||||
|
<record id="close_limit_cron" model="ir.cron"> |
||||
|
<field name="name">Check Close Limit</field> |
||||
|
<field name="model_id" ref="model_subscription_package"/> |
||||
|
<field name="state">code</field> |
||||
|
<field name="code">model.close_limit_cron()</field> |
||||
|
<field name='interval_number'>1</field> |
||||
|
<field name='interval_type'>days</field> |
||||
|
</record> |
||||
|
</odoo> |
@ -0,0 +1,12 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Sequence number generation--> |
||||
|
<data noupdate="1"> |
||||
|
<record id="sequence_reference_code" model="ir.sequence"> |
||||
|
<field name="name">Reference Code</field> |
||||
|
<field name="code">sequence.reference.code</field> |
||||
|
<field name="prefix">SUB</field> |
||||
|
<field name="padding">4</field> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
@ -0,0 +1,55 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Email Template: Email for renew subscription --> |
||||
|
<record id="mail_template_subscription_renew" model="mail.template"> |
||||
|
<field name="name">Subscription: Email Renew Alert</field> |
||||
|
<field name="model_id" ref="subscription_package.model_subscription_package"/> |
||||
|
<field name="subject">{{ object.company_id.name }}: Please check the subscription {{ object.name }}</field> |
||||
|
<field name="email_from">{{ object.company_id.email }}</field> |
||||
|
<field name="email_to">{{ object.partner_id.email }}</field> |
||||
|
<field name="auto_delete" eval="True"/> |
||||
|
<field name="lang">{{ object.partner_id.lang }}</field> |
||||
|
<field name="body_html" type="html"> |
||||
|
<div style="background:#F0F0F0;color:#515166;padding:10px 0px;font-family:Arial,Helvetica,sans-serif;font-size:14px;"> |
||||
|
<table style="width:600px;margin:5px auto;"> |
||||
|
<tbody> |
||||
|
<tr> |
||||
|
<td> |
||||
|
<a href="/"> |
||||
|
<img src="/web/binary/company_logo" |
||||
|
style="vertical-align:baseline;max-width:100px;"/> |
||||
|
</a> |
||||
|
</td> |
||||
|
<td style="text-align:right;vertical-align:middle;"> |
||||
|
Subscription Renew Alert |
||||
|
</td> |
||||
|
</tr> |
||||
|
</tbody> |
||||
|
</table> |
||||
|
<table style="width:600px;margin:0px auto;background:white;border:1px solid #e1e1e1;"> |
||||
|
<tbody> |
||||
|
<tr> |
||||
|
<td style="padding:15px 20px 10px 20px;"> |
||||
|
<p>Dear<t t-out=" object.partner_id.name or ''"/>, |
||||
|
</p> |
||||
|
<p>Your subscription Plan |
||||
|
<strong t-out="object.name or ''"/> |
||||
|
is going to Expired on |
||||
|
<strong t-out="object.close_date or ''"/>. |
||||
|
</p> |
||||
|
<p>If you have any concerns about it, please contact your representative at |
||||
|
<t |
||||
|
t-out="object.company_id.name or ''"/> |
||||
|
or reply to this email. |
||||
|
</p> |
||||
|
<p>Kind regards.</p> |
||||
|
</td> |
||||
|
</tr> |
||||
|
<tr> |
||||
|
</tr> |
||||
|
</tbody> |
||||
|
</table> |
||||
|
</div> |
||||
|
</field> |
||||
|
</record> |
||||
|
</odoo> |
@ -0,0 +1,9 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Subscription Pacakge stop close limit --> |
||||
|
<data noupdate="1"> |
||||
|
<record model="subscription.package.stop" id="Close_limit"> |
||||
|
<field name="name">Renewal Limit Exceeded</field> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
@ -0,0 +1,22 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Subscription Package Stages --> |
||||
|
<data noupdate="1"> |
||||
|
<record model="subscription.package.stage" id="draft_stage"> |
||||
|
<field name="name">Draft</field> |
||||
|
<field name="sequence">5</field> |
||||
|
<field name="category">draft</field> |
||||
|
</record> |
||||
|
<record model="subscription.package.stage" id="progress_stage"> |
||||
|
<field name="name">In Progress</field> |
||||
|
<field name="sequence">10</field> |
||||
|
<field name="category">progress</field> |
||||
|
</record> |
||||
|
<record model="subscription.package.stage" id="closed_stage"> |
||||
|
<field name="name">Closed</field> |
||||
|
<field name="sequence">15</field> |
||||
|
<field name="is_fold" eval="True"/> |
||||
|
<field name="category">closed</field> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
@ -0,0 +1,22 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Uom Creation --> |
||||
|
<data noupdate="1"> |
||||
|
<record id="uom_categ_time" model="uom.category"> |
||||
|
<field name="name">Time</field> |
||||
|
</record> |
||||
|
<record id="product_uom_months" model="uom.uom"> |
||||
|
<field name="category_id" ref="uom_categ_time"/> |
||||
|
<field name="name">Months</field> |
||||
|
<field name="factor_inv" eval="1"/> |
||||
|
<field name="uom_type">reference</field> |
||||
|
</record> |
||||
|
<record id="product_uom_years" model="uom.uom"> |
||||
|
<field name="category_id" ref="uom_categ_time"/> |
||||
|
<field name="name">Years</field> |
||||
|
<field name="factor_inv" eval="12"/> |
||||
|
<field name="uom_type">bigger</field> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
||||
|
|
@ -0,0 +1,6 @@ |
|||||
|
## Module <subscription_package> |
||||
|
#### 19.02.2025 |
||||
|
#### Version 18.0.1.0.0 |
||||
|
#### ADD |
||||
|
- Initial commit for Subscription Management |
||||
|
|
@ -0,0 +1,32 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 . import account_move |
||||
|
from . import product_template |
||||
|
from . import recurrence_period |
||||
|
from . import res_partner |
||||
|
from . import sale_order |
||||
|
from . import subscription_package |
||||
|
from . import subscription_package_plan |
||||
|
from . import subscription_package_product_line |
||||
|
from . import subscription_package_stage |
||||
|
from . import subscription_package_stop |
||||
|
from . import sale_order_line |
@ -0,0 +1,46 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 |
||||
|
|
||||
|
|
||||
|
class AccountMove(models.Model): |
||||
|
"""Inherited sale order model""" |
||||
|
_inherit = "account.move" |
||||
|
|
||||
|
is_subscription = fields.Boolean(string='Is Subscription', default=False, |
||||
|
help='Is subscription') |
||||
|
subscription_id = fields.Many2one('subscription.package', |
||||
|
string='Subscription', |
||||
|
help='Choose subscription package') |
||||
|
|
||||
|
@api.model_create_multi |
||||
|
def create(self, vals_list): |
||||
|
""" It displays subscription in account move """ |
||||
|
for rec in vals_list: |
||||
|
so_id = self.env['sale.order'].search( |
||||
|
[('name', '=', rec.get('invoice_origin'))]) |
||||
|
if so_id.is_subscription is True: |
||||
|
so_id.subscription_id.start_date = so_id.subscription_id.next_invoice_date |
||||
|
new_vals_list = [{'is_subscription': True, |
||||
|
'subscription_id': so_id.subscription_id.id}] |
||||
|
vals_list[0].update(new_vals_list[0]) |
||||
|
return super().create(vals_list) |
@ -0,0 +1,36 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 fields, models |
||||
|
|
||||
|
|
||||
|
class ProductTemplate(models.Model): |
||||
|
"""Inherited product template model""" |
||||
|
_inherit = "product.template" |
||||
|
|
||||
|
is_subscription = fields.Boolean(string='Is Subscription', default=False, |
||||
|
help='Indicates whether the product is ' |
||||
|
'associated with a subscription ' |
||||
|
'or not.') |
||||
|
subscription_plan_id = fields.Many2one('subscription.package.plan', |
||||
|
string='Subscription Plan', |
||||
|
help='Select the subscription plan ' |
||||
|
'associated with this record.') |
@ -0,0 +1,41 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 fields, models |
||||
|
|
||||
|
|
||||
|
class RecurrencePeriod(models.Model): |
||||
|
"""This class is used to create new model recurrence period""" |
||||
|
_name = "recurrence.period" |
||||
|
_description = "Recurrence Period " |
||||
|
|
||||
|
name = fields.Char(string="Name", |
||||
|
help='The name of the recurrence period. Enter a ' |
||||
|
'descriptive name for the period.') |
||||
|
duration = fields.Float(string="Duration", |
||||
|
help='The duration associated with this record. ' |
||||
|
'Enter the duration value.') |
||||
|
unit = fields.Selection([('hours', 'hours'), |
||||
|
('days', 'Days'), ('weeks', 'Weeks'), |
||||
|
('months', 'Months'), ('years', 'Years')], |
||||
|
string='Unit', |
||||
|
help='Select the unit of time associated with this ' |
||||
|
'record. Choose from the available options.') |
@ -0,0 +1,44 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 fields, models |
||||
|
|
||||
|
|
||||
|
class ResPartner(models.Model): |
||||
|
"""Inherited res partner model""" |
||||
|
_inherit = 'res.partner' |
||||
|
|
||||
|
is_active_subscription = fields.Boolean(string="Active Subscription", |
||||
|
default=False, |
||||
|
help='Is Subscription is active') |
||||
|
subscription_product_line_ids = fields.One2many( |
||||
|
'subscription.package.product.line', 'res_partner_id', |
||||
|
ondelete='restrict', string='Products Line', |
||||
|
help='Subscription product') |
||||
|
|
||||
|
def _valid_field_parameter(self, field, name): |
||||
|
""" |
||||
|
Validate field parameters, allowing custom handling for 'ondelete' |
||||
|
""" |
||||
|
if name == 'ondelete': |
||||
|
return True |
||||
|
return super(ResPartner, |
||||
|
self)._valid_field_parameter(field, name) |
@ -0,0 +1,126 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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.safe_eval import datetime |
||||
|
|
||||
|
|
||||
|
class SaleOrder(models.Model): |
||||
|
""" This class is used to inherit sale order""" |
||||
|
_inherit = 'sale.order' |
||||
|
|
||||
|
subscription_count = fields.Integer(string='Subscriptions', |
||||
|
compute='_compute_subscription_count', |
||||
|
help='Subscriptions count') |
||||
|
is_subscription = fields.Boolean(string='Is Subscription', default=False, |
||||
|
help='Is subscription') |
||||
|
subscription_id = fields.Many2one('subscription.package', |
||||
|
string='Subscription', |
||||
|
help='Choose the subscription') |
||||
|
sub_reference = fields.Char(string="Sub Reference Code", store=True, |
||||
|
compute="_compute_reference_code", |
||||
|
help='Subscription Reference Code') |
||||
|
|
||||
|
@api.model_create_multi |
||||
|
def create(self, vals_list): |
||||
|
""" It displays subscription in sale order """ |
||||
|
for vals in vals_list: |
||||
|
if vals.get('is_subscription'): |
||||
|
vals.update({ |
||||
|
'is_subscription': True, |
||||
|
'subscription_id': vals.get('subscription_id'), |
||||
|
}) |
||||
|
return super().create(vals) |
||||
|
|
||||
|
@api.depends('subscription_id') |
||||
|
def _compute_reference_code(self): |
||||
|
""" It displays subscription reference code """ |
||||
|
self.sub_reference = self.env['subscription.package'].search( |
||||
|
[('id', '=', int(self.subscription_id.id))]).reference_code |
||||
|
|
||||
|
def action_confirm(self): |
||||
|
""" It Changed the stage, to renew, start date for subscription |
||||
|
package based on sale order confirm """ |
||||
|
|
||||
|
res = super().action_confirm() |
||||
|
sale_order = self.subscription_id.sale_order_id |
||||
|
so_state = self.search([('id', '=', sale_order.id)]).state |
||||
|
if so_state in ['sale', 'done']: |
||||
|
stage = self.env['subscription.package.stage'].search( |
||||
|
[('category', '=', 'progress')], limit=1).id |
||||
|
values = {'stage_id': stage, 'is_to_renew': False, |
||||
|
'start_date': datetime.datetime.today()} |
||||
|
self.subscription_id.write(values) |
||||
|
return res |
||||
|
|
||||
|
@api.depends('subscription_count') |
||||
|
def _compute_subscription_count(self): |
||||
|
"""the compute function the count of |
||||
|
subscriptions associated with the sale order.""" |
||||
|
subscription_count = self.env[ |
||||
|
'subscription.package'].sudo().search_count( |
||||
|
[('sale_order_id', '=', self.id)]) |
||||
|
if subscription_count > 0: |
||||
|
self.subscription_count = subscription_count |
||||
|
else: |
||||
|
self.subscription_count = 0 |
||||
|
|
||||
|
def button_subscription(self): |
||||
|
"""Open the subscription packages associated with the sale order.""" |
||||
|
return { |
||||
|
'name': 'Subscription', |
||||
|
'sale_order_id': False, |
||||
|
'domain': [('sale_order_id', '=', self.id)], |
||||
|
'view_type': 'form', |
||||
|
'res_model': 'subscription.package', |
||||
|
'view_mode': 'list,form', |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'context': { |
||||
|
"create": False |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
def _action_confirm(self): |
||||
|
"""the function used to Confrim the sale order and |
||||
|
create subscriptions for subscription products""" |
||||
|
if self.subscription_count != 1: |
||||
|
if self.order_line: |
||||
|
for line in self.order_line: |
||||
|
if line.product_id.is_subscription: |
||||
|
this_products_line = [] |
||||
|
rec_list = [0, 0, {'product_id': line.product_id.id, |
||||
|
'product_qty': line.product_uom_qty, |
||||
|
'unit_price': line.price_unit}] |
||||
|
this_products_line.append(rec_list) |
||||
|
self.env['subscription.package'].create( |
||||
|
{ |
||||
|
'sale_order_id': self.id, |
||||
|
'reference_code': self.env[ |
||||
|
'ir.sequence'].next_by_code( |
||||
|
'sequence.reference.code'), |
||||
|
'start_date': fields.Date.today(), |
||||
|
'stage_id': self.env.ref( |
||||
|
'subscription_package.draft_stage').id, |
||||
|
'partner_id': self.partner_id.id, |
||||
|
'plan_id': line.product_id.subscription_plan_id.id, |
||||
|
'product_line_ids': this_products_line |
||||
|
}) |
||||
|
return super()._action_confirm() |
@ -0,0 +1,48 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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, models |
||||
|
|
||||
|
|
||||
|
class SaleOrderLine(models.Model): |
||||
|
_inherit = 'sale.order.line' |
||||
|
|
||||
|
@api.depends('qty_invoiced', 'qty_delivered', 'product_uom_qty', 'state') |
||||
|
def _compute_qty_to_invoice(self): |
||||
|
"""Over-write the _compute_qty_to_invoice function |
||||
|
to look weather order is subscriptions""" |
||||
|
for line in self: |
||||
|
if (line.order_id.subscription_id and not |
||||
|
line.order_id.subscription_id.is_closed |
||||
|
and line.order_id.is_subscription): |
||||
|
if line.product_template_id.is_subscription: |
||||
|
line.qty_to_invoice = line.product_uom_qty |
||||
|
else: |
||||
|
if line.state == 'sale' and not line.display_type: |
||||
|
if line.product_id.invoice_policy == 'order': |
||||
|
line.qty_to_invoice = ( |
||||
|
line.product_uom_qty - line.qty_invoiced) |
||||
|
else: |
||||
|
line.qty_to_invoice = ( |
||||
|
line.qty_delivered - line.qty_invoiced) |
||||
|
else: |
||||
|
line.qty_to_invoice = 0 |
@ -0,0 +1,471 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 dateutil.relativedelta import relativedelta |
||||
|
from odoo import api, fields, models, SUPERUSER_ID, _ |
||||
|
from odoo.exceptions import UserError |
||||
|
|
||||
|
|
||||
|
class SubscriptionPackage(models.Model): |
||||
|
"""Subscription Package Model""" |
||||
|
_name = 'subscription.package' |
||||
|
_description = 'Subscription Package' |
||||
|
_rec_name = 'name' |
||||
|
_inherit = ['mail.thread', 'mail.activity.mixin'] |
||||
|
|
||||
|
@api.model |
||||
|
def _read_group_stage_ids(self, stages, domain): |
||||
|
""" Read all the stages and display it in the kanban view, |
||||
|
even if it is empty.""" |
||||
|
stages_ids = stages.sudo()._search([], order=stages._order) |
||||
|
return stages.browse(stages_ids) |
||||
|
|
||||
|
def _default_stage_id(self): |
||||
|
"""Setting default stage""" |
||||
|
rec = self.env['subscription.package.stage'].search([], limit=1, |
||||
|
order='sequence ASC') |
||||
|
return rec.id if rec else None |
||||
|
|
||||
|
name = fields.Char(string='Name', default="New", compute='_compute_name', |
||||
|
store=True, required=True, |
||||
|
help='Choose the name for the subscription package.') |
||||
|
partner_id = fields.Many2one('res.partner', string='Customer', |
||||
|
help='Select the customer associated with ' |
||||
|
'this record.') |
||||
|
partner_invoice_id = fields.Many2one('res.partner', |
||||
|
help='Select the invoice address ' |
||||
|
'associated with this record.', |
||||
|
string='Invoice Address', |
||||
|
related='partner_id') |
||||
|
partner_shipping_id = fields.Many2one('res.partner', |
||||
|
help="Add shipping/service address", |
||||
|
string='Shipping/Service Address', |
||||
|
related='partner_id') |
||||
|
plan_id = fields.Many2one('subscription.package.plan', |
||||
|
string='Subscription Plan', |
||||
|
help="Choose the subscription package plan") |
||||
|
start_date = fields.Date(string='Period Start Date', |
||||
|
help='Add the period start date', |
||||
|
ondelete='restrict') |
||||
|
date_started = fields.Date(string='Subsciption Start date', |
||||
|
help='Add the Subscription package start date', |
||||
|
ondelete='restrict', readonly=True) |
||||
|
next_invoice_date = fields.Date(string='Next Invoice Date', |
||||
|
store=True, help='Add next invoice date', |
||||
|
compute="_compute_next_invoice_date", |
||||
|
inverse="_inverse_next_invoice_date") |
||||
|
company_id = fields.Many2one('res.company', string='Company', |
||||
|
help='Select the company', |
||||
|
default=lambda self: self.env.company, |
||||
|
required=True) |
||||
|
user_id = fields.Many2one('res.users', string='Sales Person', |
||||
|
help='Add the Sales person', |
||||
|
default=lambda self: self.env.user) |
||||
|
sale_order_id = fields.Many2one('sale.order', string="Sale Order", |
||||
|
help='Select the sale order', copy=False) |
||||
|
is_to_renew = fields.Boolean(string='To Renew', copy=True, |
||||
|
help='Is subscription package is renew') |
||||
|
tag_ids = fields.Many2many('account.account.tag', string='Tags', |
||||
|
help='Add the tags') |
||||
|
stage_id = fields.Many2one('subscription.package.stage', string='Stage', |
||||
|
default=lambda self: self._default_stage_id(), |
||||
|
index=True, |
||||
|
group_expand='_read_group_stage_ids', |
||||
|
help='Subscription Package stage', copy=False) |
||||
|
invoice_count = fields.Integer(string='Invoices', |
||||
|
help='Subscription package invoice count', |
||||
|
compute='_compute_invoice_count') |
||||
|
so_count = fields.Integer(string='Sales', |
||||
|
help='subscription package sales count', |
||||
|
compute='_compute_sale_count') |
||||
|
description = fields.Text(string='Description', |
||||
|
help='Subscription package description') |
||||
|
analytic_account_id = fields.Many2one('account.analytic.account', |
||||
|
help='Choose the analytic account', |
||||
|
string='Analytic Account') |
||||
|
product_line_ids = fields.One2many('subscription.package.product.line', |
||||
|
'subscription_id', ondelete='restrict', |
||||
|
string='Products Line', |
||||
|
help='Subscription package product line') |
||||
|
currency_id = fields.Many2one('res.currency', string='Currency', |
||||
|
readonly=True, default=lambda |
||||
|
self: self.env.company.currency_id, help='Add Currency') |
||||
|
current_stage = fields.Char(string='Current Stage', default='Draft', |
||||
|
help='Current stage of the ' |
||||
|
'subscription package. ' |
||||
|
'This field is computed based on ' |
||||
|
'the associated stage_id.', |
||||
|
store=True, compute='_compute_current_stage') |
||||
|
reference_code = fields.Char(string='Reference', |
||||
|
help='This field represents the ' |
||||
|
'reference code associated ' |
||||
|
'with the record.') |
||||
|
is_closed = fields.Boolean(string="Closed", default=False, |
||||
|
help='Is Closed') |
||||
|
close_reason_id = fields.Many2one('subscription.package.stop', |
||||
|
help='The reason for c' |
||||
|
'losing the subscription package.', |
||||
|
string='Close Reason') |
||||
|
closed_by = fields.Many2one('res.users', string='Closed By', |
||||
|
help="The user responsible " |
||||
|
"for closing the record") |
||||
|
close_date = fields.Date(string='Closed on', |
||||
|
help="The date on which the record was closed") |
||||
|
stage_category = fields.Selection(related='stage_id.category', |
||||
|
help="The category associated with " |
||||
|
"the current stage of the record. ", |
||||
|
store=True) |
||||
|
invoice_mode = fields.Selection(related="plan_id.invoice_mode", |
||||
|
help="The invoice mode " |
||||
|
"associated with the plan.") |
||||
|
total_recurring_price = fields.Float(string='Untaxed Amount', |
||||
|
help="The total recurring " |
||||
|
"price excluding taxes.", |
||||
|
compute='_compute_total_recurring_price', |
||||
|
store=True) |
||||
|
tax_total = fields.Float("Taxes", readonly=True, |
||||
|
help="The total amount of " |
||||
|
"taxes associated with the record") |
||||
|
total_with_tax = fields.Monetary("Total Recurring Price", readonly=True, |
||||
|
help="The total recurring " |
||||
|
"price including taxes") |
||||
|
recurrence_period_id = fields.Many2one("recurrence.period", |
||||
|
string="Recurrence Period") |
||||
|
sale_order_count = fields.Integer(string='Sale Order Count', |
||||
|
help="The count of associated " |
||||
|
"sale orders for this record.") |
||||
|
|
||||
|
def _valid_field_parameter(self, field, name): |
||||
|
"""Check the validity of a field parameter for a specific field.""" |
||||
|
if name == 'ondelete': |
||||
|
return True |
||||
|
return super(SubscriptionPackage, |
||||
|
self)._valid_field_parameter(field, name) |
||||
|
|
||||
|
@api.depends('invoice_count') |
||||
|
def _compute_invoice_count(self): |
||||
|
""" Calculate Invoice count based on subscription package """ |
||||
|
sale_id = self.env['sale.order'].search( |
||||
|
[('id', '=', self.sale_order_id.id)]) |
||||
|
invoices = sale_id.order_line.invoice_lines.move_id.filtered( |
||||
|
lambda r: r.move_type in ('out_invoice', 'out_refund')) |
||||
|
invoices.write({'subscription_id': self.id}) |
||||
|
invoice_count = self.env['account.move'].search_count( |
||||
|
[('subscription_id', '=', self.id)]) |
||||
|
if invoice_count > 0: |
||||
|
self.invoice_count = invoice_count |
||||
|
else: |
||||
|
self.invoice_count = 0 |
||||
|
|
||||
|
@api.depends('so_count') |
||||
|
def _compute_sale_count(self): |
||||
|
""" Calculate sale order count based on subscription package """ |
||||
|
self.so_count = self.env['sale.order'].search_count( |
||||
|
[('id', '=', self.sale_order_id.id)]) |
||||
|
|
||||
|
@api.depends('stage_id') |
||||
|
def _compute_current_stage(self): |
||||
|
""" It displays current stage for subscription package """ |
||||
|
for rec in self: |
||||
|
rec.current_stage = rec.env['subscription.package.stage'].search( |
||||
|
[('id', '=', rec.stage_id.id)]).category |
||||
|
|
||||
|
@api.depends('start_date') |
||||
|
def _compute_next_invoice_date(self): |
||||
|
"""The compute function is the next invoice date for subscription |
||||
|
packages based on the start date and renewal time.""" |
||||
|
for sub in self.env['subscription.package'].search([]): |
||||
|
if sub.start_date: |
||||
|
sub.next_invoice_date = sub.start_date + relativedelta( |
||||
|
days=sub.plan_id.renewal_time) |
||||
|
|
||||
|
def _inverse_next_invoice_date(self): |
||||
|
"""Inverse function for next invoice date""" |
||||
|
for sub in self.env['subscription.package'].search([]): |
||||
|
if sub.start_date: |
||||
|
return |
||||
|
|
||||
|
def button_invoice_count(self): |
||||
|
""" It displays invoice based on subscription package """ |
||||
|
return { |
||||
|
'name': 'Invoices', |
||||
|
'domain': [('subscription_id', '=', self.id)], |
||||
|
'view_type': 'form', |
||||
|
'res_model': 'account.move', |
||||
|
'view_mode': 'list,form', |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'context': { |
||||
|
"create": False |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
def button_sale_count(self): |
||||
|
""" It displays sale order based on subscription package """ |
||||
|
return { |
||||
|
'name': 'Products', |
||||
|
'domain': [('id', '=', self.sale_order_id.id)], |
||||
|
'view_type': 'form', |
||||
|
'res_model': 'sale.order', |
||||
|
'view_mode': 'list,form', |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'context': { |
||||
|
"create": False |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
def button_close(self): |
||||
|
""" Button for subscription close wizard """ |
||||
|
return { |
||||
|
'name': "Subscription Close Reason", |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'view_type': 'form', |
||||
|
'view_mode': 'form', |
||||
|
'res_model': 'subscription.close', |
||||
|
'target': 'new' |
||||
|
} |
||||
|
|
||||
|
def button_start_date(self): |
||||
|
"""Button to start subscription package""" |
||||
|
stage_id = (self.env['subscription.package.stage'].search([ |
||||
|
('category', '=', 'progress')], limit=1).id) |
||||
|
for rec in self: |
||||
|
if len(rec.env['subscription.package.stage'].search( |
||||
|
[('category', '=', 'draft')])) > 1: |
||||
|
raise UserError( |
||||
|
_('More than one stage is having category "Draft". ' |
||||
|
'Please change category of stage to "In Progress", ' |
||||
|
'only one stage is allowed to have category "Draft"')) |
||||
|
else: |
||||
|
if not rec.product_line_ids: |
||||
|
raise UserError("Empty order lines !! Please add the " |
||||
|
"subscription product.") |
||||
|
else: |
||||
|
if rec.sale_order_id: |
||||
|
rec.sale_order_id.write({'subscription_id': rec.id, |
||||
|
'is_subscription': True}) |
||||
|
for line in rec.sale_order_id.order_line.filtered( |
||||
|
lambda x: x.product_template_id.is_subscription == True): |
||||
|
line.qty_to_invoice = line.product_uom_qty |
||||
|
rec.write( |
||||
|
{'stage_id': stage_id, |
||||
|
'date_started': fields.Date.today(), |
||||
|
'start_date': fields.Date.today()}) |
||||
|
|
||||
|
def button_sale_order(self): |
||||
|
"""Button to create sale order""" |
||||
|
this_products_line = [] |
||||
|
for rec in self.product_line_ids: |
||||
|
rec_list = [0, 0, {'product_id': rec.product_id.id, |
||||
|
'product_uom_qty': rec.product_qty, |
||||
|
'discount': rec.discount}] |
||||
|
this_products_line.append(rec_list) |
||||
|
orders = self.env['sale.order'].search( |
||||
|
[('id', '=', self.sale_order_count), |
||||
|
('invoice_status', '=', 'no')]) |
||||
|
if orders: |
||||
|
for order in orders: |
||||
|
order.action_confirm() |
||||
|
so_id = self.env['sale.order'].create({ |
||||
|
'id': self.sale_order_count, |
||||
|
'partner_id': self.partner_id.id, |
||||
|
'partner_invoice_id': self.partner_id.id, |
||||
|
'partner_shipping_id': self.partner_id.id, |
||||
|
'is_subscription': True, |
||||
|
'subscription_id': self.id, |
||||
|
'order_line': this_products_line |
||||
|
}) |
||||
|
self.sale_order_id = so_id |
||||
|
return { |
||||
|
'name': _('Sales Orders'), |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'res_model': 'sale.order', |
||||
|
'domain': [('id', '=', so_id.id)], |
||||
|
'view_mode': 'list,form', |
||||
|
'context': { |
||||
|
"create": False |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
@api.model_create_multi |
||||
|
def create(self, vals_list): |
||||
|
"""It displays subscription product in partner and generate sequence""" |
||||
|
for vals in vals_list: |
||||
|
partner = self.env['res.partner'].search( |
||||
|
[('id', '=', vals.get('partner_id'))]) |
||||
|
partner.is_active_subscription = True |
||||
|
if vals.get('reference_code', 'New') is False: |
||||
|
vals['reference_code'] = self.env['ir.sequence'].next_by_code( |
||||
|
'sequence.reference.code') or 'New' |
||||
|
create_id = super().create(vals) |
||||
|
return create_id |
||||
|
|
||||
|
@api.depends('reference_code') |
||||
|
def _compute_name(self): |
||||
|
"""It displays record name as combination of short code, reference |
||||
|
code and partner name """ |
||||
|
for rec in self: |
||||
|
plan_id = self.env['subscription.package.plan'].search( |
||||
|
[('id', '=', rec.plan_id.id)]) |
||||
|
if plan_id.short_code and rec.reference_code: |
||||
|
rec.name = plan_id.short_code + '/' + rec.reference_code + '-' + rec.partner_id.name |
||||
|
|
||||
|
def set_close(self): |
||||
|
""" Button to close subscription package """ |
||||
|
stage = self.env['subscription.package.stage'].search( |
||||
|
[('category', '=', 'closed')], limit=1).id |
||||
|
for sub in self: |
||||
|
values = {'stage_id': stage, 'is_to_renew': False} |
||||
|
sub.write(values) |
||||
|
return True |
||||
|
|
||||
|
def send_renew_alert_mail(self, today, renew_date, sub_id): |
||||
|
"""The function is used to send a renewal alert email and mark the |
||||
|
subscription for renewal if today is the renewal date.""" |
||||
|
if today == renew_date: |
||||
|
self.env.ref( |
||||
|
'subscription_package' |
||||
|
'.mail_template_subscription_renew').send_mail( |
||||
|
sub_id, force_send=True) |
||||
|
subscription = self.env['subscription.package'].browse(sub_id) |
||||
|
subscription.write({'is_to_renew': True}) |
||||
|
return True |
||||
|
else: |
||||
|
return False |
||||
|
|
||||
|
def find_renew_date(self, next_invoice, date_started, end): |
||||
|
"""The function is used to calculate the renewal date, end date, |
||||
|
and close date based on subscription details.""" |
||||
|
if end == 0: |
||||
|
end_date = next_invoice |
||||
|
difference = (next_invoice - date_started).days / 10 |
||||
|
renew_date = next_invoice - relativedelta( |
||||
|
days=difference) |
||||
|
close_date = next_invoice |
||||
|
else: |
||||
|
end_date = fields.Date.add(date_started, |
||||
|
days=end) |
||||
|
close = date_started + relativedelta(days=end) |
||||
|
difference = (close - date_started).days / 10 |
||||
|
renew_date = close - relativedelta( |
||||
|
days=difference) |
||||
|
close_date = close |
||||
|
|
||||
|
data = {'renew_date': renew_date, |
||||
|
'end_date': end_date, |
||||
|
'close_date': close_date} |
||||
|
return data |
||||
|
|
||||
|
def close_limit_cron(self): |
||||
|
""" It Checks renew date, close date. It will send mail when renew |
||||
|
date and also generates invoices based on the plan. It wil close the |
||||
|
subscription automatically if renewal limit is exceeded""" |
||||
|
pending_subscriptions = self.env['subscription.package'].search( |
||||
|
[('stage_category', '=', 'progress')]) |
||||
|
today_date = fields.Date.today() |
||||
|
pending_subscription = False |
||||
|
for pending_subscription in pending_subscriptions: |
||||
|
get_dates = self.find_renew_date( |
||||
|
pending_subscription.next_invoice_date, |
||||
|
pending_subscription.date_started, |
||||
|
pending_subscription.plan_id.days_to_end) |
||||
|
renew_date = get_dates['renew_date'] |
||||
|
end_date = get_dates['end_date'] |
||||
|
pending_subscription.close_date = get_dates['close_date'] |
||||
|
if today_date == pending_subscription.next_invoice_date: |
||||
|
if pending_subscription.plan_id.invoice_mode == 'draft_invoice': |
||||
|
this_products_line = [] |
||||
|
for rec in pending_subscription.product_line_ids: |
||||
|
rec_list = [0, 0, {'product_id': rec.product_id.id, |
||||
|
'quantity': rec.product_qty, |
||||
|
'price_unit': rec.unit_price, |
||||
|
'discount': rec.discount, |
||||
|
'tax_ids': rec.tax_ids |
||||
|
}] |
||||
|
this_products_line.append(rec_list) |
||||
|
self.env['account.move'].create( |
||||
|
{ |
||||
|
'move_type': 'out_invoice', |
||||
|
'invoice_date_due': today_date, |
||||
|
'invoice_payment_term_id': False, |
||||
|
'invoice_date': today_date, |
||||
|
'state': 'draft', |
||||
|
'subscription_id': pending_subscription.id, |
||||
|
'partner_id': pending_subscription.partner_invoice_id.id, |
||||
|
'currency_id': pending_subscription.partner_invoice_id.currency_id.id, |
||||
|
'invoice_line_ids': this_products_line |
||||
|
}) |
||||
|
pending_subscription.write({ |
||||
|
'is_to_renew': False, |
||||
|
'start_date': pending_subscription.next_invoice_date}) |
||||
|
new_date = self.find_renew_date( |
||||
|
pending_subscription.next_invoice_date, |
||||
|
pending_subscription.date_started, |
||||
|
pending_subscription.plan_id.days_to_end) |
||||
|
pending_subscription.write( |
||||
|
{'close_date': new_date['close_date']}) |
||||
|
self.send_renew_alert_mail(today_date, |
||||
|
new_date['renew_date'], |
||||
|
pending_subscription.id) |
||||
|
|
||||
|
if (today_date == end_date) and ( |
||||
|
pending_subscription.plan_id.limit_choice != 'manual'): |
||||
|
display_msg = ("<h5><i>The renewal limit has been exceeded " |
||||
|
"today for this subscription based on the " |
||||
|
"current subscription plan.</i></h5>") |
||||
|
pending_subscription.message_post(body=display_msg) |
||||
|
pending_subscription.is_closed = True |
||||
|
reason = (self.env['subscription.package.stop'].search([ |
||||
|
('name', '=', 'Renewal Limit Exceeded')]).id) |
||||
|
pending_subscription.close_reason_id = reason |
||||
|
pending_subscription.closed_by = self.user_id |
||||
|
pending_subscription.close_date = fields.Date.today() |
||||
|
stage = (self.env['subscription.package.stage'].search([ |
||||
|
('category', '=', 'closed')]).id) |
||||
|
values = {'stage_id': stage, 'is_to_renew': False, |
||||
|
'next_invoice_date': False} |
||||
|
pending_subscription.write(values) |
||||
|
|
||||
|
self.send_renew_alert_mail(today_date, renew_date, |
||||
|
pending_subscription.id) |
||||
|
|
||||
|
return dict(pending=pending_subscription) |
||||
|
|
||||
|
@api.depends('product_line_ids.total_amount', |
||||
|
'product_line_ids.price_total', 'product_line_ids.tax_ids') |
||||
|
def _compute_total_recurring_price(self): |
||||
|
""" The compute function used to calculate recurring price """ |
||||
|
for record in self: |
||||
|
total_recurring = 0 |
||||
|
total_tax = 0.0 |
||||
|
for line in record.product_line_ids: |
||||
|
if line.total_amount != line.price_total: |
||||
|
line_tax = line.price_total - line.total_amount |
||||
|
total_tax += line_tax |
||||
|
total_recurring += line.total_amount |
||||
|
record['total_recurring_price'] = total_recurring |
||||
|
record['tax_total'] = total_tax |
||||
|
total_with_tax = total_recurring + total_tax |
||||
|
record['total_with_tax'] = total_with_tax |
||||
|
|
||||
|
def action_renew(self): |
||||
|
""" The function is used to perform the renewal |
||||
|
action for the subscription package.""" |
||||
|
return self.button_sale_order() |
@ -0,0 +1,156 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 |
||||
|
|
||||
|
|
||||
|
class SubscriptionPackagePlan(models.Model): |
||||
|
_name = 'subscription.package.plan' |
||||
|
_description = 'Subscription Package Plan' |
||||
|
|
||||
|
name = fields.Char(string='Plan Name', required=True, |
||||
|
help='The name of the subscription plan.') |
||||
|
renewal_value = fields.Char(string='Renewal', |
||||
|
help='A descriptive value indicating the ' |
||||
|
'renewal status or details for the ' |
||||
|
'subscription plan.') |
||||
|
renewal_period = fields.Selection([('days', 'Day(s)'), |
||||
|
('weeks', 'Week(s)'), |
||||
|
('months', 'Month(s)'), |
||||
|
('years', 'Year(s)')], |
||||
|
default='months', |
||||
|
help='Select the unit of time for the ' |
||||
|
'renewal period of the ' |
||||
|
'subscription plan.') |
||||
|
renewal_time = fields.Integer(string='Renewal Time Interval', |
||||
|
compute='_compute_renewal_time', |
||||
|
store=True, |
||||
|
help='The computed renewal time interval ' |
||||
|
'for the subscription plan, based on ' |
||||
|
'the selected renewal period.') |
||||
|
limit_choice = fields.Selection([('ones', 'Ones'), |
||||
|
('manual', 'Until Closed Manually'), |
||||
|
('custom', 'Custom')], |
||||
|
default='ones', |
||||
|
help='Select the limit choice for the ' |
||||
|
'subscription plan, specifying how ' |
||||
|
'long it will be active.') |
||||
|
limit_count = fields.Integer(string='Custom Renewal Limit', |
||||
|
help='Specify the custom renewal limit for ' |
||||
|
'the subscription plan. This field is ' |
||||
|
'relevant when the "Limit Choice" is ' |
||||
|
'set to "Custom".') |
||||
|
days_to_end = fields.Integer(string='Days End', readonly=True, |
||||
|
compute='_compute_days_to_end', store=True, |
||||
|
help="Subscription ending date") |
||||
|
invoice_mode = fields.Selection([('manual', 'Manually'), |
||||
|
('draft_invoice', 'Draft')], |
||||
|
default='draft_invoice', |
||||
|
help='Select the invoice mode for the ' |
||||
|
'subscription plan, specifying ' |
||||
|
'whether invoices are generated ' |
||||
|
'manually or in draft state.') |
||||
|
journal_id = fields.Many2one('account.journal', string='Journal', |
||||
|
domain="[('type', '=', 'sale')]") |
||||
|
company_id = fields.Many2one('res.company', string='Company', store=True, |
||||
|
default=lambda self: self.env.company) |
||||
|
short_code = fields.Char(string='Short Code') |
||||
|
terms_and_conditions = fields.Text(string='Terms and Conditions') |
||||
|
product_count = fields.Integer(string='Products', |
||||
|
compute='_compute_product_count') |
||||
|
subscription_count = fields.Integer(string='Subscriptions', |
||||
|
compute='_compute_subscription_count') |
||||
|
|
||||
|
@api.depends('product_count') |
||||
|
def _compute_product_count(self): |
||||
|
""" Calculate product count based on subscription plan """ |
||||
|
self.product_count = self.env['product.product'].search_count( |
||||
|
[('subscription_plan_id', '=', self.id)]) |
||||
|
|
||||
|
@api.depends('subscription_count') |
||||
|
def _compute_subscription_count(self): |
||||
|
""" Calculate subscription count based on subscription plan """ |
||||
|
self.subscription_count = self.env[ |
||||
|
'subscription.package'].search_count([('plan_id', '=', self.id)]) |
||||
|
|
||||
|
@api.depends('renewal_value', 'renewal_period') |
||||
|
def _compute_renewal_time(self): |
||||
|
""" This method calculate renewal time based on renewal value """ |
||||
|
for rec in self: |
||||
|
if int(rec.renewal_value) == 0 or int(rec.renewal_value) < 0: |
||||
|
rec.renewal_value = 1 |
||||
|
if rec.renewal_period == 'days': |
||||
|
rec.renewal_time = int(rec.renewal_value) |
||||
|
elif rec.renewal_period == 'weeks': |
||||
|
rec.renewal_time = int(rec.renewal_value) * 7 |
||||
|
elif rec.renewal_period == 'months': |
||||
|
rec.renewal_time = int(rec.renewal_value) * 28 |
||||
|
elif rec.renewal_period == 'years': |
||||
|
rec.renewal_time = int(rec.renewal_value) * 364 |
||||
|
if rec.name: |
||||
|
rec.short_code = str(rec.name[0:3]).upper() |
||||
|
|
||||
|
@api.depends('renewal_time', 'limit_count') |
||||
|
def _compute_days_to_end(self): |
||||
|
""" This method calculate days to end for subscription plan based on |
||||
|
limit count """ |
||||
|
for rec in self: |
||||
|
if rec.limit_count == 0 or rec.limit_count < 0: |
||||
|
rec.limit_count = 1 |
||||
|
if rec.limit_choice == 'ones': |
||||
|
rec.days_to_end = rec.renewal_time |
||||
|
if rec.limit_choice == 'manual': |
||||
|
rec.days_to_end = False |
||||
|
if rec.limit_choice == 'custom': |
||||
|
rec.days_to_end = rec.renewal_time * rec.limit_count |
||||
|
|
||||
|
def button_product_count(self): |
||||
|
""" It displays products based on subscription plan """ |
||||
|
return { |
||||
|
'name': 'Products', |
||||
|
'res_model': 'product.product', |
||||
|
'domain': [('subscription_plan_id', '=', self.id)], |
||||
|
'view_type': 'form', |
||||
|
'view_mode': 'list,form', |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'context': { |
||||
|
'default_is_subscription': True, |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
def button_sub_count(self): |
||||
|
""" It displays subscriptions based on subscription plan """ |
||||
|
return { |
||||
|
'name': 'Subscriptions', |
||||
|
'domain': [('plan_id', '=', self.id)], |
||||
|
'view_type': 'form', |
||||
|
'res_model': 'subscription.package', |
||||
|
'view_mode': 'list,form', |
||||
|
'type': 'ir.actions.act_window', |
||||
|
} |
||||
|
|
||||
|
def name_get(self): |
||||
|
""" It displays record name as combination of short code and |
||||
|
plan name """ |
||||
|
res = [] |
||||
|
for rec in self: |
||||
|
res.append((rec.id, '%s - %s' % (rec.short_code, rec.name))) |
||||
|
return res |
@ -0,0 +1,101 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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, models, fields |
||||
|
|
||||
|
|
||||
|
class SubscriptionPackageProductLine(models.Model): |
||||
|
"""Subscription Package Product Line Model""" |
||||
|
_name = 'subscription.package.product.line' |
||||
|
_description = 'Subscription Package Product Lines' |
||||
|
|
||||
|
subscription_id = fields.Many2one('subscription.package', store=True, |
||||
|
string='Subscription', |
||||
|
help='Choose Subscription Package') |
||||
|
company_id = fields.Many2one('res.company', string='Company', store=True, |
||||
|
related='subscription_id.company_id') |
||||
|
create_date = fields.Datetime(string='Create date', store=True, |
||||
|
default=fields.Datetime.now, |
||||
|
help='Add Create Date') |
||||
|
user_id = fields.Many2one('res.users', string='Salesperson', store=True, |
||||
|
related='subscription_id.user_id', |
||||
|
help='Add Salesperson') |
||||
|
product_id = fields.Many2one('product.product', string='Product', |
||||
|
store=True, ondelete='restrict', |
||||
|
domain=[('is_subscription', '=', True)], |
||||
|
help='Choose Product') |
||||
|
product_qty = fields.Float(string='Quantity', store=True, default=1.0, |
||||
|
help='Add Product Quantity') |
||||
|
product_uom_id = fields.Many2one('uom.uom', string='UoM', store=True, |
||||
|
related='product_id.uom_id', |
||||
|
ondelete='restrict', |
||||
|
help='Add Product UOM') |
||||
|
uom_catg_id = fields.Many2one('uom.category', string='UoM Category', |
||||
|
store=True, |
||||
|
related='product_id.uom_id.category_id', |
||||
|
help='Choose Product Uom quantity') |
||||
|
unit_price = fields.Float(string='Unit Price', store=True, readonly=False, |
||||
|
related='product_id.list_price', |
||||
|
help='Add Product Unit Price') |
||||
|
discount = fields.Float(string="Discount (%)", help='Add Discount') |
||||
|
tax_ids = fields.Many2many('account.tax', string="Taxes", |
||||
|
ondelete='restrict', |
||||
|
related='product_id.taxes_id', readonly=False, |
||||
|
help='Add Taxes') |
||||
|
price_total = fields.Monetary(store=True, readonly=True, |
||||
|
help='Add Product Price Total') |
||||
|
price_tax = fields.Monetary(store=True, readonly=True, string='Price Tax', |
||||
|
help='Add Price Tax') |
||||
|
currency_id = fields.Many2one('res.currency', string='Currency', |
||||
|
store=True, help='Add Subscription Currency', |
||||
|
related='subscription_id.currency_id') |
||||
|
total_amount = fields.Monetary(string='Subtotal', store=True, |
||||
|
help='Add Total Amount', |
||||
|
compute='_compute_total_amount') |
||||
|
sequence = fields.Integer('Sequence', help="Determine the display order", |
||||
|
index=True) |
||||
|
res_partner_id = fields.Many2one('res.partner', string='Partner', |
||||
|
store=True, help='Choose the Partner', |
||||
|
related='subscription_id.partner_id') |
||||
|
|
||||
|
@api.depends('product_qty', 'unit_price', 'discount', 'tax_ids', |
||||
|
'currency_id') |
||||
|
def _compute_total_amount(self): |
||||
|
""" Calculate subtotal amount of product line """ |
||||
|
for line in self: |
||||
|
price = line.unit_price * (1 - (line.discount or 0.0) / 100.0) |
||||
|
taxes = line.tax_ids._origin.compute_all(price, |
||||
|
line.subscription_id._origin.currency_id, |
||||
|
line.product_qty, |
||||
|
product=line.product_id, |
||||
|
partner=line.subscription_id._origin.partner_id) |
||||
|
line.write({ |
||||
|
'price_tax': sum( |
||||
|
t.get('amount', 0.0) for t in taxes.get('taxes', [])), |
||||
|
'price_total': taxes['total_included'], |
||||
|
'total_amount': taxes['total_excluded'], |
||||
|
}) |
||||
|
|
||||
|
def _valid_field_parameter(self, field, name): |
||||
|
if name == 'ondelete': |
||||
|
return True |
||||
|
return super(SubscriptionPackageProductLine, |
||||
|
self)._valid_field_parameter(field, name) |
@ -0,0 +1,46 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 models, fields |
||||
|
|
||||
|
|
||||
|
class SubscriptionPackageStage(models.Model): |
||||
|
_name = "subscription.package.stage" |
||||
|
_description = "Subscription Package Stages" |
||||
|
_rec_name = 'name' |
||||
|
|
||||
|
name = fields.Char(string='Stage Name', required=True, |
||||
|
help=' Enter a descriptive name for the stage') |
||||
|
sequence = fields.Integer('Sequence', help="Determine the display order", |
||||
|
index=True) |
||||
|
condition = fields.Text(string='Conditions', |
||||
|
help='Use this field to provide detailed ' |
||||
|
'information about the conditions') |
||||
|
is_fold = fields.Boolean(string='Folded in Kanban', |
||||
|
help="This stage is folded in the kanban view " |
||||
|
"when there are no records in that stage " |
||||
|
"to display.") |
||||
|
category = fields.Selection([('draft', 'Draft'), |
||||
|
('progress', 'In Progress'), |
||||
|
('closed', 'Closed')], |
||||
|
readonly=False, default='draft', |
||||
|
help='Choose the appropriate category from' |
||||
|
' the available options.') |
@ -0,0 +1,34 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 models, fields |
||||
|
|
||||
|
|
||||
|
class SubscriptionPackageStop(models.Model): |
||||
|
_name = "subscription.package.stop" |
||||
|
_description = "Subscription Package Stop Reason" |
||||
|
_order = 'sequence' |
||||
|
|
||||
|
sequence = fields.Integer(help="Determine the display order", index=True, |
||||
|
string='Sequence') |
||||
|
name = fields.Char(string='Reason', required=True, |
||||
|
help='Enter the reason for stopping the ' |
||||
|
'subscription package.') |
@ -0,0 +1,22 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 . import subscription_report |
@ -0,0 +1,71 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: SREERAG PM (<https://www.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 models, fields |
||||
|
from odoo import tools |
||||
|
|
||||
|
|
||||
|
class SubscriptionReport(models.Model): |
||||
|
_name = "subscription.report" |
||||
|
_description = "Subscription Analysis" |
||||
|
_auto = False |
||||
|
|
||||
|
total_recurring_price = fields.Float('Recurring Price', readonly=True, |
||||
|
help='Total recurring price ' |
||||
|
'associated with this ' |
||||
|
'subscription analysis.') |
||||
|
quantity = fields.Float('Quantity', readonly=True, |
||||
|
help='The quantity associated with this ' |
||||
|
'subscription analysis.') |
||||
|
user_id = fields.Many2one('res.users', 'Salesperson', readonly=True, |
||||
|
help='The salesperson associated with' |
||||
|
' this record.') |
||||
|
plan_id = fields.Many2one('subscription.package.plan', |
||||
|
'Subscription Template', readonly=True, |
||||
|
help='The subscription template ' |
||||
|
'associated with this record.') |
||||
|
|
||||
|
def _query(self): |
||||
|
select_ = """ |
||||
|
SELECT min(sl.id) as id, |
||||
|
sl.product_qty as quantity, |
||||
|
sub.total_recurring_price as total_recurring_price, |
||||
|
sub.user_id as user_id, |
||||
|
sub.plan_id as plan_id, |
||||
|
sub.name as name |
||||
|
""" |
||||
|
from_ = """ |
||||
|
subscription_package_product_line sl |
||||
|
join subscription_package sub on (sl.subscription_id = sub.id) |
||||
|
""" |
||||
|
groupby_ = """ |
||||
|
GROUP BY sl.product_qty, |
||||
|
sub.total_recurring_price, |
||||
|
sub.user_id, |
||||
|
sub.plan_id, |
||||
|
sub.name |
||||
|
""" |
||||
|
return '%s FROM ( %s ) %s' % (select_, from_, groupby_) |
||||
|
|
||||
|
def init(self): |
||||
|
tools.drop_view_if_exists(self.env.cr, self._table) |
||||
|
self.env.cr.execute("""CREATE or REPLACE VIEW %s as (%s)""" % ( |
||||
|
self._table, self._query())) |
@ -0,0 +1,36 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Subscription report pivot view --> |
||||
|
<record id="subscription_report_view_pivot" model="ir.ui.view"> |
||||
|
<field name="name">Subscription Report Pivot</field> |
||||
|
<field name="model">subscription.report</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<pivot string="Subscription Analysis" sample="1"> |
||||
|
<field name="plan_id" type="row"/> |
||||
|
<field name="user_id" type="col"/> |
||||
|
</pivot> |
||||
|
</field> |
||||
|
</record> |
||||
|
<!-- Subscription report graph view --> |
||||
|
<record id="subscription_report_view_graph" model="ir.ui.view"> |
||||
|
<field name="name">Subscription Report Graph</field> |
||||
|
<field name="model">subscription.report</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<graph string="Subscription Analysis" sample="1"> |
||||
|
<field name="plan_id"/> |
||||
|
</graph> |
||||
|
</field> |
||||
|
</record> |
||||
|
<!-- Subscription report action --> |
||||
|
<record id="subscription_report_action" model="ir.actions.act_window"> |
||||
|
<field name="name">Subscriptions Report</field> |
||||
|
<field name="res_model">subscription.report</field> |
||||
|
<field name="view_mode">pivot,graph</field> |
||||
|
</record> |
||||
|
<!-- Subscription report menu item --> |
||||
|
<menuitem id="subscription_report_menu" |
||||
|
name="Report" |
||||
|
parent="subscription_package.subscription_menu_root" |
||||
|
action="subscription_package.subscription_report_action" |
||||
|
sequence="11"/> |
||||
|
</odoo> |
|
@ -0,0 +1,23 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Subscription security group --> |
||||
|
|
||||
|
<record model="ir.module.category" id="module_subscription_category"> |
||||
|
<field name="name">Subscription</field> |
||||
|
<field name="description">Helps you handle your subscription security.</field> |
||||
|
<field name="sequence">9</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="group_subscription_user" model="res.groups"> |
||||
|
<field name="name">User</field> |
||||
|
<field name="category_id" ref="module_subscription_category"/> |
||||
|
<field name="implied_ids" eval="[(4, ref('base.group_user'))]"/> |
||||
|
</record> |
||||
|
|
||||
|
<record id="group_subscription_manager" model="res.groups"> |
||||
|
<field name="name">Subscription Administrator</field> |
||||
|
<field name="category_id" ref="module_subscription_category"/> |
||||
|
<field name="implied_ids" eval="[(4, ref('group_subscription_user'))]"/> |
||||
|
<field name="users" eval="[(4, ref('base.user_root')), (4, ref('base.user_admin'))]"/> |
||||
|
</record> |
||||
|
</odoo> |
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 28 KiB |
After Width: | Height: | Size: 628 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 210 KiB |
After Width: | Height: | Size: 209 KiB |
After Width: | Height: | Size: 109 KiB |
After Width: | Height: | Size: 495 B |
After Width: | Height: | Size: 1.0 KiB |
After Width: | Height: | Size: 624 B |
After Width: | Height: | Size: 136 KiB |
After Width: | Height: | Size: 214 KiB |
After Width: | Height: | Size: 36 KiB |
After Width: | Height: | Size: 3.6 KiB |
After Width: | Height: | Size: 310 B |
After Width: | Height: | Size: 929 B |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 3.3 KiB |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 17 KiB |
After Width: | Height: | Size: 542 B |
After Width: | Height: | Size: 576 B |
After Width: | Height: | Size: 733 B |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 4.0 KiB |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 738 KiB |
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 911 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 600 B |
After Width: | Height: | Size: 673 B |
After Width: | Height: | Size: 2.0 KiB |
After Width: | Height: | Size: 462 B |
After Width: | Height: | Size: 2.1 KiB |
After Width: | Height: | Size: 926 B |
After Width: | Height: | Size: 9.0 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 7.0 KiB |
After Width: | Height: | Size: 878 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 653 B |
After Width: | Height: | Size: 800 B |
After Width: | Height: | Size: 905 B |
After Width: | Height: | Size: 189 KiB |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 839 B |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 5.9 KiB |
After Width: | Height: | Size: 1.6 KiB |
After Width: | Height: | Size: 34 KiB |
After Width: | Height: | Size: 26 KiB |
After Width: | Height: | Size: 3.8 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 1.9 KiB |
After Width: | Height: | Size: 2.3 KiB |
After Width: | Height: | Size: 427 B |
After Width: | Height: | Size: 627 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 988 B |
After Width: | Height: | Size: 3.7 KiB |
After Width: | Height: | Size: 5.0 KiB |
After Width: | Height: | Size: 875 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 912 KiB |
After Width: | Height: | Size: 1.3 MiB |
After Width: | Height: | Size: 46 KiB |
After Width: | Height: | Size: 44 KiB |
After Width: | Height: | Size: 46 KiB |