diff --git a/product_visibility_website/README.rst b/product_visibility_website/README.rst new file mode 100644 index 000000000..577b7ef8d --- /dev/null +++ b/product_visibility_website/README.rst @@ -0,0 +1,54 @@ +.. 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 + +Website Product Visibility +=========================== +This module helps you to make visible only the filtered products and +product categories for a logged in and logged out users/visitors.Also, +it enables the user to search products and product categories only from +those available products and categories. + +Configuration +============= +No additional configurations needed. + +Company +------- +* `Cybrosys Techno Solutions `__ + +License +------- +General Public License, Version 3 (AGPL v3). +(https://www.gnu.org/licenses/agpl-3.0-standalone.html) + +Credits +------- +Developers: Shijin V, + (V14)Muhammed Nafih, + (V15)Neeraj, + (V16)Saneen K, + (V17)Jumana Haseen, +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 `__ + +Further information +=================== +HTML Description: ``__ diff --git a/product_visibility_website/__init__.py b/product_visibility_website/__init__.py new file mode 100644 index 000000000..596c7a2c9 --- /dev/null +++ b/product_visibility_website/__init__.py @@ -0,0 +1,24 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################ +from . import controllers +from . import models diff --git a/product_visibility_website/__manifest__.py b/product_visibility_website/__manifest__.py new file mode 100644 index 000000000..34fb11d5d --- /dev/null +++ b/product_visibility_website/__manifest__.py @@ -0,0 +1,45 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################ +{ + 'name': 'Website Product Visibility', + 'version': '17.0.1.0.0', + 'category': 'Website', + 'summary': 'Website Product visibility for Users', + 'description': """This module helps you to make visible only the filtered + products and product categories for a logged in and logged out users/portal + users/visitors.""", + 'author': 'Cybrosys Techno Solution', + 'company': 'Cybrosys Techno Solutions', + 'maintainer': 'Cybrosys Techno Solutions', + 'website': 'https://www.cybrosys.com', + 'depends': ['contacts', 'website_sale', 'website'], + 'data': [ + 'views/res_partner_views.xml', + 'views/res_config_settings_views.xml' + ], + 'images': ['static/description/banner.png'], + 'license': 'AGPL-3', + 'installable': True, + 'auto_install': False, + 'application': False, +} diff --git a/product_visibility_website/controllers/__init__.py b/product_visibility_website/controllers/__init__.py new file mode 100644 index 000000000..86d53e78e --- /dev/null +++ b/product_visibility_website/controllers/__init__.py @@ -0,0 +1,23 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +############################################################################### +from . import product_visibility_website diff --git a/product_visibility_website/controllers/product_visibility_website.py b/product_visibility_website/controllers/product_visibility_website.py new file mode 100644 index 000000000..fe0b55aef --- /dev/null +++ b/product_visibility_website/controllers/product_visibility_website.py @@ -0,0 +1,416 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +############################################################################### +from odoo import fields, http, tools +from ast import literal_eval +from datetime import datetime +from odoo.osv import expression +from odoo.tools import lazy +from werkzeug.exceptions import NotFound +from odoo.addons.http_routing.models.ir_http import slug +from odoo.addons.website.controllers.main import QueryURL +from odoo.addons.website_sale.controllers.main import TableCompute +from odoo.http import request +from odoo.addons.website.models.ir_http import sitemap_qs2dom +from odoo.addons.website_sale.controllers.main import WebsiteSale + + +class ProductVisibilityCon(WebsiteSale): + + def sitemap_shop(env, rule, qs): + """ The sitemap_shop function generates sitemap entries for a + website's shop or e-commerce section based on the input query + string qs. It does this by searching for product categories + that match a specific domain, appending the website domain""" + if not qs or qs.lower() in '/shop': + yield {'loc': '/shop'} + category = env['product.public.category'] + dom = sitemap_qs2dom(qs, '/shop/category', category._rec_name) + dom += env['website'].get_current_website().website_domain() + for cat in category.search(dom): + loc = '/shop/category/%s' % slug(cat) + if not qs or qs.lower() in loc: + yield {'loc': loc} + + def reset_domain(self, search, categories, available_products, + search_in_description=True): + """ + Function returns a domain consist of filter conditions + :param search: search variable + :param categories: list of category available + :param available_products: list of available product ids from + product.template + :param attrib_values:product attiribute values + :param search_in_description: boolean filed showing there is search + variable exist or not""" + domains = [request.website.sale_product_domain()] + if search: + for srch in search.split(" "): + subdomains = [ + [('name', 'ilike', srch)], + [('product_variant_ids.default_code', 'ilike', srch)]] + if search_in_description: + subdomains.append([('description', 'ilike', srch)]) + subdomains.append([('description_sale', 'ilike', srch)]) + domains.append(expression.OR(subdomains)) + if available_products: + domains.append([('id', 'in', available_products.ids)]) + if categories: + domains.append([('public_categ_ids', 'child_of', categories.ids)]) + return expression.AND(domains) + + @http.route(type='http', auth="public", website=True) + def shop(self, page=0, category=None, search='', min_price=0.0, + max_price=0.0, ppg=False, **post): + available_categ = available_products = '' + user = request.env['res.users'].sudo().search( + [('id', '=', request.env.user.id)]) + if not user: + mode = request.env['ir.config_parameter'].sudo().get_param( + 'filter_mode') + products = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.' + 'available_products_for_guest_ids', 'False')) + if mode == 'product_only': + available_products = request.env['product.template'].search( + [('id', 'in', products)]) + cat = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_cat_for_guest_ids', + 'False')) + available_categ = request.env['product.public.category'].search( + [('id', 'in', cat)]) + else: + partner = request.env['res.partner'].sudo().search( + [('id', '=', user.partner_id.id)]) + mode = partner.filter_mode + if mode == 'product_only': + available_products = self.available_products() + available_categ = partner.website_available_cat_ids + category_avail = [] + Category = request.env['product.public.category'] + for ids in available_categ: + if ids.parent_id.id not in available_categ.ids: + category_avail.append(ids.id) + categ = request.env['product.public.category'].search( + [('id', 'in', category_avail)]) + if mode == 'product_only': + categ = Category.search([('parent_id', '=', False), ( + 'product_tmpl_ids', 'in', available_products.ids)]) + # supering shop*** + if not available_categ and not available_products and \ + request.env.user.has_group( + 'base.group_portal'): + mode = request.env['ir.config_parameter'].sudo().get_param( + 'filter_mode_portal') + products = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.' + 'available_products_for_portal_ids', 'False')) + if mode == 'product_only': + available_products = request.env['product.template'].search( + [('id', 'in', products)]) + cat = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_cat_for_portal_ids', + 'False')) + available_categ = request.env['product.public.category'].search( + [('id', 'in', cat)]) + + if not available_categ and not available_products and \ + not request.env.user.has_group( + 'base.group_user'): + return super(ProductVisibilityCon, self).shop(page, category, + search, ppg, **post) + add_qty = int(post.get('add_qty', 1)) + try: + min_price = float(min_price) + except ValueError: + min_price = 0 + try: + max_price = float(max_price) + except ValueError: + max_price = 0 + Category = request.env['product.public.category'] + if category: + category = Category.search([('id', '=', int(category))], limit=1) + if not category or not category.can_access_from_current_website(): + raise NotFound() + else: + category = Category + website = request.env['website'].get_current_website() + if ppg: + try: + ppg = int(ppg) + post['ppg'] = ppg + except ValueError: + ppg = False + if not ppg: + ppg = request.env['website'].get_current_website().shop_ppg or 20 + ppr = request.env['website'].get_current_website().shop_ppr or 4 + attrib_list = request.httprequest.args.getlist('attrib') + attrib_values = [[int(x) for x in v.split("-")] for v in attrib_list if + v] + attributes_ids = {v[0] for v in attrib_values} + attrib_set = {v[1] for v in attrib_values} + domain = self._get_shop_domain(search, category, attrib_values) + product = request.env['product.template'].with_context(bin_size=True) + if available_products: + domain_pro = self.reset_domain(search, category, available_products, + attrib_values) + product = product.search(domain_pro) + keep = QueryURL('/shop', **self._shop_get_query_url_kwargs( + category and int(category), search, min_price, max_price, **post)) + now = datetime.timestamp(datetime.now()) + pricelist = request.env['product.pricelist'].browse( + request.session.get('website_sale_current_pl')) + if not pricelist or request.session.get('website_sale_pricelist_time', + 0) < now - 60 * 60: + # test: 1 hour in session + pricelist = website._get_current_pricelist() + request.session['website_sale_pricelist_time'] = now + request.session['website_sale_current_pl'] = pricelist.id + request.update_context(pricelist=pricelist.id, + partner=request.env.user.partner_id) + filter_by_price_enabled = website.is_view_active( + 'website_sale.filter_products_price') + if filter_by_price_enabled: + company_currency = website.company_id.currency_id + conversion_rate = request.env['res.currency']._get_conversion_rate( + company_currency, pricelist.currency_id, + request.website.company_id, fields.Date.today()) + else: + conversion_rate = 1 + url = "/shop" + if search: + post["search"] = search + if attrib_list: + post['attrib'] = attrib_list + options = self._get_search_options( + category=category, + attrib_values=attrib_values, + pricelist=pricelist, + min_price=min_price, + max_price=max_price, + conversion_rate=conversion_rate, + **post + ) + fuzzy_search_term, product_count, search_product = \ + self._shop_lookup_products( + attrib_set, options, post, search, website) + filter_by_price_enabled = website.is_view_active( + 'website_sale.filter_products_price') + if filter_by_price_enabled: + # TODO Find an alternative way to obtain the domain through the + # search metadata. + product = request.env['product.template'].with_context( + bin_size=True) + domain = self._get_shop_domain(search, category, attrib_values) + # This is ~4 times more efficient than a search for the cheapest + # and most expensive products + from_clause, where_clause, where_params = product._where_calc( + domain).get_sql() + query = f""" + SELECT COALESCE(MIN(list_price), 0) * {conversion_rate}, + COALESCE(MAX(list_price), 0) * {conversion_rate} + FROM {from_clause} + WHERE {where_clause} + """ + request.env.cr.execute(query, where_params) + available_min_price, available_max_price = request.env.cr.fetchone() + if min_price or max_price: + # The if/else condition in the min_price / max_price value + # assignment + # tackles the case where we switch to a list of products with + # different + # available min / max prices than the ones set in the previous + # page. + # In order to have logical results and not yield empty product + # lists, the + # price filter is set to their respective available prices when + # the specified + # min exceeds the max, and / or the specified max is lower than + # the available min. + if min_price: + min_price = min_price if min_price <= available_max_price \ + else available_min_price + post['min_price'] = min_price + if max_price: + max_price = max_price if max_price >= available_min_price \ + else available_max_price + post['max_price'] = max_price + website_domain = website.website_domain() + categs_domain = [('parent_id', '=', False)] + website_domain + if not category: + domain = self.reset_domain(search, available_categ, + available_products, attrib_values) + search_product = product.search(domain) + website_domain = request.website.website_domain() + categs_domain = [('parent_id', '=', False), ( + 'product_tmpl_ids', 'in', search_product.ids)] + website_domain + if search: + search_categories = Category.search( + [( + 'product_tmpl_ids', 'in', + search_product.ids)] + website_domain + ).parents_and_self + categs_domain.append(('id', 'in', search_categories.ids)) + else: + search_categories = available_categ + categs = lazy(lambda: Category.search(categs_domain)) + if category: + url = "/shop/category/%s" % slug(category) + product_count = len(search_product) + pager = website.pager(url=url, total=product_count, page=page, step=ppg, + scope=7, url_args=post) + offset = pager['offset'] + products = product.search(domain, limit=ppg, offset=pager['offset'], + order=self._get_search_order(post)) + if not category: + if available_products: + products = product.search(domain_pro, limit=ppg, + offset=pager['offset'], + order=self._get_search_order(post)) + else: + products = product.search(domain, limit=ppg, + offset=pager['offset'], + order=self._get_search_order(post)) + else: + if available_products: + products = product.search(domain_pro, limit=ppg, + offset=pager['offset'], + order=self._get_search_order(post)) + else: + products = product.search(domain, limit=ppg, + offset=pager['offset'], + order=self._get_search_order(post)) + productAttribute = request.env['product.attribute'] + if products: + # get all products without limit + attributes = lazy(lambda: productAttribute.search([ + ('product_tmpl_ids', 'in', search_product.ids), + ('visibility', '=', 'visible'), + ])) + else: + attributes = lazy(lambda: productAttribute.browse(attributes_ids)) + layout_mode = request.session.get('website_sale_shop_layout_mode') + if not layout_mode: + if website.viewref('website_sale.products_list_view').active: + layout_mode = 'list' + else: + layout_mode = 'grid' + request.session['website_sale_shop_layout_mode'] = layout_mode + fiscal_position = website.fiscal_position_id.sudo() + products_prices = lazy( + lambda: products._get_sales_prices(pricelist, fiscal_position)) + values = { + 'search': fuzzy_search_term or search, + 'original_search': fuzzy_search_term and search, + 'order': post.get('order', ''), + 'category': category, + 'attrib_values': attrib_values, + 'attrib_set': attrib_set, + 'pager': pager, + 'pricelist': pricelist, + 'add_qty': add_qty, + 'products': products, + 'search_product': search_product, + 'search_count': product_count, # common for all searchbox + 'bins': TableCompute().process(products, ppg, ppr), + 'ppg': ppg, + 'ppr': ppr, + 'categories': categ, + 'attributes': attributes, + 'keep': keep, + 'search_categories_ids': categ.ids, + 'layout_mode': layout_mode, + 'products_prices': products_prices, + 'get_product_prices': lambda product: lazy( + lambda: products_prices[product.id]), + 'float_round': tools.float_round, + } + if filter_by_price_enabled: + values['min_price'] = min_price or available_min_price + values['max_price'] = max_price or available_max_price + values['available_min_price'] = tools.float_round( + available_min_price, 2) + values['available_max_price'] = tools.float_round( + available_max_price, 2) + if category: + values['main_object'] = category + values.update(self._get_additional_shop_values(values)) + return request.render("website_sale.products", values) + + def available_products(self): + """Returns the available product (product.template) ids""" + user = request.env['res.users'].sudo().search( + [('id', '=', request.env.user.id)]) + partner = request.env['res.partner'].sudo().search( + [('id', '=', user.partner_id.id)]) + return partner.website_available_product_ids + + def _get_current_pricelist(self): + """:returns: The current pricelist record""" + self = self.with_company(self.company_id) + ProductPricelist = self.env['product.pricelist'] + pricelist = ProductPricelist + if request and request.session.get('website_sale_current_pl'): + # `website_sale_current_pl` is set only if the user specifically + # chose it: + # - Either, he chose it from the pricelist selection + # - Either, he entered a coupon code + pricelist = ProductPricelist.browse( + request.session['website_sale_current_pl']).exists().sudo() + country_code = self._get_geoip_country_code() + if not pricelist or not pricelist._is_available_on_website( + self) or not pricelist._is_available_in_country( + country_code): + request.session.pop('website_sale_current_pl') + pricelist = ProductPricelist + if not pricelist: + partner_sudo = self.env.user.partner_id + # If the user has a saved cart, it take the pricelist of this last + # unconfirmed cart + pricelist = partner_sudo.last_website_so_id.pricelist_id + if not pricelist: + # The pricelist of the user set on its partner form. + # If the user is not signed in, it's the public user pricelist + pricelist = partner_sudo.property_product_pricelist + # The list of available pricelists for this user. + # If the user is signed in, and has a pricelist set different than + # the public user pricelist + # then this pricelist will always be considered as available + available_pricelists = self.get_pricelist_available() + if available_pricelists and pricelist not in available_pricelists: + # If there is at least one pricelist in the available pricelists + # and the chosen pricelist is not within them + # it then choose the first available pricelist. + # This can only happen when the pricelist is the public user + # pricelist and this pricelist is not in the available pricelist + # for this localization + # If the user is signed in, and has a special pricelist + # (different than the public user pricelist), + # then this special pricelist is amongs these available + # pricelists, and therefore it won't fall in this case. + pricelist = available_pricelists[0] + return pricelist diff --git a/product_visibility_website/doc/RELEASE_NOTES.md b/product_visibility_website/doc/RELEASE_NOTES.md new file mode 100644 index 000000000..7e6970bc7 --- /dev/null +++ b/product_visibility_website/doc/RELEASE_NOTES.md @@ -0,0 +1,6 @@ +## Module +#### 18.12.2023 +#### Version 17.0.1.0.0 +##### ADD +- Initial Commit for Website Product Visibility + diff --git a/product_visibility_website/models/__init__.py b/product_visibility_website/models/__init__.py new file mode 100644 index 000000000..80cdb7cfc --- /dev/null +++ b/product_visibility_website/models/__init__.py @@ -0,0 +1,25 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################ +from . import res_config_settings +from . import res_partner +from . import website diff --git a/product_visibility_website/models/res_config_settings.py b/product_visibility_website/models/res_config_settings.py new file mode 100644 index 000000000..23da562be --- /dev/null +++ b/product_visibility_website/models/res_config_settings.py @@ -0,0 +1,164 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################ +from odoo import api, fields, models +from ast import literal_eval + + +class ResConfigSettings(models.TransientModel): + """Inherit the model res.config.settings for adding the fields for + selecting the product for website visitors and portal users""" + _inherit = 'res.config.settings' + + product_visibility_guest_user = fields.Boolean( + string="Product visibility Guest User", help="Product Visibility") + filter_mode = fields.Selection([('product_only', 'Product Wise'), + ('categ_only', 'Category Wise')], + string='Filter Mode', default='product_only', + help="Filter mode of product visibility") + available_products_for_guest_ids = fields.Many2many( + 'product.template', + string='Available Product', + domain="[('is_published', '=', True)]", + help='The website will only display products which are within one of ' + 'the selected category trees. If no category is specified,all ' + 'available products will be shown') + available_cat_for_guest_ids = fields.Many2many( + 'product.public.category', + string='Available Product Categories', + help='The website will only display products which are selected. If no ' + 'product is specified,all available products will be shown') + product_visibility_portal_user = fields.Boolean( + string="Product visibility Portal User", help="Product Visibility") + filter_mode_portal = fields.Selection([('product_only', 'Product Wise'), + ('categ_only', 'Category Wise')], + string='Filter Mode Portal', + default='product_only', + help="Filter mode in portal") + available_products_for_portal_ids = fields.Many2many( + 'product.template', + relation="available_product_for_portal_rel", + string='Available Product for Portal', + domain="[('is_published', '=', True)]", + help='The website will only display products which are within one of ' + 'the selected category trees. If no category is specified,all ' + 'available products will be shown') + available_cat_for_portal_ids = fields.Many2many( + 'product.public.category', + relation="available_cat_for_portal_rel", + string='Available Portal Product Categories', + help='The website will only display products which are selected. If no ' + 'product is specified,all available products will be shown') + + @api.model + def set_values(self): + """This function is responsible for setting configuration parameters + based on the values of certain attributes in the model. + These parameters are used to control various settings and + behaviors of the module.""" + res = super(ResConfigSettings, self).set_values() + self.env['ir.config_parameter'].sudo().set_param( + 'product_visibility_guest_user', + self.product_visibility_guest_user) + self.env['ir.config_parameter'].sudo().set_param( + 'product_visibility_portal_user', + self.product_visibility_portal_user) + self.env['ir.config_parameter'].sudo().set_param('filter_mode', + self.filter_mode) + self.env['ir.config_parameter'].sudo().set_param( + 'filter_mode_portal', self.filter_mode_portal) + if not self.product_visibility_guest_user: + self.available_cat_for_guest_ids = None + self.available_products_for_guest_ids = None + self.env['ir.config_parameter'].sudo().set_param('filter_mode', + 'product_only') + if not self.product_visibility_portal_user: + self.available_products_for_portal_ids = None + self.available_cat_for_portal_ids = None + if self.filter_mode == 'product_only': + self.available_cat_for_guest_ids = None + elif self.filter_mode == 'categ_only': + self.available_products_for_guest_ids = None + if self.filter_mode_portal == 'product_only': + self.available_cat_for_portal_ids = None + elif self.filter_mode_portal == 'categ_only': + self.available_products_for_portal_ids = None + self.env['ir.config_parameter'].sudo().set_param( + 'website_product_visibility.available_products_for_guest_ids', + self.available_products_for_guest_ids.ids) + self.env['ir.config_parameter'].sudo().set_param( + 'website_product_visibility.available_products_for_portal_ids', + self.available_products_for_portal_ids.ids) + self.env['ir.config_parameter'].sudo().set_param( + 'website_product_visibility.available_cat_for_guest_ids', + self.available_cat_for_guest_ids.ids) + self.env['ir.config_parameter'].sudo().set_param( + 'website_product_visibility.available_cat_for_portal_ids', + self.available_cat_for_portal_ids.ids) + return res + + @api.model + def get_values(self): + """This code retrieves and processes various configuration parameters, + and then returns them as a dictionary to be used for configuring the + behavior of the Odoo module.""" + res = super(ResConfigSettings, self).get_values() + product_visibility_guest_user = self.env[ + 'ir.config_parameter'].sudo().get_param( + 'product_visibility_guest_user') + product_visibility_portal_user = self.env[ + 'ir.config_parameter'].sudo().get_param( + 'product_visibility_portal_user') + filter_mode = self.env['ir.config_parameter'].sudo().get_param( + 'filter_mode') + filter_mode_portal = self.env['ir.config_parameter'].sudo().get_param( + 'filter_mode_portal') + available_products_for_guest_ids = literal_eval( + self.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_products_for_guest_ids', + 'False')) or [] + available_products_for_portal_ids = literal_eval( + self.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_products_for_portal_ids', + 'False')) or [] + available_cat_for_guest_ids = literal_eval( + self.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_cat_for_guest_ids', + 'False')) or [] + available_cat_for_portal_ids = literal_eval( + self.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_cat_for_portal_ids', + 'False')) or [] + res.update( + product_visibility_guest_user=product_visibility_guest_user, + product_visibility_portal_user=product_visibility_portal_user, + filter_mode=filter_mode if filter_mode else 'product_only', + filter_mode_portal=filter_mode_portal if filter_mode_portal + else 'product_only', + available_products_for_guest_ids=[ + (6, 0, available_products_for_guest_ids)], + available_products_for_portal_ids=[ + (6, 0, available_products_for_portal_ids)], + available_cat_for_guest_ids=[(6, 0, available_cat_for_guest_ids)], + available_cat_for_portal_ids=[(6, 0, available_cat_for_portal_ids)] + ) + return res diff --git a/product_visibility_website/models/res_partner.py b/product_visibility_website/models/res_partner.py new file mode 100644 index 000000000..909441a9e --- /dev/null +++ b/product_visibility_website/models/res_partner.py @@ -0,0 +1,71 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################ +from odoo import api, fields, models + + +class ProductVisibility(models.Model): + """Inherit 'res.partner' for adding the fields for selecting the products""" + _inherit = 'res.partner' + + filter_mode = fields.Selection( + [('null', 'No Filter'),('product_only', 'Product Wise'), + ('categ_only', 'Category Wise')], string='Filter Mode', default="null", + help="Select any mode") + website_available_product_ids = fields.Many2many('product.template', + string='Available Product', + domain="[('is_published" + "', '=', True)]", + help='The website will' + ' only ' + 'display products ' + 'which ' + 'are selected.' + 'If no product is ' + 'specified,' + 'all available ' + 'products ' + 'will be shown', + required=True) + website_available_cat_ids = fields.Many2many('product.public.category', + string='Available Product ' + 'Categories', + help='The website will ' + 'only display ' + 'products which are ' + 'within one' + 'of the selected ' + 'category trees. If ' + 'no category is ' + 'specified,' + 'all available ' + 'products will be ' + 'shown', required=True + ) + + @api.onchange('filter_mode') + def onchange_filter_mode(self): + if self.filter_mode == 'null': + self.website_available_cat_ids = None + self.website_available_product_ids = None + self.website_available_cat_ids = False + self.website_available_product_ids = False diff --git a/product_visibility_website/models/website.py b/product_visibility_website/models/website.py new file mode 100644 index 000000000..be93a10e1 --- /dev/null +++ b/product_visibility_website/models/website.py @@ -0,0 +1,125 @@ +# -*- coding: utf-8 -*- +################################################################################ +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies () +# Author: Jumana Haseen () +# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# 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 for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################ +from ast import literal_eval +from odoo import models +from odoo.http import request + + +class Website(models.Model): + """Extends the 'website' model to filter product search.""" + _inherit = "website" + _description = "Website" + + def _search_with_fuzzy(self, search_type, search, limit, order, options): + """This method extends the base search functionality to include + additional filtering""" + res = super()._search_with_fuzzy( + search_type, search, limit, order, options) + response = list(res) + available_products = False + user = request.env['res.users'].sudo().search( + [('id', '=', request.env.user.id)]) + if response[1][0] and (response[1][0].get( + 'model', '') == 'product.template' or response[1][0].get( + 'model', '') == 'product.public.category'): + if not user: + mode = request.env['ir.config_parameter'].sudo().get_param( + 'filter_mode') + products = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.' + 'available_products_for_guest_ids', 'False')) + if mode == 'product_only': + available_products = request.env['product.template'].search( + [('id', 'in', products)]) + cat = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_' + 'visibility.available_cat_for_guest_ids', + 'False')) + available_categ = request.env['product.public.category'].search( + [('id', 'in', cat)]) + else: + partner = request.env['res.partner'].sudo().search( + [('id', '=', user.partner_id.id)]) + mode = partner.filter_mode + if mode == 'product_only': + available_products = self.available_products() + available_categ = partner.website_available_cat_ids + Category_avail = [] + Category = request.env['product.public.category'] + for ids in available_categ: + if not ids.parent_id.id in available_categ.ids: + Category_avail.append(ids.id) + if mode == 'product_only': + categ = Category.search([('parent_id', '=', False), ( + 'product_tmpl_ids', 'in', available_products.ids)]) + # supering shop*** + if not available_categ and not available_products and \ + request.env.user.has_group( + 'base.group_portal'): + mode = request.env['ir.config_parameter'].sudo().get_param( + 'filter_mode_portal') + products = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.' + 'available_products_for_portal_ids', 'False')) + if mode == 'product_only': + available_products = request.env['product.template'].search( + [('id', 'in', products)]) + cat = literal_eval( + request.env['ir.config_parameter'].sudo().get_param( + 'website_product_visibility.available_cat_' + 'for_portal_ids', + 'False')) + available_categ = request.env['product.public.category'].search( + [('id', 'in', cat)]) + if available_products: + product_category = available_products.mapped('public_categ_ids') + category = set(response[1][0]['results'].ids).intersection(set( + product_category.ids)) + products = set(response[1][-1]['results'].ids).intersection(set( + available_products.ids)) + response[1][-1]['results'] = request.env[ + 'product.template'].browse(products) + response[1][0]['results'] = request.env[ + 'product.public.category'].browse(category) + if available_categ: + categ_products = available_categ.mapped('product_tmpl_ids') + products = set(response[1][-1]['results'].ids).intersection(set( + categ_products.ids)) + category = set(response[1][0]['results'].ids).intersection(set( + available_categ.ids)) + response[1][0]['results'] = request.env[ + 'product.public.category'].browse(category) + response[1][-1]['results'] = request.env[ + 'product.template'].browse(products) + return tuple(response) + + def available_products(self): + """Returns the available product (product.template) ids""" + user = request.env['res.users'].sudo().search( + [('id', '=', request.env.user.id)]) + partner = request.env['res.partner'].sudo().search( + [('id', '=', user.partner_id.id)]) + return partner.website_available_product_ids diff --git a/product_visibility_website/static/description/assets/icons/capture (1).png b/product_visibility_website/static/description/assets/icons/capture (1).png new file mode 100644 index 000000000..8824deafc Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/capture (1).png differ diff --git a/product_visibility_website/static/description/assets/icons/check.png b/product_visibility_website/static/description/assets/icons/check.png new file mode 100644 index 000000000..c8e85f51d Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/check.png differ diff --git a/product_visibility_website/static/description/assets/icons/chevron.png b/product_visibility_website/static/description/assets/icons/chevron.png new file mode 100644 index 000000000..2089293d6 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/chevron.png differ diff --git a/product_visibility_website/static/description/assets/icons/cogs.png b/product_visibility_website/static/description/assets/icons/cogs.png new file mode 100644 index 000000000..95d0bad62 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/cogs.png differ diff --git a/product_visibility_website/static/description/assets/icons/consultation.png b/product_visibility_website/static/description/assets/icons/consultation.png new file mode 100644 index 000000000..8319d4baa Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/consultation.png differ diff --git a/product_visibility_website/static/description/assets/icons/ecom-black.png b/product_visibility_website/static/description/assets/icons/ecom-black.png new file mode 100644 index 000000000..a9385ff13 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/ecom-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/education-black.png b/product_visibility_website/static/description/assets/icons/education-black.png new file mode 100644 index 000000000..3eb09b27b Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/education-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/hotel-black.png b/product_visibility_website/static/description/assets/icons/hotel-black.png new file mode 100644 index 000000000..130f613be Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/hotel-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/img.png b/product_visibility_website/static/description/assets/icons/img.png new file mode 100644 index 000000000..70197f477 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/img.png differ diff --git a/product_visibility_website/static/description/assets/icons/license.png b/product_visibility_website/static/description/assets/icons/license.png new file mode 100644 index 000000000..a5869797e Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/license.png differ diff --git a/product_visibility_website/static/description/assets/icons/lifebuoy.png b/product_visibility_website/static/description/assets/icons/lifebuoy.png new file mode 100644 index 000000000..658d56ccc Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/lifebuoy.png differ diff --git a/product_visibility_website/static/description/assets/icons/manufacturing-black.png b/product_visibility_website/static/description/assets/icons/manufacturing-black.png new file mode 100644 index 000000000..697eb0e9f Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/manufacturing-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/photo-capture.png b/product_visibility_website/static/description/assets/icons/photo-capture.png new file mode 100644 index 000000000..06c111758 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/photo-capture.png differ diff --git a/product_visibility_website/static/description/assets/icons/pos-black.png b/product_visibility_website/static/description/assets/icons/pos-black.png new file mode 100644 index 000000000..97c0f90c1 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/pos-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/puzzle.png b/product_visibility_website/static/description/assets/icons/puzzle.png new file mode 100644 index 000000000..65cf854e7 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/puzzle.png differ diff --git a/product_visibility_website/static/description/assets/icons/restaurant-black.png b/product_visibility_website/static/description/assets/icons/restaurant-black.png new file mode 100644 index 000000000..4a35eb939 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/restaurant-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/service-black.png b/product_visibility_website/static/description/assets/icons/service-black.png new file mode 100644 index 000000000..301ab51cb Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/service-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/trading-black.png b/product_visibility_website/static/description/assets/icons/trading-black.png new file mode 100644 index 000000000..9398ba2f1 Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/trading-black.png differ diff --git a/product_visibility_website/static/description/assets/icons/training.png b/product_visibility_website/static/description/assets/icons/training.png new file mode 100644 index 000000000..884ca024d Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/training.png differ diff --git a/product_visibility_website/static/description/assets/icons/update.png b/product_visibility_website/static/description/assets/icons/update.png new file mode 100644 index 000000000..ecbc5a01a Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/update.png differ diff --git a/product_visibility_website/static/description/assets/icons/user.png b/product_visibility_website/static/description/assets/icons/user.png new file mode 100644 index 000000000..6ffb23d9f Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/user.png differ diff --git a/product_visibility_website/static/description/assets/icons/wrench.png b/product_visibility_website/static/description/assets/icons/wrench.png new file mode 100644 index 000000000..6c04dea0f Binary files /dev/null and b/product_visibility_website/static/description/assets/icons/wrench.png differ diff --git a/product_visibility_website/static/description/assets/misc/Cybrosys R.png b/product_visibility_website/static/description/assets/misc/Cybrosys R.png new file mode 100644 index 000000000..da4058087 Binary files /dev/null and b/product_visibility_website/static/description/assets/misc/Cybrosys R.png differ diff --git a/product_visibility_website/static/description/assets/misc/email.svg b/product_visibility_website/static/description/assets/misc/email.svg new file mode 100644 index 000000000..15291cdc3 --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/email.svg @@ -0,0 +1,33 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/product_visibility_website/static/description/assets/misc/phone.svg b/product_visibility_website/static/description/assets/misc/phone.svg new file mode 100644 index 000000000..b7bd7f251 --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/phone.svg @@ -0,0 +1,3 @@ + + + diff --git a/product_visibility_website/static/description/assets/misc/star (1) 2.svg b/product_visibility_website/static/description/assets/misc/star (1) 2.svg new file mode 100644 index 000000000..5ae9f507a --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/star (1) 2.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/product_visibility_website/static/description/assets/misc/support (1) 1.svg b/product_visibility_website/static/description/assets/misc/support (1) 1.svg new file mode 100644 index 000000000..7d37a8f30 --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/support (1) 1.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/product_visibility_website/static/description/assets/misc/support-email.svg b/product_visibility_website/static/description/assets/misc/support-email.svg new file mode 100644 index 000000000..eb70370d6 --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/support-email.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/product_visibility_website/static/description/assets/misc/tick-mark.svg b/product_visibility_website/static/description/assets/misc/tick-mark.svg new file mode 100644 index 000000000..2dbb40187 --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/tick-mark.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/product_visibility_website/static/description/assets/misc/whatsapp 1.svg b/product_visibility_website/static/description/assets/misc/whatsapp 1.svg new file mode 100644 index 000000000..0bfaf8fc6 --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/whatsapp 1.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/product_visibility_website/static/description/assets/misc/whatsapp.svg b/product_visibility_website/static/description/assets/misc/whatsapp.svg new file mode 100644 index 000000000..b618aea1d --- /dev/null +++ b/product_visibility_website/static/description/assets/misc/whatsapp.svg @@ -0,0 +1,33 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/product_visibility_website/static/description/assets/modules/1.png b/product_visibility_website/static/description/assets/modules/1.png new file mode 100644 index 000000000..3edf4ade9 Binary files /dev/null and b/product_visibility_website/static/description/assets/modules/1.png differ diff --git a/product_visibility_website/static/description/assets/modules/2.png b/product_visibility_website/static/description/assets/modules/2.png new file mode 100644 index 000000000..58349e677 Binary files /dev/null and b/product_visibility_website/static/description/assets/modules/2.png differ diff --git a/product_visibility_website/static/description/assets/modules/3.png b/product_visibility_website/static/description/assets/modules/3.png new file mode 100644 index 000000000..d092b81de Binary files /dev/null and b/product_visibility_website/static/description/assets/modules/3.png differ diff --git a/product_visibility_website/static/description/assets/modules/4.png b/product_visibility_website/static/description/assets/modules/4.png new file mode 100644 index 000000000..81ba7eace Binary files /dev/null and b/product_visibility_website/static/description/assets/modules/4.png differ diff --git a/product_visibility_website/static/description/assets/modules/5.png b/product_visibility_website/static/description/assets/modules/5.png new file mode 100644 index 000000000..acf8e77cc Binary files /dev/null and b/product_visibility_website/static/description/assets/modules/5.png differ diff --git a/product_visibility_website/static/description/assets/modules/6.png b/product_visibility_website/static/description/assets/modules/6.png new file mode 100644 index 000000000..1b1742670 Binary files /dev/null and b/product_visibility_website/static/description/assets/modules/6.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/1.png b/product_visibility_website/static/description/assets/screenshots/1.png new file mode 100644 index 000000000..4a6b2d80e Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/1.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/10.png b/product_visibility_website/static/description/assets/screenshots/10.png new file mode 100644 index 000000000..c06f5064d Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/10.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/11.png b/product_visibility_website/static/description/assets/screenshots/11.png new file mode 100644 index 000000000..f8c981305 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/11.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/12.png b/product_visibility_website/static/description/assets/screenshots/12.png new file mode 100644 index 000000000..bc54a7777 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/12.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/13.png b/product_visibility_website/static/description/assets/screenshots/13.png new file mode 100644 index 000000000..02f662b3a Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/13.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/14.png b/product_visibility_website/static/description/assets/screenshots/14.png new file mode 100644 index 000000000..d75068445 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/14.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/15.png b/product_visibility_website/static/description/assets/screenshots/15.png new file mode 100644 index 000000000..4c59baf91 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/15.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/2.png b/product_visibility_website/static/description/assets/screenshots/2.png new file mode 100644 index 000000000..93e483ce7 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/2.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/3.png b/product_visibility_website/static/description/assets/screenshots/3.png new file mode 100644 index 000000000..67bc29536 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/3.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/4.png b/product_visibility_website/static/description/assets/screenshots/4.png new file mode 100644 index 000000000..ea0e4e5b0 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/4.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/5.png b/product_visibility_website/static/description/assets/screenshots/5.png new file mode 100644 index 000000000..9163325fb Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/5.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/6.png b/product_visibility_website/static/description/assets/screenshots/6.png new file mode 100644 index 000000000..38fe10e40 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/6.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/7.png b/product_visibility_website/static/description/assets/screenshots/7.png new file mode 100644 index 000000000..a2c25bb0b Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/7.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/8.png b/product_visibility_website/static/description/assets/screenshots/8.png new file mode 100644 index 000000000..ef59d4943 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/8.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/9.png b/product_visibility_website/static/description/assets/screenshots/9.png new file mode 100644 index 000000000..4b5c62d16 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/9.png differ diff --git a/product_visibility_website/static/description/assets/screenshots/hero.gif b/product_visibility_website/static/description/assets/screenshots/hero.gif new file mode 100644 index 000000000..1e9054200 Binary files /dev/null and b/product_visibility_website/static/description/assets/screenshots/hero.gif differ diff --git a/product_visibility_website/static/description/banner.png b/product_visibility_website/static/description/banner.png new file mode 100644 index 000000000..511d4d392 Binary files /dev/null and b/product_visibility_website/static/description/banner.png differ diff --git a/product_visibility_website/static/description/icon.png b/product_visibility_website/static/description/icon.png new file mode 100644 index 000000000..54ab3e8f2 Binary files /dev/null and b/product_visibility_website/static/description/icon.png differ diff --git a/product_visibility_website/static/description/index.html b/product_visibility_website/static/description/index.html new file mode 100644 index 000000000..cabca0806 --- /dev/null +++ b/product_visibility_website/static/description/index.html @@ -0,0 +1,981 @@ + + + + + + + Odoo App 3 Index + + + + + + + + +
+
+
+
+
+ +
+
+
+ Community +
+
+ Enterprise +
+
+
+
+
+
+

+ Website Product Visibility

+

+ Website Product Visibility for Logged in and Logged out + Users/Portal Users/Visitors +

+
+ +
+
+
+
+
+

+ Key Highlights +

+
+
+
+
+
+ +
+
+

+ Filter according to the logged users

+

Filter product + and product categories according to the logged + user. +

+
+
+
+
+
+
+ +
+
+

+ Filter according to the public users/portal + users/visitors.

+

Filter product + and product categories according to the public + users/portal users/visitors. +

+
+
+
+
+
+
+ +
+
+

+ Restrict user searches to available + product/category only.

+

Approval + User can only search product and category among + from the + available product/category. +

+
+
+
+
+
+
+ +
+
+
+
+
+ +
+
+

+ Set filter for a logged user

+

+ Setting the filtering mode for a logged + user.

+
+
+
+
+
+
+ +
+
+

+ Product wise filtering for website shop + (Logged in Users)

+

+ Product wise filtering mode for logged user + in website shop.

+
+
+
+
+
+
+ +
+
+

+ Product wise visibility for website shop + (Product wise)

+

+ Product visibility (Product wise) for the + logged user in website + shop.

+
+
+
+ +
+
+
+ +
+
+

+ Category wise filtering for website shop + (Logged in Users)

+

+ Category wise filtering mode for logged user + in website + shop.

+
+
+
+
+
+
+ +
+
+

+ Product visibility for website shop + (Category wise)

+

+ Product visibility (Category wise) for the + logged user in + website shop.

+
+ +
+
+
+
+
+ +
+
+

+ Product wise filtering for website shop + (Portal Users)

+

+ Product wise filtering mode for the portal + users in the website + shop.

+
+
+
+
+
+
+ +
+
+

+ Set filter for a portal user

+

+ Setting the filtering mode for a portal user + as No Filter

+
+
+
+
+
+
+ +
+
+

+ Product wise visibility for website shop + (Portal Users)

+

+ Product visibility (Product wise) for the + portal users in + website shop when no filter is set for + them.

+
+
+
+
+
+
+ +
+
+

+ Category wise filtering for website shop + (Portal Users)

+

+ Category wise filtering mode for the portal + users in the website + shop.

+
+
+
+
+
+
+ +
+
+

+ Product visibility for website shop portal + users (Category + wise)

+

+ Product visibility (Category wise) for the + portal users in + website shop when no filter is set for + them.

+
+
+
+
+
+
+ +
+
+

+ Product wise filtering for website shop + (Visitors)

+

+ Product wise filtering mode for the visitors + in the website + shop.

+
+
+
+
+
+
+ +
+
+

+ Product wise visibility for website shop + (Visitors)

+

+ Product visibility (Product wise) for the + visitors in website + shop.

+
+
+
+
+
+
+ +
+
+

+ Category wise filtering for website shop + (Visitors)

+

+ Category wise filtering mode for the + visitors in the website + shop.

+
+
+
+
+
+
+ +
+
+

+ Product visibility for website shop visitors + (Category + wise)

+

+ Product visibility (Category wise) for the + visitors in website + shop.

+
+
+
+
+
+
+ +
+
+

+ Search available product from the list

+

+ Search product from the available product + list.

+
+
+
+
+
+
+
    +
  • + Website Product + Visibility for Logged in and Logged out + Users/Portal Users/Visitors +
  • +
  • + Enables the + user to search products and product categories + only + from those available products and categories +
  • +
  • + Website Product + Visibility +
      +
    • This module helps you to make visible + only the filtered products and + product categories for a logged in and + logged out users/portal + users/visitors. +
    • +
    +
  • + +
+
+
+
+
+
+
Version + 17.0.1.0.0|Released on:18th December 2023 +
+

+ + Initial Commit for Website Product + Visibility

+
+
+
+
+
+
+
+

+ Related Products

+
+
+ +
+
+

+ Our Services

+ +
+
+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Customization

+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Implementation

+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Support

+
+
+
+
+
+
+ service-icon +
+
+

Hire + Odoo Developer

+
+
+
+
+ +
+
+ service-icon +
+
+

Odoo + Integration

+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Migration

+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Consultancy

+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Implementation

+
+
+
+
+
+
+ service-icon +
+
+

Odoo + Licensing Consultancy

+
+
+
+
+
+
+

+ Our Industries

+ +
+
+
+
+
+
+ +

Trading

+

Easily procure and sell your products

+
+
+
+
+ +

POS

+

Easy configuration and convivial experience

+
+
+
+
+ +

+ Education

+

A platform for educational management

+
+
+
+
+ +

+ Manufacturing

+

Plan, track and schedule your operations

+
+
+
+
+ +

E-commerce & + Website

+

Mobile friendly, awe-inspiring product pages

+
+
+
+
+ +

Service + Management

+

Keep track of services and invoice

+
+
+
+
+ +

+ Restaurant

+

Run your bar or restaurant methodically

+
+
+
+
+ +

Hotel + Management

+

An all-inclusive hotel management application

+
+
+
+
+
+
+

+ Support

+
+
+
+
+
+
+
+ +
+ Need + Help? +

Got + questions or need help? Get in touch.

+
odoo@cybrosys.com +
+
+
+
+
+
+
+
+ +
+ WhatsApp +

Say hi to + us on WhatsApp!

+
+91 + 99456767686 +
+
+
+
+
+
+
+
+
+ + + + + + diff --git a/product_visibility_website/views/res_config_settings_views.xml b/product_visibility_website/views/res_config_settings_views.xml new file mode 100644 index 000000000..8d7ab2b90 --- /dev/null +++ b/product_visibility_website/views/res_config_settings_views.xml @@ -0,0 +1,87 @@ + + + + + + res.config.settings.view.form.inherit.product.visibility.website + + res.config.settings + + + +

Product Visibility

+
+
+
+ +
+
+
+
+
+
+
+
+ +
+
+
+
+
+
+
+
+
diff --git a/product_visibility_website/views/res_partner_views.xml b/product_visibility_website/views/res_partner_views.xml new file mode 100644 index 000000000..08a2a4c26 --- /dev/null +++ b/product_visibility_website/views/res_partner_views.xml @@ -0,0 +1,34 @@ + + + + + + + res.partner.view.form.inherit.product.visibility.website + + res.partner + + + + + + + + + + + + + + + + + + +