diff --git a/setup/website_sale_menu_top_user_selling/odoo/addons/website_sale_menu_top_user_selling b/setup/website_sale_menu_top_user_selling/odoo/addons/website_sale_menu_top_user_selling new file mode 120000 index 0000000000..f8d27bbae8 --- /dev/null +++ b/setup/website_sale_menu_top_user_selling/odoo/addons/website_sale_menu_top_user_selling @@ -0,0 +1 @@ +../../../../website_sale_menu_top_user_selling \ No newline at end of file diff --git a/setup/website_sale_menu_top_user_selling/setup.py b/setup/website_sale_menu_top_user_selling/setup.py new file mode 100644 index 0000000000..28c57bb640 --- /dev/null +++ b/setup/website_sale_menu_top_user_selling/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +) diff --git a/website_sale_menu_top_user_selling/README.rst b/website_sale_menu_top_user_selling/README.rst new file mode 100644 index 0000000000..fb45e81f2e --- /dev/null +++ b/website_sale_menu_top_user_selling/README.rst @@ -0,0 +1,99 @@ +================================== +Website Sale Menu Top User Selling +================================== + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:72c89f7b3b6be4bf5e26518bbcc2d2ddab1c32e8c6d51624dc39914ef20e6598 + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png + :target: https://odoo-community.org/page/development-status + :alt: Beta +.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png + :target: http://www.gnu.org/licenses/agpl-3.0-standalone.html + :alt: License: AGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fe--commerce-lightgray.png?logo=github + :target: https://github.com/OCA/e-commerce/tree/15.0/website_sale_menu_top_user_selling + :alt: OCA/e-commerce +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/e-commerce-15-0/e-commerce-15-0-website_sale_menu_top_user_selling + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runboat-Try%20me-875A7B.png + :target: https://runboat.odoo-community.org/builds?repo=OCA/e-commerce&target_branch=15.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module allows you to display the products most purchased by the user in the online +shop. + +**Table of contents** + +.. contents:: + :local: + +Configuration +============= + +The module allows you to configure the number of products to be displayed by means of a +configuration parameter. +Limit settings: +Go to Settings > Website > Products and adjust the value in the corresponding field to +set the maximum number of products to be displayed on the top selling products page. +The category panel must also be active in /shop. + +Usage +===== + +The best-selling products can be viewed by accessing the URL /shop/top_selling_products +on the website. +This URL can be accessed from */shop* by clicking on *Top selling products* in the +category menu. +The page will display the user's best-selling products, sorted by the number of +units sold. + +Bug Tracker +=========== + +Bugs are tracked on `GitHub Issues `_. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us to smash it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +~~~~~~~ + +* Tecnativa + +Contributors +~~~~~~~~~~~~ + +* `Tecnativa `_ + + * Pilar Vargas + +Maintainers +~~~~~~~~~~~ + +This module is maintained by the OCA. + +.. image:: https://odoo-community.org/logo.png + :alt: Odoo Community Association + :target: https://odoo-community.org + +OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use. + +This module is part of the `OCA/e-commerce `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/website_sale_menu_top_user_selling/__init__.py b/website_sale_menu_top_user_selling/__init__.py new file mode 100644 index 0000000000..91c5580fed --- /dev/null +++ b/website_sale_menu_top_user_selling/__init__.py @@ -0,0 +1,2 @@ +from . import controllers +from . import models diff --git a/website_sale_menu_top_user_selling/__manifest__.py b/website_sale_menu_top_user_selling/__manifest__.py new file mode 100644 index 0000000000..e73297e738 --- /dev/null +++ b/website_sale_menu_top_user_selling/__manifest__.py @@ -0,0 +1,18 @@ +# Copyright 2024 Tecnativa - Pilar Vargas +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). +{ + "name": "Website Sale Menu Top User Selling", + "summary": "Displays the products most sold by the user in the e-commerce.", + "version": "15.0.1.0.0", + "category": "Website", + "website": "https://github.com/OCA/e-commerce", + "author": "Tecnativa, Odoo Community Association (OCA)", + "license": "AGPL-3", + "application": False, + "installable": True, + "depends": ["website_sale"], + "data": ["views/res_config_settings_views.xml", "views/templates.xml"], + "web.assets_tests": [ + "website_sale_menu_top_user_selling/static/src/js/tours/*.js", + ], +} diff --git a/website_sale_menu_top_user_selling/controllers/__init__.py b/website_sale_menu_top_user_selling/controllers/__init__.py new file mode 100644 index 0000000000..12a7e529b6 --- /dev/null +++ b/website_sale_menu_top_user_selling/controllers/__init__.py @@ -0,0 +1 @@ +from . import main diff --git a/website_sale_menu_top_user_selling/controllers/main.py b/website_sale_menu_top_user_selling/controllers/main.py new file mode 100644 index 0000000000..a718937b79 --- /dev/null +++ b/website_sale_menu_top_user_selling/controllers/main.py @@ -0,0 +1,90 @@ +from odoo import http +from odoo.http import request + +from odoo.addons.website_sale.controllers.main import TableCompute, WebsiteSale + + +class WebsiteSale(WebsiteSale): + @http.route( + ["/shop/top_selling_products", "/shop/top_selling_products/page/"], + type="http", + auth="public", + website=True, + ) + def user_top_products(self, page=0, ppg=False, **kwargs): + if request.env.user.has_group("base.group_public"): + return request.redirect("/web/login") + param_limit = int( + request.env["ir.config_parameter"] + .sudo() + .get_param("website_sale_menu_top_user_selling.limit", 10) + ) + # Get best-selling products from the user + product_data = ( + request.env["sale.order.line"] + .sudo() + .read_group( + [("order_id.partner_id", "=", request.env.user.partner_id.id)], + ["product_id", "product_uom_qty:sum"], + ["product_id"], + orderby="product_uom_qty DESC", + ) + ) + top_product_ids = [ + rec["product_id"][0] for rec in product_data if rec["product_id"] + ] + # Search for templates of best-selling products + product_templates = ( + request.env["product.product"] + .sudo() + .search([("id", "in", top_product_ids)]) + ) + template_quantities = {} + for product in product_templates: + template_id = product.product_tmpl_id.id + if template_id not in template_quantities: + template_quantities[template_id] = 0 + template_quantities[template_id] += next( + rec["product_uom_qty"] + for rec in product_data + if rec["product_id"][0] == product.id + ) + # Sort the templates by total quantity sold and limit + sorted_template_ids = sorted( + template_quantities.keys(), + key=lambda tmpl_id: template_quantities[tmpl_id], + reverse=True, + ) + limited_template_ids = sorted_template_ids[:param_limit] + templates = request.env["product.template"].sudo().browse(limited_template_ids) + # Pagination + ppg = ppg or 20 + total_products = len(templates) + page_count = (total_products + ppg - 1) // ppg + page = max(0, min(page, page_count - 1)) + offset = page * ppg + products_on_page = templates[offset : offset + ppg] + pager = request.website.pager( + url="/shop/top_selling_products", + total=total_products, + page=page + 1, + step=ppg, + scope=5, + url_args=kwargs, + ) + # Shop context for the view + shop_context = self.shop(page=page, ppg=ppg, **kwargs) + shop_context.qcontext.update( + { + "pager": pager, + "products": products_on_page, + "search_product": products_on_page, + "search_count": total_products, + "bins": TableCompute().process( + products_on_page, + ppg, + request.env["website"].get_current_website().shop_ppr or 4, + ), + } + ) + return request.render("website_sale.products", shop_context.qcontext) diff --git a/website_sale_menu_top_user_selling/models/__init__.py b/website_sale_menu_top_user_selling/models/__init__.py new file mode 100644 index 0000000000..0deb68c468 --- /dev/null +++ b/website_sale_menu_top_user_selling/models/__init__.py @@ -0,0 +1 @@ +from . import res_config_settings diff --git a/website_sale_menu_top_user_selling/models/res_config_settings.py b/website_sale_menu_top_user_selling/models/res_config_settings.py new file mode 100644 index 0000000000..2b910eaf5f --- /dev/null +++ b/website_sale_menu_top_user_selling/models/res_config_settings.py @@ -0,0 +1,30 @@ +# Copyright 2024 Tecnativa - Pilar Vargas +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from odoo import api, fields, models + + +class ResConfigSettings(models.TransientModel): + + _inherit = "res.config.settings" + + top_selling_products_limit = fields.Integer(default=10) + + @api.model + def get_values(self): + res = super().get_values() + res.update( + top_selling_products_limit=int( + self.env["ir.config_parameter"] + .sudo() + .get_param("website_sale_menu_top_user_selling.limit", 20) + ), + ) + return res + + def set_values(self): + res = super().set_values() + self.env["ir.config_parameter"].sudo().set_param( + "website_sale_menu_top_user_selling.limit", self.top_selling_products_limit + ) + return res diff --git a/website_sale_menu_top_user_selling/readme/CONFIGURE.rst b/website_sale_menu_top_user_selling/readme/CONFIGURE.rst new file mode 100644 index 0000000000..38f7c6f369 --- /dev/null +++ b/website_sale_menu_top_user_selling/readme/CONFIGURE.rst @@ -0,0 +1,6 @@ +The module allows you to configure the number of products to be displayed by means of a +configuration parameter. +Limit settings: +Go to Settings > Website > Products and adjust the value in the corresponding field to +set the maximum number of products to be displayed on the top selling products page. +The category panel must also be active in /shop. diff --git a/website_sale_menu_top_user_selling/readme/CONTRIBUTORS.rst b/website_sale_menu_top_user_selling/readme/CONTRIBUTORS.rst new file mode 100644 index 0000000000..7029b234a0 --- /dev/null +++ b/website_sale_menu_top_user_selling/readme/CONTRIBUTORS.rst @@ -0,0 +1,3 @@ +* `Tecnativa `_ + + * Pilar Vargas diff --git a/website_sale_menu_top_user_selling/readme/DESCRIPTION.rst b/website_sale_menu_top_user_selling/readme/DESCRIPTION.rst new file mode 100644 index 0000000000..1144e3b7b9 --- /dev/null +++ b/website_sale_menu_top_user_selling/readme/DESCRIPTION.rst @@ -0,0 +1,2 @@ +This module allows you to display the products most purchased by the user in the online +shop. diff --git a/website_sale_menu_top_user_selling/readme/USAGE.rst b/website_sale_menu_top_user_selling/readme/USAGE.rst new file mode 100644 index 0000000000..692d3954b3 --- /dev/null +++ b/website_sale_menu_top_user_selling/readme/USAGE.rst @@ -0,0 +1,6 @@ +The best-selling products can be viewed by accessing the URL /shop/top_selling_products +on the website. +This URL can be accessed from */shop* by clicking on *Top selling products* in the +category menu. +The page will display the user's best-selling products, sorted by the number of +units sold. diff --git a/website_sale_menu_top_user_selling/static/description/icon.png b/website_sale_menu_top_user_selling/static/description/icon.png new file mode 100644 index 0000000000..3a0328b516 Binary files /dev/null and b/website_sale_menu_top_user_selling/static/description/icon.png differ diff --git a/website_sale_menu_top_user_selling/static/description/index.html b/website_sale_menu_top_user_selling/static/description/index.html new file mode 100644 index 0000000000..ee66db8cb3 --- /dev/null +++ b/website_sale_menu_top_user_selling/static/description/index.html @@ -0,0 +1,447 @@ + + + + + +Website Sale Menu Top User Selling + + + +
+

