# -*- coding: utf-8 -*- ############################################################################# # # Cybrosys Technologies Pvt. Ltd. # # Copyright (C) 2024-TODAY Cybrosys Technologies() # Author: Cybrosys Techno Solutions() # # 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 . # ############################################################################# from collections import defaultdict from datetime import datetime from itertools import product as cartesian_product from werkzeug.exceptions import NotFound from odoo import http, tools from odoo.http import request from odoo.osv import expression from odoo.tools import lazy from odoo.addons.website.controllers.main import QueryURL from odoo.addons.website.models.ir_http import sitemap_qs2dom class TableCompute(object): """This is used to compute the table""" def __init__(self): self.table = {} def _check_place(self, posx, posy, sizex, sizey, ppr): """Checks the position """ res = True for y in range(sizey): for x in range(sizex): if posx + x >= ppr: res = False break row = self.table.setdefault(posy + y, {}) if row.setdefault(posx + x) is not None: res = False break for x in range(ppr): self.table[posy + y].setdefault(x, None) return res def process(self, products, ppg=20, ppr=4): """Compute products positions on the grid""" minpos = 0 index = 0 maxy = 0 x = 0 for p in products: x = min(max(p.website_size_x, 1), ppr) y = min(max(p.website_size_y, 1), ppr) if index >= ppg: x = y = 1 pos = minpos while not self._check_place(pos % ppr, pos // ppr, x, y, ppr): pos += 1 # if 21st products (index 20) and the last line is full , break # (pos + 1.0) / ppr is the line where the product would be inserted # maxy is the number of existing lines # + 1.0 is because pos begins at 0, # thus pos 20 is actually the 21st block # and to force python to not round the division operation if index >= ppg and ((pos + 1.0) // ppr) > maxy: break if x == 1 and y == 1: # simple heuristic for CPU optimization minpos = pos // ppr for y2 in range(y): for x2 in range(x): self.table[(pos // ppr) + y2][(pos % ppr) + x2] = False self.table[pos // ppr][pos % ppr] = { 'product': p, 'x': x, 'y': y, 'ribbon': p.sudo().website_ribbon_id, } if index <= ppg: maxy = max(maxy, y + (pos // ppr)) index += 1 # Format table according to HTML needs rows = sorted(self.table.items()) rows = [r[1] for r in rows] for col in range(len(rows)): cols = sorted(rows[col].items()) x += len(cols) rows[col] = [r[1] for r in cols if r[1]] return rows class ThemeCoffeeMenu(http.Controller): """ Controller for rendering datas to menu page """ # def _get_search_order(self, post): """ OrderBy will be parsed in orm and so no direct sql injection id is added to be sure that order is a unique sort key """ order = post.get('order') or request.env[ 'website'].get_current_website().shop_default_sort return 'is_published desc, %s, id desc' % order def _get_search_domain(self, search, category, attrib_values, search_in_description=True): """Function for getting search domain""" 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([('website_description', 'ilike', srch)]) subdomains.append([('description_sale', 'ilike', srch)]) domains.append(expression.OR(subdomains)) if category: domains.append([('public_categ_ids', 'child_of', int(category))]) if attrib_values: attrib = None ids = [] for value in attrib_values: if not attrib: attrib = value[0] ids.append(value[1]) elif value[0] == attrib: ids.append(value[1]) else: domains.append([ ('attribute_line_ids.value_ids', 'in', ids)]) attrib = value[0] ids = [value[1]] if attrib: domains.append([('attribute_line_ids.value_ids', 'in', ids)]) return expression.AND(domains) def sitemap_shop(env, rule, qs): """Sitemap for shop""" if not qs or qs.lower() in '/menu': yield {'loc': '/menu'} category = env['product.public.category'] dom = sitemap_qs2dom(qs, '/menu/category', category._rec_name) dom += env['website'].get_current_website().website_domain() for cat in category.search(dom): loc = '/menu/category/%s' % env['ir.http']._slug(cat) if not qs or qs.lower() in loc: yield {'loc': loc} def _get_search_options( self, category=None, attrib_values=None, pricelist=None, **post): """Function for returning search options""" return { 'displayDescription': True, 'displayDetail': True, 'displayExtraDetail': True, 'displayExtraLink': True, 'displayImage': True, 'allowFuzzy': not post.get('noFuzzy'), 'category': str(category.id) if category else None, 'attrib_values': attrib_values, 'display_currency': pricelist.currency_id, } def _shop_lookup_products(self, attrib_set, options, post, search, website): """ No limit because attributes are obtained from complete product list""" product_count, details, fuzzy_search_term = website._search_with_fuzzy( "products_only", search, limit=None, order=self._get_search_order(post), options=options ) search_result = details[0].get( 'results', request.env['product.template'] ).with_context(bin_size=True) if attrib_set: attribute_values = request.env[ 'product.attribute.value'].browse(attrib_set) values_per_attribute = defaultdict( lambda: request.env['product.attribute.value']) multi_value_attribute = False for value in attribute_values: values_per_attribute[value.attribute_id] |= value if len(values_per_attribute[value.attribute_id]) > 1: multi_value_attribute = True def filter_template(template, attribute_values_list): """Transform product.attribute.value to product.template.attribute.value """ attribute_value_to_ptav = {} for ptav in \ template.attribute_line_ids.product_template_value_ids: attribute_value_to_ptav[ ptav.product_attribute_value_id] = ptav.id possible_combinations = False for attribute_values in attribute_values_list: ptav_ids = [attribute_value_to_ptav[val] for val in attribute_values if val in attribute_value_to_ptav] ptavs = request.env[ 'product.template.attribute.value'].browse(ptav_ids) if len(ptavs) < len(attribute_values): continue if len(ptavs) == len(template.attribute_line_ids): if template._is_combination_possible(ptavs): return True elif len(ptavs) < len(template.attribute_line_ids): if len(attribute_values_list) == 1: possible_combinations = ( template._get_possible_combinations( necessary_values=ptavs)) if any(possible_combinations): return True if not possible_combinations: possible_combinations = ( template._get_possible_combinations()) for combination in possible_combinations: if ptavs.issubset(combination): return True return False if not multi_value_attribute: possible_attrib_values_list = [attribute_values] else: possible_attrib_values_list = [attribute_values] if not \ multi_value_attribute else \ [request.env['product.attribute.value'] .browse([rec.id for rec in values]) for values in cartesian_product(*values_per_attribute.values())] search_result = search_result.filtered( lambda tmpl: filter_template(possible_attrib_values_list) ) return fuzzy_search_term, product_count, search_result def _menu_get_query_url_kwargs(self, category, search, attrib=None, order=None): """Function for returning category, search, order and attribute""" return { 'category': category, 'search': search, 'attrib': attrib, 'order': order, } @http.route([ '/menu', '/menu/page/', '/menu/category/', '/menu/category//page/', ], type='http', auth="public", website=True, sitemap=sitemap_shop) def menu_page(self, page=0, category=None, search='', min_price=0.0, max_price=0.0, ppg=False, **post): """Function for rendering menu page""" category_id = request.env['product.public.category'] if category: category = category_id.search([('id', '=', int(category))], limit=1) if not category or not category.can_access_from_current_website(): raise NotFound() else: category = category_id website = request.env['website'].get_current_website() if ppg: try: ppg = int(ppg) post['ppg'] = ppg except ValueError: ppg = False if not ppg: ppg = website.shop_ppg or 20 ppr = website.shop_ppr or 4 attrib_list = request.httprequest.args.getlist('attrib') attrib_values = [[int(rec) for rec in res.split("-")] for res in attrib_list if res ] attributes_ids = {res[0] for res in attrib_values} attrib_set = {res[1] for res in attrib_values} keep = QueryURL('/menu', **self._menu_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: pricelist = website.pricelist_id 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) url = "/menu" 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, **post ) fuzzy_search_term, product_count, search_product = \ self._shop_lookup_products( attrib_set, options, post, search, website) website_domain = website.website_domain() categs_domain = [('parent_id', '=', False)] + website_domain if search: search_categories = category_id.search( [('product_tmpl_ids', 'in', search_product.ids) ] + website_domain ).parents_and_self categs_domain.append(('id', 'in', search_categories.ids)) else: search_categories = category_id categs = lazy(lambda: category_id.search(categs_domain)) if category: url = "/menu/category/%s" % request.env['ir.http']._slug(category) pager = website.pager(url=url, total=product_count, page=page, step=ppg, scope=7, url_args=post) offset = pager['offset'] products = search_product[offset:offset + ppg] product_attribute = request.env['product.attribute'] if products: attributes = lazy(lambda: product_attribute.search([ ('product_tmpl_ids', 'in', search_product.ids), ('visibility', '=', 'visible'), ])) else: attributes = lazy(lambda: product_attribute.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 # Try to fetch geoip based fpos or fallback on partner one products_prices = lazy( lambda: products._get_sales_prices(website)) values = { 'order': post.get('order', ''), 'category': category, 'attrib_values': attrib_values, 'attrib_set': attrib_set, 'pager': pager, 'pricelist': pricelist, 'add_qty': int(post.get('add_qty', 1)), 'products': products, 'search_product': search_product, 'search_count': product_count, 'bins': lazy(lambda: TableCompute().process( products, ppg, ppr)), 'ppg': ppg, 'ppr': ppr, 'categories': categs, 'attributes': attributes, 'keep': keep, 'search_categories_ids': search_categories.ids, 'products_prices': products_prices, 'get_product_prices': lambda product: lazy( lambda: products_prices[product.id] ), 'float_round': tools.float_round, } if category: values['main_object'] = category return request.render("theme_coffee_shop.coffee_menu", values) @http.route(['/menu/'], type='http', auth="public", website=True, sitemap=True) def product(self, product, category='', search='', **kwargs): """Function for rendering product page""" if not request.website.has_ecommerce_access(): return request.redirect('/web/login') return request.render("website_sale.product", self._prepare_product_values(product, category, search, **kwargs))