@ -0,0 +1,48 @@ |
|||||
|
.. image:: https://img.shields.io/badge/license-LGPL--3-green.svg |
||||
|
:target: https://www.gnu.org/licenses/lgpl-3.0-standalone.html |
||||
|
:alt: License: LGPL-3 |
||||
|
|
||||
|
Odoo Vendor Portal |
||||
|
================== |
||||
|
* This module helps to sent quotations for a product to multiple vendors and vendors can add their price in their portal and can choose best quotation for product. |
||||
|
|
||||
|
Configuration |
||||
|
============= |
||||
|
* No additional configurations needed |
||||
|
|
||||
|
Company |
||||
|
------- |
||||
|
* `Cybrosys Techno Solutions <https://cybrosys.com/>`__ |
||||
|
|
||||
|
License |
||||
|
------- |
||||
|
General Public License, Version 3 (LGPL v3). |
||||
|
(https://www.gnu.org/licenses/lgpl-3.0-standalone.html) |
||||
|
|
||||
|
Credits |
||||
|
------- |
||||
|
* Developer: (V17) Ashwin A, |
||||
|
(V18) Akhil, |
||||
|
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: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from . import controllers |
||||
|
from . import models |
||||
|
from . import wizard |
@ -0,0 +1,54 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
################################################################################ |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
################################################################################ |
||||
|
{ |
||||
|
'name': 'Odoo Vendor Portal', |
||||
|
'version': '18.0.1.0.0', |
||||
|
'category': 'Purchases', |
||||
|
'summary': """Vendor Portal Management in Odoo""", |
||||
|
'description': """This module helps to sent quotations for a product |
||||
|
to multiple vendors and vendors can add their |
||||
|
price in their portal, and can choose best quotation for product""", |
||||
|
'author': 'Cybrosys Techno Solutions', |
||||
|
'website': "https://www.cybrosys.com", |
||||
|
'company': 'Cybrosys Techno Solutions', |
||||
|
'maintainer': 'Cybrosys Techno Solutions', |
||||
|
'depends': ['website', 'purchase', 'portal', 'contacts', 'stock'], |
||||
|
'data': [ |
||||
|
'security/vendor_rfq_security.xml', |
||||
|
'security/ir.model.access.csv', |
||||
|
'data/rfq_sequence.xml', |
||||
|
'data/rfq_mail_templates.xml', |
||||
|
'data/rfq_cron.xml', |
||||
|
'wizard/register_vendor_views.xml', |
||||
|
'views/res_partner_views.xml', |
||||
|
'views/vendor_rfq_views.xml', |
||||
|
'views/res_config_settings_views.xml', |
||||
|
'views/portal_rfq_templates.xml', |
||||
|
'views/vendor_portal_menus.xml', |
||||
|
'wizard/rfq_done_views.xml', |
||||
|
], |
||||
|
'images': ['static/description/banner.png'], |
||||
|
'license': 'LGPL-3', |
||||
|
'installable': True, |
||||
|
'auto_install': False, |
||||
|
'application': True, |
||||
|
} |
@ -0,0 +1,22 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from . import vendor_portal_odoo |
@ -0,0 +1,144 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from collections import OrderedDict |
||||
|
from odoo import http, _ |
||||
|
from odoo.http import request |
||||
|
from odoo.addons.portal.controllers.portal import pager as portal_pager, \ |
||||
|
CustomerPortal |
||||
|
|
||||
|
|
||||
|
class RFQCustomerPortal(CustomerPortal): |
||||
|
|
||||
|
def _prepare_home_portal_values(self, counter): |
||||
|
"""Retrieves and prepares the values to be displayed on the home portal |
||||
|
for the user.""" |
||||
|
values = super()._prepare_home_portal_values(counter) |
||||
|
partner_id = request.env.user.partner_id |
||||
|
values['my_rfq_count'] = request.env['vendor.rfq'].sudo().search_count( |
||||
|
[('vendor_ids', 'in', partner_id.ids), |
||||
|
('state', 'not in', ['draft'])]) |
||||
|
return values |
||||
|
|
||||
|
def _rfq_get_page_view_values(self, vendor_rfq, access_token, **kwargs): |
||||
|
"""RFQ Page values""" |
||||
|
values = { |
||||
|
'page_name': 'vendor_rfq', |
||||
|
'vendor_rfq': vendor_rfq, |
||||
|
} |
||||
|
return self._get_page_view_values(vendor_rfq, access_token, values, |
||||
|
'my_rfq_history', False, **kwargs) |
||||
|
|
||||
|
@http.route(['/my/vendor_rfqs', '/my/vendor_rfqs/page/<int:page>'], |
||||
|
type='http', auth="public", website=True) |
||||
|
def portal_my_vendor_rfqs(self, page=1, date_begin=None, |
||||
|
date_end=None, sortby=None, filterby=None, **kw): |
||||
|
""" Displays the portal page for vendor RFQs (Request for Quotations) |
||||
|
for the logged-in user.This method is responsible for rendering the |
||||
|
vendor RFQs in the portal with various filtering, sorting, and |
||||
|
pagination options.""" |
||||
|
values = self._prepare_portal_layout_values() |
||||
|
user_partner = request.env.user.partner_id |
||||
|
vendor_rfq = request.env['vendor.rfq'].sudo().search([]) |
||||
|
domain = [ |
||||
|
('vendor_ids', 'in', user_partner.ids), |
||||
|
('state', 'not in', ['draft'])] |
||||
|
if date_begin and date_end: |
||||
|
domain += [('create_date', '>', date_begin), |
||||
|
('create_date', '<=', date_end)] |
||||
|
|
||||
|
searchbar_sortings = { |
||||
|
'date': {'label': _('Newest'), |
||||
|
'order': 'create_date desc, id desc'}, |
||||
|
'name': {'label': _('Name'), 'order': 'name asc, id asc'}, |
||||
|
} |
||||
|
if not sortby: |
||||
|
sortby = 'name' |
||||
|
order = searchbar_sortings[sortby]['order'] |
||||
|
searchbar_filters = { |
||||
|
'all': {'label': _('All'), 'domain': [ |
||||
|
('state', 'in', ['draft', 'in_progress', 'pending', |
||||
|
'done', 'cancel'])]}, |
||||
|
'Done': {'label': _('Done'), 'domain': [('state', '=', 'done')]}, |
||||
|
'In Progress': {'label': _('In Progress'), |
||||
|
'domain': [('state', '=', 'in_progress')]}, |
||||
|
} |
||||
|
# default filter by value |
||||
|
if not filterby: |
||||
|
filterby = 'all' |
||||
|
domain += searchbar_filters[filterby]['domain'] |
||||
|
rfq_unit_count = vendor_rfq.search_count(domain) |
||||
|
pager = portal_pager( |
||||
|
url="/my/vendor_rfqs", |
||||
|
url_args={'date_begin': date_begin, 'date_end': date_end, |
||||
|
'sortby': sortby, 'filterby': filterby}, |
||||
|
total=rfq_unit_count, |
||||
|
page=page, |
||||
|
step=self._items_per_page |
||||
|
) |
||||
|
|
||||
|
orders = vendor_rfq.search( |
||||
|
domain, |
||||
|
order=order, |
||||
|
limit=self._items_per_page, |
||||
|
offset=pager['offset'] |
||||
|
) |
||||
|
values.update({ |
||||
|
'date': date_begin, |
||||
|
'rfqs': orders, |
||||
|
'page_name': 'vendor_rfq', |
||||
|
'pager': pager, |
||||
|
'searchbar_sortings': searchbar_sortings, |
||||
|
'sortby': sortby, |
||||
|
'searchbar_filters': OrderedDict(sorted(searchbar_filters.items())), |
||||
|
'filterby': filterby, |
||||
|
'default_url': '/my/vendor_rfqs', |
||||
|
}) |
||||
|
return request.render( |
||||
|
"vendor_portal_odoo.portal_my_rfq", |
||||
|
values) |
||||
|
|
||||
|
@http.route(['/my/vendor_rfq/<int:rfq_id>'], type='http', auth="public", |
||||
|
website=True) |
||||
|
def portal_my_vendor_rfq(self, rfq_id, access_token=None, **kw): |
||||
|
""" Displays the details of a specific vendor RFQ (Request for Quotation) |
||||
|
for the logged-in user.""" |
||||
|
rfq_details = request.env['vendor.rfq'].sudo().browse(int(rfq_id)) |
||||
|
vendor_quote = rfq_details.vendor_quote_history_ids.filtered( |
||||
|
lambda x: x.vendor_id.id == request.env.user.partner_id.id) |
||||
|
quoted_price = vendor_quote.quoted_price |
||||
|
values = self._rfq_get_page_view_values(rfq_details, access_token, **kw) |
||||
|
values['quoted_price'] = quoted_price |
||||
|
values['vendor_quote'] = vendor_quote |
||||
|
return request.render( |
||||
|
"vendor_portal_odoo.portal_my_vendor_rfq", values) |
||||
|
|
||||
|
@http.route(['/quote/details'], type='http', auth="public", website=True) |
||||
|
def quote_details(self, **post): |
||||
|
"""Handle the submission of a vendor's quote details.""" |
||||
|
request.env['vendor.quote.history'].sudo().create({ |
||||
|
'vendor_id': request.env.user.partner_id.id, |
||||
|
'quoted_price': float(post.get('price')), |
||||
|
'estimate_date': post.get('delivery_date'), |
||||
|
'note': post.get('additional_note'), |
||||
|
'quote_id': post.get('rfq_id'), |
||||
|
}) |
||||
|
return request.redirect('/my/vendor_rfq/%s' % (post.get('rfq_id'))) |
@ -0,0 +1,13 @@ |
|||||
|
<?xml version="1.0" encoding="UTF-8" ?> |
||||
|
<odoo> |
||||
|
<!-- Cron job for Done the RFQ--> |
||||
|
<record id="ir_cron_quote_rfq" model="ir.cron"> |
||||
|
<field name="name">Set RFQs as Done</field> |
||||
|
<field name="model_id" ref="model_vendor_rfq"/> |
||||
|
<field name="state">code</field> |
||||
|
<field name="code">model.set_rfq_done()</field> |
||||
|
<field name="user_id" ref="base.user_root"/> |
||||
|
<field name="interval_number">1</field> |
||||
|
<field name="interval_type">days</field> |
||||
|
</record> |
||||
|
</odoo> |
@ -0,0 +1,77 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8" ?> |
||||
|
<odoo> |
||||
|
<data> |
||||
|
<!-- Template for RFQ--> |
||||
|
<record id="email_template_vendor_rfq_request" model="mail.template"> |
||||
|
<field name="name">Purchase Order: Send RFQ</field> |
||||
|
<field name="model_id" ref="vendor_portal_odoo.model_vendor_rfq"/> |
||||
|
<field name="subject">RFQ Request</field> |
||||
|
<field name="partner_to">{{ ctx.get('partner_to') }}</field> |
||||
|
<field name="body_html" type="html"> |
||||
|
<div style="margin: 0px; padding: 0px;"> |
||||
|
<p style="margin: 0px; padding: 0px; font-size: 13px;"> |
||||
|
Dear <t t-out="ctx.get('name') or ''"/>, |
||||
|
<br/> |
||||
|
<br/> |
||||
|
We would like to request a quotation for the Product |
||||
|
<strong t-out="object.product_id.name or ''"/> |
||||
|
of |
||||
|
<strong t-out="object.quantity"/> |
||||
|
<t t-out="object.uom_id.name"/> |
||||
|
Quantity from you. |
||||
|
<br/> |
||||
|
<br/> |
||||
|
If you are interested, please check and let us know the quote |
||||
|
<br/> |
||||
|
<br/> |
||||
|
Best regards, |
||||
|
<br/> |
||||
|
<t t-out="object.user_id.name"/> |
||||
|
</p> |
||||
|
</div> |
||||
|
</field> |
||||
|
<field name="lang">{{ ctx.get('lang') }}</field> |
||||
|
<field name="auto_delete" eval="True"/> |
||||
|
</record> |
||||
|
<!-- Template for Accepted Quote--> |
||||
|
<record id="email_template_vendor_rfq_mark_done" model="mail.template"> |
||||
|
<field name="name">Purchase Order: Quote Accepted</field> |
||||
|
<field name="model_id" ref="vendor_portal_odoo.model_vendor_rfq"/> |
||||
|
<field name="subject">Quote Accepted</field> |
||||
|
<field name="partner_to">{{ ctx.get('partner_to') }}</field> |
||||
|
<field name="body_html" type="html"> |
||||
|
<div style="margin: 0px; padding: 0px;"> |
||||
|
<p style="margin: 0px; padding: 0px; font-size: 13px;"> |
||||
|
Dear<t t-out="ctx.get('name') or ''"/>, |
||||
|
<br/> |
||||
|
<br/> |
||||
|
We want to let you know that we accept your quoted price of |
||||
|
<strong t-esc="ctx.get('price') or ''" |
||||
|
t-options='{"widget": "monetary", "display_currency": ctx.get("currency_id")}'/> |
||||
|
for the |
||||
|
<strong t-out="object.product_id.name or ''"/> |
||||
|
within your specified delivery date |
||||
|
<strong t-out="ctx.get('delivery_date') or ''"/> |
||||
|
from you. |
||||
|
<br/> |
||||
|
<br/> |
||||
|
We'd like to buy |
||||
|
<strong t-out="object.quantity or ''"/> |
||||
|
units of |
||||
|
<strong t-out="object.product_id.name or ''"/> |
||||
|
from you by the estimated date. |
||||
|
<br/> |
||||
|
We will send you a purchase order as soon as possible. |
||||
|
<br/> |
||||
|
<br/> |
||||
|
Best regards, |
||||
|
<br/> |
||||
|
<t t-out="object.user_id.name"/> |
||||
|
</p> |
||||
|
</div> |
||||
|
</field> |
||||
|
<field name="lang">{{ ctx.get('lang') }}</field> |
||||
|
<field name="auto_delete" eval="True"/> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
@ -0,0 +1,13 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8" ?> |
||||
|
<odoo> |
||||
|
<data noupdate="1"> |
||||
|
<!-- Sequence of vendor RFQ--> |
||||
|
<record id="seq_vendor_rfq" model="ir.sequence"> |
||||
|
<field name="name">RFQ Sequence</field> |
||||
|
<field name="code">vendor.rfq</field> |
||||
|
<field name="prefix">RFQ</field> |
||||
|
<field name="padding">5</field> |
||||
|
<field name="company_id" eval="False"/> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
@ -0,0 +1,6 @@ |
|||||
|
## Module <vendor_portal_odoo> |
||||
|
|
||||
|
#### 26.03.2025 |
||||
|
#### Version 18.0.1.0.0 |
||||
|
##### ADD |
||||
|
-Initial Commit for Odoo Vendor Portal |
@ -0,0 +1,25 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from . import res_config_settings |
||||
|
from . import res_partner |
||||
|
from . import vendor_quote_history |
||||
|
from . import vendor_rfq |
@ -0,0 +1,73 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from odoo import fields, models |
||||
|
|
||||
|
|
||||
|
class ResConfigSettings(models.TransientModel): |
||||
|
"""Class used to add field to res.config.settings""" |
||||
|
_inherit = 'res.config.settings' |
||||
|
|
||||
|
rfq_done_based_on = fields.Selection([ |
||||
|
('based_on_price', 'Minimum Quoted Price'), |
||||
|
('based_on_delivery_time', 'Minimum Delivery time') |
||||
|
], string="Set RFQs Based on", default='based_on_price', help='RFQ Based On') |
||||
|
quote_submission_msg = fields.Text("Quote Submission", |
||||
|
help="Status message to display if a quote was submitted") |
||||
|
quote_accept_msg = fields.Text("Quote Acceptance", |
||||
|
help="Status message to display if a quote was accepted") |
||||
|
quote_not_accept_msg = fields.Text("Quote not Accepted", |
||||
|
help="Status message to display if a quote was not accepted" |
||||
|
) |
||||
|
quote_cancel_msg = fields.Text("Quote Cancelled", |
||||
|
help="Status message to display if a quote was cancelled") |
||||
|
quote_to_po_msg = fields.Text("PO created for the RFQ", |
||||
|
help="Status message to display if a quote was converted to PO") |
||||
|
|
||||
|
def set_values(self): |
||||
|
"""Setting field values""" |
||||
|
res = super(ResConfigSettings, self).set_values() |
||||
|
config_parm = self.env['ir.config_parameter'].sudo() |
||||
|
config_parm.set_param( |
||||
|
'vendor_portal_odoo.rfq_done_based_on', self.rfq_done_based_on) |
||||
|
config_parm.set_param( |
||||
|
'vendor_portal_odoo.quote_submission_msg', self.quote_submission_msg) |
||||
|
config_parm.set_param( |
||||
|
'vendor_portal_odoo.quote_accept_msg', self.quote_accept_msg) |
||||
|
config_parm.set_param( |
||||
|
'vendor_portal_odoo.quote_not_accept_msg', self.quote_not_accept_msg) |
||||
|
config_parm.set_param( |
||||
|
'vendor_portal_odoo.quote_cancel_msg', self.quote_cancel_msg) |
||||
|
config_parm.set_param( |
||||
|
'vendor_portal_odoo.quote_to_po_msg', self.quote_to_po_msg) |
||||
|
return res |
||||
|
|
||||
|
def get_values(self): |
||||
|
"""Getting field values""" |
||||
|
res = super(ResConfigSettings, self).get_values() |
||||
|
config_parm = self.env['ir.config_parameter'].sudo() |
||||
|
res['rfq_done_based_on'] = config_parm.get_param('vendor_portal_odoo.rfq_done_based_on') |
||||
|
res['quote_submission_msg'] = config_parm.get_param('vendor_portal_odoo.quote_submission_msg') |
||||
|
res['quote_accept_msg'] = config_parm.get_param('vendor_portal_odoo.quote_accept_msg') |
||||
|
res['quote_not_accept_msg'] = config_parm.get_param('vendor_portal_odoo.quote_not_accept_msg') |
||||
|
res['quote_cancel_msg'] = config_parm.get_param('vendor_portal_odoo.quote_cancel_msg') |
||||
|
res['quote_to_po_msg'] = config_parm.get_param('vendor_portal_odoo.quote_to_po_msg') |
||||
|
return res |
@ -0,0 +1,32 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from odoo import fields, models |
||||
|
|
||||
|
|
||||
|
class ResPartner(models.Model): |
||||
|
"""Class used to add field to res.partner for differentiate |
||||
|
registered vendors""" |
||||
|
_inherit = 'res.partner' |
||||
|
|
||||
|
is_registered = fields.Boolean("Is Registered Vendor", default=False, |
||||
|
help="To denote the is the partner is " |
||||
|
"registered as vendor") |
@ -0,0 +1,44 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from odoo import fields, models |
||||
|
|
||||
|
|
||||
|
class VendorQuoteHistory(models.Model): |
||||
|
"""Vendor Quotation History""" |
||||
|
_name = 'vendor.quote.history' |
||||
|
_description = "Vendor Quotation History" |
||||
|
_rec_name = 'vendor_id' |
||||
|
|
||||
|
vendor_id = fields.Many2one('res.partner', |
||||
|
domain="[('is_registered', '=', True)]", |
||||
|
string="Vendor", |
||||
|
help="Select Vendor") |
||||
|
quoted_price = fields.Monetary(currency_field='currency_id', |
||||
|
help="Quoted Price") |
||||
|
currency_id = fields.Many2one('res.currency', string='Currency', |
||||
|
help="Currency", |
||||
|
required=True, |
||||
|
default=lambda |
||||
|
self: self.env.user.company_id.currency_id) |
||||
|
estimate_date = fields.Date(string="Estimate date", help="Estimated Date") |
||||
|
note = fields.Text(string="Note", help="Additional Note") |
||||
|
quote_id = fields.Many2one('vendor.rfq', help="Quote ID") |
@ -0,0 +1,184 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################# |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU LESSER |
||||
|
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
||||
|
# (LGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################# |
||||
|
from odoo import api, fields, models, _ |
||||
|
|
||||
|
|
||||
|
class VendorRFQ(models.Model): |
||||
|
"""Vendor RFQ model""" |
||||
|
_name = 'vendor.rfq' |
||||
|
_inherit = ['mail.thread', 'mail.activity.mixin'] |
||||
|
_description = 'Vendor RFQ' |
||||
|
|
||||
|
name = fields.Char('RFQ Reference', required=True, index=True, |
||||
|
help="Name", copy=False, default=lambda x: _('New')) |
||||
|
product_id = fields.Many2one('product.product', string='Product', |
||||
|
help="Select Product") |
||||
|
quantity = fields.Float(string="Quantity", help="Quantity Required") |
||||
|
uom_id = fields.Many2one('uom.uom', string='UoM', help="UOM") |
||||
|
estimated_quote = fields.Monetary("Estimated Quote", |
||||
|
currency_field='currency_id', |
||||
|
help="Estimated Quote Price") |
||||
|
currency_id = fields.Many2one('res.currency', string='Currency', |
||||
|
help="Currency", |
||||
|
required=True, |
||||
|
default=lambda |
||||
|
self: self.env.user.company_id.currency_id) |
||||
|
notes = fields.Html(string='Notes', help="Additional Note") |
||||
|
estimated_delivery_date = fields.Date(string="Delivery date", |
||||
|
help="Vendor's delivery date") |
||||
|
quote_date = fields.Date(default=fields.Datetime.now(), help="Quote Date ") |
||||
|
closing_date = fields.Date(string="Closing date", |
||||
|
help="Quotation closing date") |
||||
|
vendor_ids = fields.Many2many('res.partner', |
||||
|
domain="[('is_registered', '=', True)]", |
||||
|
help="Vendors you want to send quotations") |
||||
|
vendor_quote_history_ids = fields.One2many('vendor.quote.history', |
||||
|
'quote_id', |
||||
|
string="Vendor Quote History", |
||||
|
help="Vendor Quote History") |
||||
|
user_id = fields.Many2one('res.users', default=lambda self: self.env.user, |
||||
|
string="Responsible", help="Responsible Person") |
||||
|
approved_vendor_id = fields.Many2one('res.partner', string="Approved Vendor", help="Approved Vendor") |
||||
|
order_id = fields.Many2one('purchase.order', string='Order ID', help="Order ID") |
||||
|
state = fields.Selection([ |
||||
|
('draft', 'Draft'), |
||||
|
('pending', 'Pending'), |
||||
|
('in_progress', 'In Progress'), |
||||
|
('done', 'Done'), |
||||
|
('cancel', 'Cancelled'), ('order', 'Purchase Order'), |
||||
|
], string="Status", default='draft', help="Different stages") |
||||
|
company_id = fields.Many2one( |
||||
|
'res.company', string='Company', |
||||
|
default=lambda self: self.env.company, help="Select Company" |
||||
|
) |
||||
|
|
||||
|
@api.model |
||||
|
def create(self, vals): |
||||
|
"""Override the create method to assign a unique sequence to the |
||||
|
'name' field.""" |
||||
|
if vals.get('name', 'New') == 'New': |
||||
|
vals['name'] = self.env['ir.sequence'].next_by_code( |
||||
|
'vendor.rfq') or '/' |
||||
|
res = super(VendorRFQ, self).create(vals) |
||||
|
return res |
||||
|
|
||||
|
def action_send_by_mail(self): |
||||
|
"""This method sends an email to each vendor listed in the `vendor_ids` |
||||
|
field, using a predefined email template. The template is personalized |
||||
|
with the vendor's name and language. The email is sent from the current |
||||
|
user's email address.""" |
||||
|
template = self.env.ref( |
||||
|
'vendor_portal_odoo.email_template_vendor_rfq_request').id |
||||
|
for vendor in self.vendor_ids: |
||||
|
context = { |
||||
|
'name': vendor.name, |
||||
|
'lang': vendor.lang, |
||||
|
} |
||||
|
email_values = { |
||||
|
'email_to': vendor.email, |
||||
|
'email_from': self.env.user.partner_id.email, |
||||
|
} |
||||
|
self.env['mail.template'].browse(template).with_context( |
||||
|
context).send_mail(self.id, email_values=email_values, |
||||
|
force_send=True) |
||||
|
self.state = 'in_progress' |
||||
|
|
||||
|
def action_pending(self): |
||||
|
"""For changing state to pending""" |
||||
|
self.state = 'pending' |
||||
|
|
||||
|
def action_cancel(self): |
||||
|
"""For changing state to cancel""" |
||||
|
self.state = 'cancel' |
||||
|
|
||||
|
def action_done(self): |
||||
|
"""For mark as done""" |
||||
|
|
||||
|
return { |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'view_mode': 'form', |
||||
|
'res_model': 'rfq.done', |
||||
|
'target': 'new', |
||||
|
'views': [[False, 'form']], |
||||
|
'context': {'default_quote_ids': self.vendor_quote_history_ids.ids}, |
||||
|
} |
||||
|
|
||||
|
def action_create_quotation(self): |
||||
|
"""For creating purchase RFQ from vendor quotations""" |
||||
|
rfq_quote = self.env['vendor.quote.history'].search([ |
||||
|
('vendor_id', '=', self.approved_vendor_id.id), |
||||
|
('quote_id', '=', self.id)]) |
||||
|
price = rfq_quote.quoted_price |
||||
|
order = self.env['purchase.order'].sudo().create({ |
||||
|
'partner_id': self.approved_vendor_id.id, |
||||
|
'order_line': [ |
||||
|
(0, 0, { |
||||
|
'name': self.product_id.name, |
||||
|
'product_id': self.product_id.id, |
||||
|
'product_qty': self.quantity, |
||||
|
'product_uom': self.product_id.uom_po_id.id, |
||||
|
'price_unit': price, |
||||
|
'date_planned': rfq_quote.estimate_date, |
||||
|
'taxes_id': [(6, 0, self.product_id.supplier_taxes_id.ids)], |
||||
|
})], |
||||
|
}) |
||||
|
self.write({ |
||||
|
'state': 'order', |
||||
|
'order_id': order.id |
||||
|
}) |
||||
|
return { |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'res_model': 'purchase.order', |
||||
|
'res_id': order.id, |
||||
|
'target': 'current', |
||||
|
'views': [(False, 'form')], |
||||
|
} |
||||
|
|
||||
|
def set_rfq_done(self): |
||||
|
"""Set the RFQ as done""" |
||||
|
quotes = self.search([('state', '=', 'in_progress'), |
||||
|
('vendor_quote_history_ids', |
||||
|
'!=', False), |
||||
|
('closing_date', '=', |
||||
|
fields.Date.today())]) |
||||
|
if quotes: |
||||
|
rfq_done_based_on = self.env['ir.config_parameter'].get_param( |
||||
|
'vendor_portal_odoo.rfq_done_based_on') |
||||
|
for rec in quotes: |
||||
|
order = 'quoted_price asc' if rfq_done_based_on == 'based_on_price' else 'estimate_date asc' |
||||
|
vendor_quotes = rec.vendor_quote_history_ids.search([], |
||||
|
limit=1, |
||||
|
order=order) |
||||
|
rec.write({ |
||||
|
'approved_vendor_id': vendor_quotes.vendor_id.id, |
||||
|
'state': 'done' |
||||
|
}) |
||||
|
|
||||
|
def get_purchase_order(self): |
||||
|
"""Retrieve the purchase order associated with the current record.""" |
||||
|
return { |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'res_model': 'purchase.order', |
||||
|
'res_id': self.order_id.id, |
||||
|
'target': 'current', |
||||
|
'views': [(False, 'form')], |
||||
|
} |
|
@ -0,0 +1,9 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Multi company Rule--> |
||||
|
<record id="vendor_rfq_rule_company" model="ir.rule"> |
||||
|
<field name="name">Vendor Quotation multi company rule</field> |
||||
|
<field name="model_id" ref="model_vendor_rfq"/> |
||||
|
<field name="domain_force">['|', ('company_id', '=', False), ('company_id', 'in', company_ids)]</field> |
||||
|
</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: 392 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: 767 KiB |
After Width: | Height: | Size: 138 KiB |
After Width: | Height: | Size: 760 KiB |
After Width: | Height: | Size: 92 KiB |
After Width: | Height: | Size: 697 KiB |
After Width: | Height: | Size: 1.1 MiB |
After Width: | Height: | Size: 166 KiB |
After Width: | Height: | Size: 151 KiB |
After Width: | Height: | Size: 117 KiB |
After Width: | Height: | Size: 114 KiB |
After Width: | Height: | Size: 82 KiB |
After Width: | Height: | Size: 150 KiB |
After Width: | Height: | Size: 47 KiB |
After Width: | Height: | Size: 80 KiB |
After Width: | Height: | Size: 102 KiB |
After Width: | Height: | Size: 111 KiB |