@ -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  |