Website Sale Menu Top User Selling

+ + +

Beta License: AGPL-3 OCA/e-commerce Translate me on Weblate Try me on Runboat

+

This module allows you to display the products most purchased by the user in the online +shop.

+

Table of contents

+ +
+

Configuration

+

The module allows you to configure the number of products to be displayed by means of a +configuration parameter. +Limit settings: +Go to Settings > Website > Products and adjust the value in the corresponding field to +set the maximum number of products to be displayed on the top selling products page. +The category panel must also be active in /shop.

+
+
+

Usage

+

The best-selling products can be viewed by accessing the URL /shop/top_selling_products +on the website. +This URL can be accessed from /shop by clicking on Top selling products in the +category menu. +The page will display the user’s best-selling products, sorted by the number of +units sold.

+
+
+

Bug Tracker

+

Bugs are tracked on GitHub Issues. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us to smash it by providing a detailed and welcomed +feedback.

+

Do not contact contributors directly about support or help with technical issues.

+
+
+

Credits

+
+

Authors

+
    +
  • Tecnativa
  • +
+
+
+

Contributors

+ +
+
+

Maintainers

+

This module is maintained by the OCA.

+ +Odoo Community Association + +

OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use.

+

This module is part of the OCA/e-commerce project on GitHub.

+

You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

