diff --git a/product_main_seller/README.md b/product_main_seller/README.md new file mode 100644 index 0000000..b58dfea --- /dev/null +++ b/product_main_seller/README.md @@ -0,0 +1,100 @@ +# Product Main Seller + +## Summary + +Adds a "Main Vendor" field to products that automatically tracks the primary supplier based on supplierinfo sequence. + +## Features + +- **Automatic Main Vendor**: Computed field showing the primary supplier +- **Based on Sequence**: Uses the supplierinfo with the lowest sequence number +- **Visible in Views**: Shows in product form and tree views +- **Searchable**: Can search and filter products by main vendor +- **Related Fields**: Access to vendor name and reference + +## Technical Details + +### Models Extended + +- `product.template`: Main product template +- `product.product`: Product variants + +### Fields Added + +- `main_seller_id` (Many2one → res.partner, computed, stored) + - The main vendor for this product + - Computed from `seller_ids` with lowest sequence + - Stored for performance + - Searchable and filterable + +### Computation Logic + +The main vendor is determined by: +1. Looking at all supplierinfo records (`seller_ids`) +2. Filtering for valid suppliers (active partners) +3. Selecting the one with the **lowest sequence** number +4. If no suppliers, returns empty + +## Dependencies + +- `purchase` (Odoo core) + +## Installation + +```bash +docker-compose exec odoo odoo -d odoo -u product_main_seller --stop-after-init +``` + +## Usage + +## Usage + +### Viewing Main Vendor + +1. Open a product form (Products > Products > [Product]) +2. See "Main Vendor" field (usually in Purchase tab) +3. Field is automatically computed from vendor list + +### Changing Main Vendor + +To change the main vendor: +1. Go to product form > Purchase tab +2. Edit the vendor list (`seller_ids`) +3. Change the sequence numbers (lower = more priority) +4. Save - the "Main Vendor" will update automatically + +### Searching by Vendor + +```python +# Find all products from a specific vendor +products = self.env['product.template'].search([ + ('main_seller_id', '=', vendor_id) +]) +``` + +## Use Cases in Kidekoop + +This module is critical for: +- Vendor performance analysis +- Purchase order management +- Inventory planning by supplier +- Default supplier selection in purchase workflows + +## Views Modified + +- Product Template Form View +- Product Template Tree View +- Product Variant Form View + +## Translations + +- ✅ Spanish (es) +- ✅ Euskera (eu) + +Located in `i18n/` directory. + +--- + +**Version**: 18.0.1.0.0 +**Category**: Purchase +**License**: AGPL-3 diff --git a/product_main_seller/README.rst b/product_main_seller/README.rst new file mode 100644 index 0000000..22ed7f8 --- /dev/null +++ b/product_main_seller/README.rst @@ -0,0 +1,97 @@ +.. image:: https://odoo-community.org/readme-banner-image + :target: https://odoo-community.org/get-involved?utm_source=readme + :alt: Odoo Community Association + +=================== +Product Main Vendor +=================== + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:c353b8931ca2140d6d80974f00d4e5e073b737283dc2770fe718a8842cd6bd4e + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |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/license-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%2Fpurchase--workflow-lightgray.png?logo=github + :target: https://github.com/OCA/purchase-workflow/tree/18.0/product_main_seller + :alt: OCA/purchase-workflow +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/purchase-workflow-18-0/purchase-workflow-18-0-product_main_seller + :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/purchase-workflow&target_branch=18.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module extends the Odoo Product module to compute and display the +main Vendor of each products. The main vendor is the first vendor in the +vendors list. + +|image1| + +.. |image1| image:: https://raw.githubusercontent.com/OCA/purchase-workflow/18.0/product_main_seller/static/description/product_tree_view.png + +**Table of contents** + +.. contents:: + :local: + +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 +------- + +* GRAP + +Contributors +------------ + +- Quentin Dupont (quentin.dupont@grap.coop) + +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. + +.. |maintainer-legalsylvain| image:: https://github.com/legalsylvain.png?size=40px + :target: https://github.com/legalsylvain + :alt: legalsylvain +.. |maintainer-quentinDupont| image:: https://github.com/quentinDupont.png?size=40px + :target: https://github.com/quentinDupont + :alt: quentinDupont + +Current `maintainers `__: + +|maintainer-legalsylvain| |maintainer-quentinDupont| + +This module is part of the `OCA/purchase-workflow `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/product_main_seller/__init__.py b/product_main_seller/__init__.py new file mode 100644 index 0000000..6d58305 --- /dev/null +++ b/product_main_seller/__init__.py @@ -0,0 +1,2 @@ +from . import models +from .hooks import pre_init_hook diff --git a/product_main_seller/__manifest__.py b/product_main_seller/__manifest__.py new file mode 100644 index 0000000..76c768f --- /dev/null +++ b/product_main_seller/__manifest__.py @@ -0,0 +1,21 @@ +# Copyright (C) 2022 - Today: GRAP (http://www.grap.coop) +# @author: Quentin Dupont (quentin.dupont@grap.coop) +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + +{ + "name": "Product Main Vendor", + "summary": "Main Vendor for a product", + "version": "18.0.1.0.0", + "category": "Purchase", + "author": "GRAP,Odoo Community Association (OCA)", + "website": "https://github.com/OCA/purchase-workflow", + "license": "AGPL-3", + "depends": ["purchase"], + "maintainers": ["legalsylvain", "quentinDupont"], + "data": [ + "views/view_product_product.xml", + "views/view_product_template.xml", + ], + "installable": True, + "pre_init_hook": "pre_init_hook", +} diff --git a/product_main_seller/hooks.py b/product_main_seller/hooks.py new file mode 100644 index 0000000..eceb699 --- /dev/null +++ b/product_main_seller/hooks.py @@ -0,0 +1,34 @@ +# Copyright 2024-Today - Sylvain Le GAL (GRAP) +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +import logging + +_logger = logging.getLogger(__name__) + + +def pre_init_hook(env): + _logger.info("Initializing column main_seller_id on table product_template") + cr = env.cr + cr.execute(""" + ALTER TABLE product_template + ADD COLUMN IF NOT EXISTS main_seller_id integer; + """) + cr.execute(""" + WITH numbered_supplierinfos as ( + SELECT *, ROW_number() over ( + partition BY product_tmpl_id + ORDER BY sequence, min_qty desc, price + ) as row_number + FROM product_supplierinfo + ), + + first_supplierinfos as ( + SELECT * from numbered_supplierinfos + WHERE row_number = 1 + ) + + UPDATE product_template pt + SET main_seller_id = first_supplierinfos.partner_id + FROM first_supplierinfos + WHERE pt.id = first_supplierinfos.product_tmpl_id; + """) diff --git a/product_main_seller/i18n/fr.po b/product_main_seller/i18n/fr.po new file mode 100644 index 0000000..f245722 --- /dev/null +++ b/product_main_seller/i18n/fr.po @@ -0,0 +1,37 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * product_main_seller +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 16.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2024-07-08 20:26+0000\n" +"PO-Revision-Date: 2024-07-08 20:26+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: product_main_seller +#: model:ir.model.fields,field_description:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,field_description:product_main_seller.field_product_template__main_seller_id +#: model_terms:ir.ui.view,arch_db:product_main_seller.view_product_template_search +msgid "Main Vendor" +msgstr "Fournisseur principal" + +#. module: product_main_seller +#: model:ir.model,name:product_main_seller.model_product_template +msgid "Product" +msgstr "Produit" + +#. module: product_main_seller +#: model:ir.model.fields,help:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,help:product_main_seller.field_product_template__main_seller_id +msgid "Put your supplier info in first position to set as main vendor" +msgstr "" +"Définir une information fournisseur en première position pour le définir " +"comme fournisseur principal" diff --git a/product_main_seller/i18n/it.po b/product_main_seller/i18n/it.po new file mode 100644 index 0000000..2bf8009 --- /dev/null +++ b/product_main_seller/i18n/it.po @@ -0,0 +1,37 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * product_main_seller +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 16.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2024-09-06 15:06+0000\n" +"Last-Translator: mymage \n" +"Language-Team: none\n" +"Language: it\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=2; plural=n != 1;\n" +"X-Generator: Weblate 5.6.2\n" + +#. module: product_main_seller +#: model:ir.model.fields,field_description:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,field_description:product_main_seller.field_product_template__main_seller_id +#: model_terms:ir.ui.view,arch_db:product_main_seller.view_product_template_search +msgid "Main Vendor" +msgstr "Fornitore principale" + +#. module: product_main_seller +#: model:ir.model,name:product_main_seller.model_product_template +msgid "Product" +msgstr "Prodotto" + +#. module: product_main_seller +#: model:ir.model.fields,help:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,help:product_main_seller.field_product_template__main_seller_id +msgid "Put your supplier info in first position to set as main vendor" +msgstr "" +"Inserire le informazioni fornitore nella prima posizione per impostarlo come " +"fornitore principale" diff --git a/product_main_seller/i18n/nl.po b/product_main_seller/i18n/nl.po new file mode 100644 index 0000000..1d03c36 --- /dev/null +++ b/product_main_seller/i18n/nl.po @@ -0,0 +1,38 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * product_main_seller +# bosd , 2025. +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 16.0\n" +"Report-Msgid-Bugs-To: \n" +"Last-Translator: bosd \n" +"Language-Team: Dutch\n" +"Language: nl\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"PO-Revision-Date: 2025-04-18 13:34+0200\n" +"X-Generator: Gtranslator 47.1\n" + +#. module: product_main_seller +#: model:ir.model.fields,field_description:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,field_description:product_main_seller.field_product_template__main_seller_id +#: model_terms:ir.ui.view,arch_db:product_main_seller.view_product_template_search +msgid "Main Vendor" +msgstr "Hoofdleverancier" + +#. module: product_main_seller +#: model:ir.model,name:product_main_seller.model_product_template +msgid "Product" +msgstr "Product" + +#. module: product_main_seller +#: model:ir.model.fields,help:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,help:product_main_seller.field_product_template__main_seller_id +msgid "Put your supplier info in first position to set as main vendor" +msgstr "" +"Zet uw leverancier op de eerste plaats om deze als hoofdleverancier in te " +"stellen" diff --git a/product_main_seller/i18n/product_main_seller.pot b/product_main_seller/i18n/product_main_seller.pot new file mode 100644 index 0000000..f425184 --- /dev/null +++ b/product_main_seller/i18n/product_main_seller.pot @@ -0,0 +1,32 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * product_main_seller +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: product_main_seller +#: model:ir.model.fields,field_description:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,field_description:product_main_seller.field_product_template__main_seller_id +#: model_terms:ir.ui.view,arch_db:product_main_seller.view_product_template_search +msgid "Main Vendor" +msgstr "" + +#. module: product_main_seller +#: model:ir.model,name:product_main_seller.model_product_template +msgid "Product" +msgstr "" + +#. module: product_main_seller +#: model:ir.model.fields,help:product_main_seller.field_product_product__main_seller_id +#: model:ir.model.fields,help:product_main_seller.field_product_template__main_seller_id +msgid "Put your supplier info in first position to set as main vendor" +msgstr "" diff --git a/product_main_seller/models/__init__.py b/product_main_seller/models/__init__.py new file mode 100644 index 0000000..e8fa8f6 --- /dev/null +++ b/product_main_seller/models/__init__.py @@ -0,0 +1 @@ +from . import product_template diff --git a/product_main_seller/models/product_template.py b/product_main_seller/models/product_template.py new file mode 100644 index 0000000..f4d35ee --- /dev/null +++ b/product_main_seller/models/product_template.py @@ -0,0 +1,31 @@ +# Copyright (C) 2022 - Today: GRAP (http://www.grap.coop) +# @author: Quentin DUPONT (quentin.dupont@grap.coop) +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + +from odoo import api +from odoo import fields +from odoo import models + + +class ProductTemplate(models.Model): + _inherit = "product.template" + + main_seller_id = fields.Many2one( + comodel_name="res.partner", + string="Main Vendor", + help="Put your supplier info in first position to set as main vendor", + compute="_compute_main_seller_id", + store=True, + ) + + @api.depends("variant_seller_ids.sequence", "variant_seller_ids.partner_id.active") + def _compute_main_seller_id(self): + for template in self: + if template.variant_seller_ids: + template.main_seller_id = fields.first( + template.variant_seller_ids.filtered( + lambda seller: seller.partner_id.active + ) + ).partner_id + else: + template.main_seller_id = False diff --git a/product_main_seller/pyproject.toml b/product_main_seller/pyproject.toml new file mode 100644 index 0000000..4231d0c --- /dev/null +++ b/product_main_seller/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/product_main_seller/readme/CONTRIBUTORS.md b/product_main_seller/readme/CONTRIBUTORS.md new file mode 100644 index 0000000..65c3000 --- /dev/null +++ b/product_main_seller/readme/CONTRIBUTORS.md @@ -0,0 +1 @@ +- Quentin Dupont () diff --git a/product_main_seller/readme/DESCRIPTION.md b/product_main_seller/readme/DESCRIPTION.md new file mode 100644 index 0000000..20a1cce --- /dev/null +++ b/product_main_seller/readme/DESCRIPTION.md @@ -0,0 +1,5 @@ +This module extends the Odoo Product module to compute and display the +main Vendor of each products. The main vendor is the first vendor in the +vendors list. + +![](../static/description/product_tree_view.png) diff --git a/product_main_seller/static/description/icon.png b/product_main_seller/static/description/icon.png new file mode 100644 index 0000000..2dcd8fd Binary files /dev/null and b/product_main_seller/static/description/icon.png differ diff --git a/product_main_seller/static/description/index.html b/product_main_seller/static/description/index.html new file mode 100644 index 0000000..8c4e691 --- /dev/null +++ b/product_main_seller/static/description/index.html @@ -0,0 +1,434 @@ + + + + + +README.rst + + + +
+ + + +Odoo Community Association + +
+