+
+
+
+ + diff --git a/website_sale_menu_top_user_selling/static/src/js/tours/website_sale_menu_top_user_selling_tour.js b/website_sale_menu_top_user_selling/static/src/js/tours/website_sale_menu_top_user_selling_tour.js new file mode 100644 index 0000000000..8108b868ff --- /dev/null +++ b/website_sale_menu_top_user_selling/static/src/js/tours/website_sale_menu_top_user_selling_tour.js @@ -0,0 +1,27 @@ +/* Copyright 2024 Pilar Vargas + * License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). */ + +odoo.define("website_sale_menu_top_user_selling.tour", function (require) { + "use strict"; + + const tour = require("web_tour.tour"); + + var steps = [ + { + trigger: "a:contains('Product 5')", + extra_trigger: + "#products_grid:has(a:contains('Product 3')):not(:has(a:contains('Product 1'))):not(:has(a:contains('Product 2'))):not(:has(a:contains('Product 4')))", + }, + ]; + tour.register( + "website_sale_menu_top_user_selling", + { + url: "/shop/top_selling_products", + test: true, + }, + steps + ); + return { + steps: steps, + }; +}); diff --git a/website_sale_menu_top_user_selling/tests/__init__.py b/website_sale_menu_top_user_selling/tests/__init__.py new file mode 100644 index 0000000000..6dab214ac8 --- /dev/null +++ b/website_sale_menu_top_user_selling/tests/__init__.py @@ -0,0 +1 @@ +from . import test_ui diff --git a/website_sale_menu_top_user_selling/tests/test_ui.py b/website_sale_menu_top_user_selling/tests/test_ui.py new file mode 100644 index 0000000000..b5a12f3884 --- /dev/null +++ b/website_sale_menu_top_user_selling/tests/test_ui.py @@ -0,0 +1,60 @@ +# Copyright 2024 Tecnativa - Pilar Vargas +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). +from odoo.tests.common import HttpCase, tagged + + +@tagged("post_install", "-at_install") +class TestUi(HttpCase): + @classmethod + def setUpClass(cls): + super().setUpClass() + # Remove this variable in v16 and put instead: + # from odoo.addons.base.tests.common import DISABLED_MAIL_CONTEXT + DISABLED_MAIL_CONTEXT = { + "tracking_disable": True, + "mail_create_nolog": True, + "mail_create_nosubscribe": True, + "mail_notrack": True, + "no_reset_password": True, + } + cls.env = cls.env(context=dict(cls.env.context, **DISABLED_MAIL_CONTEXT)) + cls.env.ref("website_sale.products_categories").active = True + cls.env["ir.config_parameter"].sudo().set_param( + "website_sale_menu_top_user_selling.limit", 2 + ) + cls.admin_user = cls.env.ref("base.user_admin") + cls.partner = cls.admin_user.partner_id + cls.products = [] + for i in range(1, 6): + product = cls.env["product.product"].create( + { + "name": f"Product {i}", + "list_price": 10.0 * i, + } + ) + cls.products.append(product) + cls.sale_order = cls.env["sale.order"].create( + { + "partner_id": cls.partner.id, + } + ) + quantities = [50, 30, 70, 10, 90] + for product, qty in zip(cls.products, quantities): + cls.env["sale.order.line"].create( + { + "order_id": cls.sale_order.id, + "product_id": product.id, + "product_uom_qty": qty, + "price_unit": product.list_price, + } + ) + cls.sale_order.action_confirm() + + def test_ui(self): + """Test frontend tour.""" + # import wdb; wdb.set_trace() + self.start_tour( + "/shop/top_selling_products", + "website_sale_menu_top_user_selling", + login="admin", + ) diff --git a/website_sale_menu_top_user_selling/views/res_config_settings_views.xml b/website_sale_menu_top_user_selling/views/res_config_settings_views.xml new file mode 100644 index 0000000000..1454cc2a33 --- /dev/null +++ b/website_sale_menu_top_user_selling/views/res_config_settings_views.xml @@ -0,0 +1,33 @@ + + + + + res.config.settings + + + +
+
+
+
+
+ + + + diff --git a/website_sale_menu_top_user_selling/views/templates.xml b/website_sale_menu_top_user_selling/views/templates.xml new file mode 100644 index 0000000000..726e4db274 --- /dev/null +++ b/website_sale_menu_top_user_selling/views/templates.xml @@ -0,0 +1,35 @@ + + + +