Product Main Vendor

+ +

Beta License: AGPL-3 OCA/purchase-workflow Translate me on Weblate Try me on Runboat

+

This module extends the Odoo Product module to compute and display the +main Vendor of each products. The main vendor is the first vendor in the +vendors list.

+

image1

+

Table of contents

+ +
+

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

+
    +
  • GRAP
  • +
+
+
+

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.

+

Current maintainers:

+

legalsylvain quentinDupont

+

This module is part of the OCA/purchase-workflow project on GitHub.

+

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

+
+
+
+
+ + diff --git a/product_main_seller/static/description/product_tree_view.png b/product_main_seller/static/description/product_tree_view.png new file mode 100644 index 0000000..3c770c8 Binary files /dev/null and b/product_main_seller/static/description/product_tree_view.png differ diff --git a/product_main_seller/tests/__init__.py b/product_main_seller/tests/__init__.py new file mode 100644 index 0000000..d412bad --- /dev/null +++ b/product_main_seller/tests/__init__.py @@ -0,0 +1 @@ +from . import test_seller diff --git a/product_main_seller/tests/test_seller.py b/product_main_seller/tests/test_seller.py new file mode 100644 index 0000000..9ab5958 --- /dev/null +++ b/product_main_seller/tests/test_seller.py @@ -0,0 +1,72 @@ +# Copyright (C) 2022 - Today: GRAP (http://www.grap.coop) +# @author: Quentin DUPONT (quentin.dupont@grap.coop) +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + +from odoo import Command +from odoo.tests.common import TransactionCase + + +class TestSeller(TransactionCase): + @classmethod + def setUpClass(cls): + super().setUpClass() + cls.product_workplace = cls.env.ref("product.product_product_24") + cls.product_acoustic = cls.env.ref("product.product_product_25") + cls.product_with_var_chair = cls.env.ref("product.product_product_11") + cls.product_without_seller_desk = cls.env.ref("product.product_product_3") + + cls.partner_woodcorner = cls.env.ref("base.res_partner_1") + cls.partner_azure = cls.env.ref("base.res_partner_12") + + def test_01_computed_main_vendor(self): + self.assertEqual( + self.product_acoustic.main_seller_id, + self.product_acoustic.seller_ids[0].partner_id, + ) + self.assertEqual( + self.product_with_var_chair.main_seller_id, + self.product_acoustic.product_variant_ids[0] + .variant_seller_ids[0] + .partner_id, + ) + + def test_02_replace_supplierinfo(self): + self.product_acoustic.seller_ids = [ + Command.clear(), + Command.create({"partner_id": self.partner_azure.id}), + ] + self.assertEqual(self.product_acoustic.main_seller_id.id, self.partner_azure.id) + + def test_03_add_supplierinfo_no_existing_supplierinfo(self): + self.product_without_seller_desk.seller_ids = [ + Command.create({"partner_id": self.partner_azure.id}), + ] + self.assertEqual( + self.product_without_seller_desk.main_seller_id.id, self.partner_azure.id + ) + + def test_03_add_supplierinfo_low_sequence(self): + self.product_workplace.seller_ids.write({"sequence": 1}) + self.product_workplace.seller_ids = [ + Command.create({"sequence": 100, "partner_id": self.partner_azure.id}), + ] + self.assertNotEqual( + self.product_workplace.main_seller_id.id, self.partner_azure.id + ) + + def test_03_add_supplierinfo_high_sequence(self): + self.product_workplace.seller_ids.write({"sequence": 1000}) + self.product_workplace.seller_ids = [ + Command.create({"sequence": 100, "partner_id": self.partner_azure.id}), + ] + self.assertEqual( + self.product_workplace.main_seller_id.id, self.partner_azure.id + ) + + def test_04_update_supplierinfo(self): + self.product_acoustic.seller_ids.write({"partner_id": self.partner_azure.id}) + self.assertEqual(self.product_acoustic.main_seller_id.id, self.partner_azure.id) + + def test_05_unlink_supplierinfo(self): + self.product_acoustic.seller_ids.unlink() + self.assertEqual(self.product_acoustic.main_seller_id.id, False) diff --git a/product_main_seller/views/view_product_product.xml b/product_main_seller/views/view_product_product.xml new file mode 100644 index 0000000..30ee16d --- /dev/null +++ b/product_main_seller/views/view_product_product.xml @@ -0,0 +1,17 @@ + + + + + product.product + + + + + + + + diff --git a/product_main_seller/views/view_product_template.xml b/product_main_seller/views/view_product_template.xml new file mode 100644 index 0000000..e9251c1 --- /dev/null +++ b/product_main_seller/views/view_product_template.xml @@ -0,0 +1,34 @@ + + + + + product.template + + + + + + + + + + + + + product.template + + + + + + + + diff --git a/website_sale_aplicoop/__manifest__.py b/website_sale_aplicoop/__manifest__.py index 296b487..69bf0ba 100644 --- a/website_sale_aplicoop/__manifest__.py +++ b/website_sale_aplicoop/__manifest__.py @@ -17,7 +17,6 @@ "stock", "account", "product_get_price_helper", - "product_origin_char", ], "data": [ # Datos: Grupos propios